mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-07 19:30:12 +00:00
7115ccd3b0
The current approach of building the introspection files for GTK works, but is often cumbersome as one needs to set many environmental variables before launching a solution file, which runs a Windows batch script to generate the .gir/.typelib files. It was also possible to hand-run the batch script from the Visual Studio command prompt, but even more environmental variables need to be set. This changes the approach to build the introspection files using an NMake Makefile (but elimating from the Visual Studio Project Files the part to build the introspection files) to: -Make it clearer to the person building the introspection files what environmental variables are needed, specifically for PKG_CONFIG_PATH and MINGWDIR and CFG (formerly CONF). Setting stuff like VSVER, PLAT and BASEDIR is no longer required, which was a bit clunky. -Allows some more easier flexibility on the build of the intropsection files.
262 lines
9.9 KiB
Python
262 lines
9.9 KiB
Python
#!/usr/bin/python
|
|
# vim: encoding=utf-8
|
|
#expand *.in files
|
|
import os
|
|
import sys
|
|
import re
|
|
import optparse
|
|
|
|
def parent_dir(path):
|
|
if not os.path.isabs(path):
|
|
path = os.path.abspath(path)
|
|
if os.path.isfile(path):
|
|
path = os.path.dirname(path)
|
|
return os.path.split(path)[0]
|
|
|
|
def check_output_type (btype):
|
|
print_bad_type = False
|
|
output_type = -1
|
|
if (btype is None):
|
|
output_type = -1
|
|
print_bad_type = False
|
|
elif (btype == "vs9"):
|
|
output_type = 1
|
|
elif (btype == "vs10"):
|
|
output_type = 2
|
|
elif (btype == "nmake-exe"):
|
|
output_type = 3
|
|
else:
|
|
output_type = -1
|
|
print_bad_type = True
|
|
if (output_type == -1):
|
|
if (print_bad_type is True):
|
|
print ("The entered output build file type '%s' is not valid" % btype)
|
|
else:
|
|
print ("Output build file type is not specified.\nUse -t <type> to specify the output build file type.")
|
|
print ("Valid output build file types are: nmake-exe, vs9 , vs10")
|
|
return output_type
|
|
|
|
def read_vars_from_AM(path, vars = {}, conds = {}, filters = None):
|
|
'''
|
|
path: path to the Makefile.am
|
|
vars: predefined variables
|
|
conds: condition variables for Makefile
|
|
filters: if None, all variables defined are returned,
|
|
otherwise, it is a list contains that variables should be returned
|
|
'''
|
|
cur_vars = vars.copy()
|
|
RE_AM_VAR_REF = re.compile(r'\$\((\w+?)\)')
|
|
RE_AM_VAR = re.compile(r'^\s*(\w+)\s*=(.*)$')
|
|
RE_AM_INCLUDE = re.compile(r'^\s*include\s+(\w+)')
|
|
RE_AM_VAR_ADD = re.compile(r'^\s*(\w+)\s*\+=(.*)$')
|
|
RE_AM_CONTINUING = re.compile(r'\\\s*$')
|
|
RE_AM_IF = re.compile(r'^\s*if\s+(\w+)')
|
|
RE_AM_IFNOT = re.compile(r'^\s*if\s!+(\w+)')
|
|
RE_AM_ELSE = re.compile(r'^\s*else')
|
|
RE_AM_ENDIF = re.compile(r'^\s*endif')
|
|
def am_eval(cont):
|
|
return RE_AM_VAR_REF.sub(lambda x: cur_vars.get(x.group(1), ''), cont)
|
|
with open(path, 'r') as f:
|
|
contents = f.readlines()
|
|
#combine continuing lines
|
|
i = 0
|
|
ncont = []
|
|
while i < len(contents):
|
|
line = contents[i]
|
|
if RE_AM_CONTINUING.search(line):
|
|
line = RE_AM_CONTINUING.sub('', line)
|
|
j = i + 1
|
|
while j < len(contents) and RE_AM_CONTINUING.search(contents[j]):
|
|
line += RE_AM_CONTINUING.sub('', contents[j])
|
|
j += 1
|
|
else:
|
|
if j < len(contents):
|
|
line += contents[j]
|
|
i = j
|
|
else:
|
|
i += 1
|
|
ncont.append(line)
|
|
|
|
#include, var define, var evaluation
|
|
i = -1
|
|
skip = False
|
|
oldskip = []
|
|
while i < len(ncont) - 1:
|
|
i += 1
|
|
line = ncont[i]
|
|
mo = RE_AM_IF.search(line)
|
|
if mo:
|
|
oldskip.append(skip)
|
|
skip = False if mo.group(1) in conds and conds[mo.group(1)] \
|
|
else True
|
|
continue
|
|
mo = RE_AM_IFNOT.search(line)
|
|
if mo:
|
|
oldskip.append(skip)
|
|
skip = False if mo.group(1) not in conds and conds[mo.group(1)] \
|
|
else True
|
|
continue
|
|
mo = RE_AM_ELSE.search(line)
|
|
if mo:
|
|
skip = not skip
|
|
continue
|
|
mo = RE_AM_ENDIF.search(line)
|
|
if mo:
|
|
if oldskip:
|
|
skip = oldskip.pop()
|
|
continue
|
|
if not skip:
|
|
mo = RE_AM_INCLUDE.search(line)
|
|
if mo:
|
|
cur_vars.update(read_vars_from_AM(am_eval(mo.group(1)), cur_vars, conds, None))
|
|
continue
|
|
mo = RE_AM_VAR.search(line)
|
|
if mo:
|
|
cur_vars[mo.group(1)] = am_eval(mo.group(2).strip())
|
|
continue
|
|
mo = RE_AM_VAR_ADD.search(line)
|
|
if mo:
|
|
try:
|
|
cur_vars[mo.group(1)] += ' '
|
|
except KeyError:
|
|
cur_vars[mo.group(1)] = ''
|
|
cur_vars[mo.group(1)] += am_eval(mo.group(2).strip())
|
|
continue
|
|
|
|
#filter:
|
|
if filters != None:
|
|
ret = {}
|
|
for i in filters:
|
|
ret[i] = cur_vars.get(i, '')
|
|
return ret
|
|
else:
|
|
return cur_vars
|
|
|
|
def process_include(src, dest, includes):
|
|
RE_INCLUDE = re.compile(r'^\s*#include\s+"(.*)"')
|
|
with open(src, 'r') as s:
|
|
with open(dest, 'w') as d:
|
|
for i in s:
|
|
mo = RE_INCLUDE.search(i)
|
|
if mo:
|
|
target = ''
|
|
for j in includes:
|
|
#print "searching in ", j
|
|
if mo.group(1) in os.listdir(j):
|
|
target = os.path.join(j, mo.group(1))
|
|
break
|
|
if not target:
|
|
raise Exception("Couldn't find include file %s" % mo.group(1))
|
|
else:
|
|
with open(target, 'r') as t:
|
|
for inc in t.readlines():
|
|
d.write(inc)
|
|
else:
|
|
d.write(i)
|
|
|
|
#Generate the source files listing that is used
|
|
def generate_src_list (srcroot, srcdir, filters_src, filter_conds, filter_c, mk_am_file):
|
|
mkfile = ''
|
|
if mk_am_file is None or mk_am_file == '':
|
|
mkfile = 'Makefile.am'
|
|
else:
|
|
mkfile = mk_am_file
|
|
vars = read_vars_from_AM(os.path.join(srcdir, mkfile),
|
|
vars = {'top_srcdir': srcroot},
|
|
conds = filter_conds,
|
|
filters = filters_src)
|
|
files = []
|
|
for src_filters_item in filters_src:
|
|
files += vars[src_filters_item].split()
|
|
if filter_c is True:
|
|
sources = [i for i in files if i.endswith('.c') ]
|
|
return sources
|
|
else:
|
|
return files
|
|
|
|
# Generate the Visual Studio 2008 Project Files from the templates
|
|
def gen_vs9_project (projname, srcroot, srcdir_name, sources_list):
|
|
vs_file_list_dir = os.path.join (srcroot, 'build', 'win32')
|
|
|
|
with open (os.path.join (vs_file_list_dir,
|
|
projname + '.sourcefiles'), 'w') as vs9srclist:
|
|
for i in sources_list:
|
|
vs9srclist.write ('\t\t\t<File RelativePath="..\\..\\..\\' + srcdir_name + '\\' + i.replace('/', '\\') + '" />\n')
|
|
|
|
process_include (os.path.join(srcroot, 'build', 'win32', 'vs9', projname + '.vcprojin'),
|
|
os.path.join(srcroot, 'build', 'win32', 'vs9', projname + '.vcproj'),
|
|
includes = [vs_file_list_dir])
|
|
|
|
os.unlink(os.path.join(srcroot, 'build', 'win32', projname + '.sourcefiles'))
|
|
|
|
# Generate the Visual Studio 2010 Project Files from the templates
|
|
def gen_vs10_project (projname, srcroot, srcdir_name, sources_list):
|
|
vs_file_list_dir = os.path.join (srcroot, 'build', 'win32')
|
|
|
|
with open (os.path.join (vs_file_list_dir,
|
|
projname + '.vs10.sourcefiles'), 'w') as vs10srclist:
|
|
for j in sources_list:
|
|
vs10srclist.write (' <ClCompile Include="..\\..\\..\\' + srcdir_name + '\\' + j.replace('/', '\\') + '" />\n')
|
|
|
|
with open (os.path.join (vs_file_list_dir,
|
|
projname + '.vs10.sourcefiles.filters'), 'w') as vs10srclist_filter:
|
|
for k in sources_list:
|
|
vs10srclist_filter.write (' <ClCompile Include="..\\..\\..\\' + srcdir_name + '\\' + k.replace('/', '\\') + '"><Filter>Source Files</Filter></ClCompile>\n')
|
|
|
|
process_include (os.path.join(srcroot, 'build', 'win32', 'vs10', projname + '.vcxprojin'),
|
|
os.path.join(srcroot, 'build', 'win32', 'vs10', projname + '.vcxproj'),
|
|
includes = [vs_file_list_dir])
|
|
process_include (os.path.join(srcroot, 'build', 'win32', 'vs10', projname + '.vcxproj.filtersin'),
|
|
os.path.join(srcroot, 'build', 'win32', 'vs10', projname + '.vcxproj.filters'),
|
|
includes = [vs_file_list_dir])
|
|
|
|
os.unlink(os.path.join(srcroot, 'build', 'win32', projname + '.vs10.sourcefiles'))
|
|
os.unlink(os.path.join(srcroot, 'build', 'win32', projname + '.vs10.sourcefiles.filters'))
|
|
|
|
def gen_vs_inst_list (projname, srcroot, srcdirs, inst_lists, destdir_names, isVS9):
|
|
vs_file_list_dir = os.path.join (srcroot, 'build', 'win32')
|
|
vsver = ''
|
|
vsprops_line_ending = ''
|
|
vsprops_file_ext = ''
|
|
if isVS9 is True:
|
|
vsver = '9'
|
|
vsprops_line_ending = '
\n'
|
|
vsprops_file_ext = '.vsprops'
|
|
else:
|
|
vsver = '10'
|
|
vsprops_line_ending = '\n\n'
|
|
vsprops_file_ext = '.props'
|
|
|
|
with open (os.path.join (vs_file_list_dir,
|
|
projname + '.vs' + vsver + 'instfiles'), 'w') as vsinstlist:
|
|
|
|
for file_list, srcdir, dir_name in zip (inst_lists, srcdirs, destdir_names):
|
|
for i in file_list:
|
|
vsinstlist.write ('copy ..\\..\\..\\' +
|
|
srcdir + '\\' +
|
|
i.replace ('/', '\\') +
|
|
' $(CopyDir)\\' +
|
|
dir_name +
|
|
vsprops_line_ending)
|
|
process_include (os.path.join(srcroot, 'build', 'win32', 'vs' + vsver, projname + '-install' + vsprops_file_ext + 'in'),
|
|
os.path.join(srcroot, 'build', 'win32', 'vs' + vsver, projname + '-install' + vsprops_file_ext),
|
|
includes = [vs_file_list_dir])
|
|
|
|
os.unlink(os.path.join (vs_file_list_dir, projname + '.vs' + vsver + 'instfiles'))
|
|
|
|
def generate_nmake_makefiles(srcroot, srcdir, base_name, makefile_name, progs_list):
|
|
file_list_dir = os.path.join (srcroot, 'build', 'win32')
|
|
|
|
with open (os.path.join (file_list_dir,
|
|
base_name + '_progs'), 'w') as proglist:
|
|
for i in progs_list:
|
|
proglist.write ('\t' + i + '$(EXEEXT)\t\\\n')
|
|
|
|
|
|
process_include (os.path.join(srcdir, makefile_name + 'in'),
|
|
os.path.join(srcdir, makefile_name),
|
|
includes = [file_list_dir])
|
|
|
|
os.unlink(os.path.join (file_list_dir, base_name + '_progs'))
|
|
|