1000 lines
28 KiB
Python
1000 lines
28 KiB
Python
'''
|
|
Set up Visual Sudio 2010 to build a specified MPIR configuration
|
|
|
|
Copyright (C) 2011, Brian Gladman
|
|
|
|
32-bit:
|
|
x86w: x86, i386, i486, i586, pentium, pentiummmx, k6, k62, k63, k7, athlon, viac3, viac32, x86_64
|
|
p6: pentiumpro, i686
|
|
mmx: pentium2
|
|
sse2: pentium4, prescott, core, netburst, netburstlahf
|
|
p3mmx: pentium3, k8, k10, k102, k103, bulldozer, bobcat, core2, penryn, nehalem, westmere, sandybridge, atom, nano
|
|
|
|
64-bit:
|
|
x86_64: x86_64w
|
|
netburst: netburst, netburstlahf
|
|
k8only: k8, nano
|
|
k10: k10
|
|
k102: k102, k103, bulldozer
|
|
bobcat: bobcat
|
|
core2: core2
|
|
penryn: penryn
|
|
nehalem: nehalem
|
|
westmere: westmere
|
|
sandybridge: sandybridge
|
|
atom atom
|
|
'''
|
|
|
|
from __future__ import print_function
|
|
from operator import itemgetter
|
|
from os import listdir, walk, unlink, makedirs, getcwd
|
|
from os.path import split, splitext, isdir, relpath, join, exists
|
|
from copy import deepcopy
|
|
from sys import argv, exit
|
|
from filecmp import cmp
|
|
from shutil import copy
|
|
from re import compile, search
|
|
from collections import defaultdict
|
|
from uuid import uuid1
|
|
from time import sleep
|
|
|
|
# for script debugging
|
|
debug = False
|
|
# either add a prebuild step to the project files or do it here
|
|
add_prebuild = True
|
|
# output a build project for the C++ static library
|
|
add_cpp_lib = False
|
|
|
|
# The path to the mpir root directory
|
|
build_vc = 'build.vc10/'
|
|
mpir_dir = '../'
|
|
build_dir = mpir_dir + build_vc
|
|
|
|
# paths that might include source files(*.c, *.h, *.asm)
|
|
c_directories = ( '', 'build.vc10', 'fft', 'mpf', 'mpq', 'mpz', 'printf', 'scanf' )
|
|
|
|
# files that are to be excluded from the build
|
|
exclude_file_list = ('config.guess', 'cfg', 'getopt', 'getrusage', 'gettimeofday', 'cpuid',
|
|
'obsolete', 'win_timing', 'gmp-mparam', 'tal-debug', 'tal-notreent',
|
|
'new_fft', 'new_fft_with_flint', 'compat', 'udiv_w_sdiv' )
|
|
|
|
# copy from file ipath to file opath but avoid copying if
|
|
# opath exists and is the same as ipath (this is to avoid
|
|
# triggering an unecessary rebuild).
|
|
|
|
def write_f(ipath, opath):
|
|
if exists(ipath) and not isdir(ipath):
|
|
if exists(opath):
|
|
if isdir(opath) or cmp(ipath, opath):
|
|
return
|
|
copy(ipath, opath)
|
|
|
|
# append a file (ipath) to an existing file (opath)
|
|
|
|
def append_f(ipath, opath):
|
|
try:
|
|
with open(opath, 'ab') as out_file:
|
|
try:
|
|
with open(ipath, 'rb') as in_file:
|
|
buf = in_file.read(8192)
|
|
while buf:
|
|
out_file.write(buf)
|
|
buf = in_file.read(8192)
|
|
except IOError:
|
|
print('error reading {0:s} for input'.format(f))
|
|
return
|
|
except IOError:
|
|
print('error opening {0:s} for output'.format(opath))
|
|
|
|
# copy files in a list from in_dir to out_dir
|
|
def copy_files(file_list, in_dir, out_dir):
|
|
try:
|
|
makedirs(out_dir)
|
|
except:
|
|
IOError
|
|
for f in file_list:
|
|
copy(join(in_dir, f), out_dir)
|
|
|
|
# Recursively search a given directory tree to find header,
|
|
# C and assembler code files that either replace or augment
|
|
# the generic C source files in the input list 'src_list'.
|
|
# As the directory tree is searched, files in each directory
|
|
# become the source code files for the current directory and
|
|
# the default source code files for its child directories.
|
|
#
|
|
# Lists of default header, C and assembler source code files
|
|
# are maintained as the tree is traversed and if a file in
|
|
# the current directory matches the name of a file in the
|
|
# default file list (name matches ignore file extensions),
|
|
# the name in the list is removed and is replaced by the new
|
|
# file found. On return each directory in the tree had an
|
|
# entry in the returned dictionary that contains:
|
|
#
|
|
# 1. The list of header files
|
|
#
|
|
# 2. The list of C source code files for the directory
|
|
#
|
|
# 3. The list of assembler code files that replace C files
|
|
#
|
|
# 4. The list of assembler files that are not C replacements
|
|
#
|
|
|
|
def find_asm(path, cf_list):
|
|
d = dict()
|
|
for root, dirs, files in walk(path):
|
|
if '.svn' in dirs: # ignore SVN directories
|
|
dirs.remove('.svn')
|
|
if 'fat' in dirs: # ignore fat directory
|
|
dirs.remove('fat')
|
|
relp = relpath(root, path) # path from asm root
|
|
relr = relpath(root, mpir_dir) # path from MPIR root
|
|
if relp == '.': # set C files as default
|
|
relp = h = t = ''
|
|
d[''] = [ [], deepcopy(cf_list), [], [], relr ]
|
|
else:
|
|
h, t = split(relp) # h = parent, t = this directory
|
|
# copy defaults from this directories parent
|
|
d[relp] = [ deepcopy(d[h][0]), deepcopy(d[h][1]),
|
|
deepcopy(d[h][2]), deepcopy(d[h][3]), relr ]
|
|
for f in files: # for the files in this directory
|
|
n, x = splitext(f)
|
|
if x == '.h': # if it is a header file, remove
|
|
for cf in reversed(d[relp][0]): # any matching default
|
|
if cf[0] == n:
|
|
d[relp][0].remove(cf)
|
|
d[relp][0] += [(n, x, relr)] # and add the local header file
|
|
if x == '.c': # if it is a C file, remove
|
|
for cf in reversed(d[relp][1]): # any matching default
|
|
if cf[0] == n:
|
|
d[relp][1].remove(cf)
|
|
d[relp][1] += [(n, x, relr)] # and add the local C file
|
|
if x == '.asm': # if it is an assembler file
|
|
match = False
|
|
for cf in reversed(d[relp][1]): # remove any matching C file
|
|
if cf[0] == n:
|
|
d[relp][1].remove(cf)
|
|
match = True
|
|
break
|
|
for cf in reversed(d[relp][2]): # and remove any matching
|
|
if cf[0] == n: # assembler file
|
|
d[relp][2].remove(cf)
|
|
match = True
|
|
break
|
|
if match: # if a match was found, put the
|
|
d[relp][2] += [(n, x, relr)] # file in the replacement list
|
|
else: # otherwise look for it in the
|
|
for cf in reversed(d[relp][3]): # additional files list
|
|
if cf[0] == n:
|
|
d[relp][3].remove(cf)
|
|
break
|
|
d[relp][3] += [(n, x, relr)]
|
|
for k in d: # additional assembler list
|
|
for i in range(4):
|
|
d[k][i].sort(key=itemgetter(0)) # sort the four file lists
|
|
return d
|
|
|
|
# create 4 lists of c, h, cc (or cpp) and asm (or as) files in a directory
|
|
|
|
def find_src(dir_list):
|
|
# list number from file extension
|
|
di = { '.h': 0, '.c': 1, '.cc': 2, '.cpp': 2, '.asm': 3, '.as': 3 }
|
|
list = [ [], [], [], [] ]
|
|
for d in dir_list:
|
|
for f in listdir(join(mpir_dir, d)):
|
|
if f == '.svn':
|
|
continue # ignore SVN directories
|
|
if not isdir(f):
|
|
n, x = splitext(f) # split into name + extension
|
|
if x in di and not n in exclude_file_list:
|
|
list[di[x]] += [(n, x, d)] # if of the right type and is
|
|
for x in list: # not in the exclude list
|
|
x.sort(key=itemgetter(2, 0, 1)) # add it to appropriate list
|
|
return list
|
|
|
|
# scan the files in the input set and find the symbols
|
|
# defined in the files
|
|
fr_sym = compile(r'LEAF_PROC\s+(\w+)')
|
|
lf_sym = compile(r'FRAME_PROC\s+(\w+)')
|
|
wf_sym = compile(r'WIN64_GCC_PROC\s+(\w+)')
|
|
g3_sym = compile(r'global\s+___g(\w+)')
|
|
g2_sym = compile(r'global\s+__g(\w+)')
|
|
|
|
def get_symbols(setf, sym_dir):
|
|
for f in setf:
|
|
fn = join(mpir_dir, f[2], f[0] + f[1])
|
|
with open(fn, 'r') as inf:
|
|
lines = inf.readlines()
|
|
for l in lines:
|
|
m = fr_sym.search(l)
|
|
if m:
|
|
sym_dir[f] |= set((m.groups(1)[0],))
|
|
m = lf_sym.search(l)
|
|
if m:
|
|
sym_dir[f] |= set((m.groups(1)[0],))
|
|
m = wf_sym.search(l)
|
|
if m:
|
|
sym_dir[f] |= set((m.groups(1)[0],))
|
|
m = g3_sym.search(l)
|
|
if m:
|
|
sym_dir[f] |= set((m.groups(1)[0],))
|
|
else:
|
|
m = g2_sym.search(l)
|
|
if m:
|
|
sym_dir[f] |= set((m.groups(1)[0],))
|
|
|
|
def file_symbols(cf):
|
|
sym_dir = defaultdict(set)
|
|
for c in cf:
|
|
if c == 'fat':
|
|
continue
|
|
setf = set()
|
|
for f in cf[c][2] + cf[c][3]:
|
|
setf |= set((f,))
|
|
get_symbols(setf, sym_dir)
|
|
return sym_dir
|
|
|
|
def gen_have_lists(cf, sym_dir):
|
|
for c in cf:
|
|
if c == 'fat':
|
|
continue
|
|
set_sym2 = set()
|
|
for f in cf[c][2]:
|
|
set_sym2 |= sym_dir[f]
|
|
set_sym3 = set()
|
|
for f in cf[c][3]:
|
|
set_sym3 |= sym_dir[f]
|
|
cf[c] += [sorted(list(set_sym2)), sorted(list(set_sym3))]
|
|
with open(join(mpir_dir, cf[c][4], 'cfg.h'), 'w') as outf:
|
|
for sym in sorted(set_sym2 | set_sym3):
|
|
print(sym, file=outf)
|
|
# print('/* assembler symbols also available in C files */', file=outf)
|
|
# for sym in sorted(set_sym2):
|
|
# print(sym, file=outf)
|
|
# print('/* assembler symbols not available in C files */', file=outf)
|
|
# for sym in sorted(set_sym3):
|
|
# print(sym, file=outf)
|
|
|
|
# generate Visual Studio 2010 IDE Filter
|
|
|
|
def filter_folders(cf_list, af_list, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
<Filter Include="Header Files" />
|
|
<Filter Include="Source Files" />
|
|
'''
|
|
f2 = r''' <Filter Include="Source Files\{0:s}" />
|
|
'''
|
|
f3 = r''' </ItemGroup>
|
|
'''
|
|
c_dirs = set(i[2] for i in cf_list)
|
|
a_dirs = set(i[2] for i in af_list)
|
|
if a_dirs:
|
|
c_dirs |= set(('mpn\yasm',))
|
|
outf.write(f1)
|
|
for d in sorted(c_dirs):
|
|
if d:
|
|
t = d if d != 'mpn\generic' else 'mpn'
|
|
outf.write(f2.format(t))
|
|
outf.write(f3)
|
|
|
|
filter_hdr_item = ' <ClInclude Include="..\..\{}">\n <Filter>Header Files</Filter>\n </ClInclude>\n'
|
|
|
|
def filter_headers(hdr_list, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
'''
|
|
f2 = r''' <ClInclude Include="..\..\{}">
|
|
<Filter>Header Files</Filter>
|
|
</ClInclude>
|
|
'''
|
|
f3 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for h in hdr_list:
|
|
outf.write(f2.format(h))
|
|
outf.write(f3)
|
|
|
|
def filter_csrc(cf_list, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
'''
|
|
f2 = r''' <ClCompile Include="..\..\{0:s}">
|
|
<Filter>Source Files</Filter>
|
|
</ClCompile>
|
|
'''
|
|
f3 = r''' <ClCompile Include="..\..\{1:s}\{0:s}">
|
|
<Filter>Source Files\{2:s}</Filter>
|
|
</ClCompile>
|
|
'''
|
|
f4 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for i in cf_list:
|
|
if not i[2]:
|
|
outf.write(f2.format(i[0] + i[1]))
|
|
else:
|
|
t = 'mpn' if i[2].endswith('generic') else i[2]
|
|
outf.write(f3.format(i[0] + i[1], i[2], t))
|
|
outf.write(f4)
|
|
|
|
def filter_asrc(af_list, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
'''
|
|
f2 = r''' <YASM Include="..\..\{1:s}\{0:s}">
|
|
<Filter>Source Files\mpn\yasm</Filter>
|
|
</YASM>
|
|
'''
|
|
f3 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for i in af_list:
|
|
outf.write(f2.format(i[0] + i[1], i[2], i[2]))
|
|
outf.write(f3)
|
|
|
|
def gen_filter(name, hf_list, cf_list, af_list):
|
|
|
|
f1 = r'''<?xml version="1.0" encoding="utf-8"?>
|
|
<Project ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
|
'''
|
|
f2 = r''' <ItemGroup>
|
|
<None Include="..\..\gmp-h.in" />
|
|
</ItemGroup>
|
|
</Project>
|
|
'''
|
|
|
|
fn = join(build_dir, name)
|
|
try:
|
|
makedirs(split(fn)[0])
|
|
except:
|
|
IOError
|
|
with open(fn, 'w') as outf:
|
|
|
|
outf.write(f1)
|
|
filter_folders(cf_list, af_list, outf)
|
|
filter_headers(hf_list, outf)
|
|
filter_csrc(cf_list, outf)
|
|
if af_list:
|
|
filter_asrc(af_list, outf)
|
|
outf.write(f2)
|
|
|
|
# generate vcxproj file
|
|
|
|
def vcx_proj_cfg(plat, outf):
|
|
|
|
f1 = r''' <ItemGroup Label="ProjectConfigurations">
|
|
'''
|
|
f2 = r''' <ProjectConfiguration Include="{1:s}|{0:s}">
|
|
<Configuration>{1:s}</Configuration>
|
|
<Platform>{0:s}</Platform>
|
|
</ProjectConfiguration>
|
|
'''
|
|
f3 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for pl in plat:
|
|
for conf in ('Release', 'Debug'):
|
|
outf.write(f2.format(pl, conf))
|
|
outf.write(f3)
|
|
|
|
def vcx_globals(name, guid, outf):
|
|
|
|
f1 = r''' <PropertyGroup Label="Globals">
|
|
<RootNamespace>{0:s}</RootNamespace>
|
|
<Keyword>Win32Proj</Keyword>
|
|
<ProjectGuid>{1:s}</ProjectGuid>
|
|
</PropertyGroup>
|
|
'''
|
|
outf.write(f1.format(name, guid))
|
|
|
|
def vcx_default_cpp_props(outf):
|
|
|
|
f1 = r''' <Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
|
|
'''
|
|
outf.write(f1)
|
|
|
|
def vcx_library_type(plat, is_dll, outf):
|
|
|
|
f1 = r''' <PropertyGroup Condition="'$(Configuration)|$(Platform)'=='{1:s}|{0:s}'" Label="Configuration">
|
|
<ConfigurationType>{2:s}Library</ConfigurationType>
|
|
<CharacterSet>MultiByte</CharacterSet>
|
|
</PropertyGroup>
|
|
'''
|
|
for pl in plat:
|
|
for conf in ('Release', 'Debug'):
|
|
outf.write(f1.format(pl, conf, 'Dynamic' if is_dll else 'Static' ))
|
|
|
|
def vcx_cpp_props(outf):
|
|
|
|
f1 = r''' <Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
|
|
'''
|
|
outf.write(f1)
|
|
|
|
def vcx_extensions(outf):
|
|
|
|
f1 = r''' <ImportGroup Label="ExtensionSettings">
|
|
<Import Project="..\vsyasm.props" />
|
|
</ImportGroup>
|
|
'''
|
|
outf.write(f1)
|
|
|
|
def vcx_user_props(plat, outf):
|
|
|
|
f1 = r''' <ImportGroup Condition="'$(Configuration)|$(Platform)'=='{1:s}|{0:s}'" Label="PropertySheets">
|
|
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" />
|
|
</ImportGroup>
|
|
'''
|
|
for pl in plat:
|
|
for conf in ('Release', 'Debug'):
|
|
outf.write(f1.format(pl, conf))
|
|
|
|
def vcx_target_name_and_dirs(name, plat, is_dll, outf):
|
|
|
|
f1 = r''' <PropertyGroup>
|
|
<_ProjectFileVersion>10.0.21006.1</_ProjectFileVersion>
|
|
'''
|
|
f2 = r''' <TargetName Condition="'$(Configuration)|$(Platform)'=='{1:s}|{0:s}'">{2:s}</TargetName>
|
|
<IntDir Condition="'$(Configuration)|$(Platform)'=='{1:s}|{0:s}'">$(Platform)\$(Configuration)\</IntDir>
|
|
<OutDir Condition="'$(Configuration)|$(Platform)'=='{1:s}|{0:s}'">$(SolutionDir)$(Platform)\$(Configuration)\</OutDir>
|
|
'''
|
|
f3 = r''' </PropertyGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for pl in plat:
|
|
for conf in ('Release', 'Debug'):
|
|
outf.write(f2.format(pl, conf, name))
|
|
outf.write(f3)
|
|
|
|
def yasm_options(plat, is_dll, outf):
|
|
|
|
f1 = r''' <YASM>
|
|
<Defines>{0:s}</Defines>
|
|
<IncludePaths>..\..\mpn\x86{1:s}w\</IncludePaths>
|
|
<Debug>true</Debug>
|
|
<ObjectFileName>$(IntDir)mpn\</ObjectFileName>
|
|
<ObjectFile>$(IntDir)mpn\</ObjectFile>
|
|
</YASM>
|
|
'''
|
|
|
|
outf.write(f1.format('DLL' if is_dll else '', '' if plat == 'Win32' else '_64'))
|
|
|
|
def compiler_options(plat, is_dll, is_debug, outf):
|
|
|
|
f1 = r''' <ClCompile>
|
|
<Optimization>{0:s}</Optimization>
|
|
<IntrinsicFunctions>true</IntrinsicFunctions>
|
|
<AdditionalIncludeDirectories>..\..\</AdditionalIncludeDirectories>
|
|
<PreprocessorDefinitions>{1:s}%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
|
<RuntimeLibrary>MultiThreaded{2:s}</RuntimeLibrary>
|
|
<ProgramDataBaseFileName>$(TargetDir)$(TargetName).pdb</ProgramDataBaseFileName>
|
|
<DebugInformationFormat>ProgramDatabase</DebugInformationFormat>
|
|
</ClCompile>
|
|
'''
|
|
|
|
if is_dll:
|
|
s1 = 'DEBUG;HAVE_CONFIG_H;__GMP_LIBGMP_DLL;__GMP_WITHIN_GMP;__GMP_WITHIN_GMPXX;'
|
|
s2 = 'DLL'
|
|
else:
|
|
s1 = 'DEBUG;_LIB;HAVE_CONFIG_H;'
|
|
s2 = ''
|
|
if plat == 'x64':
|
|
s1 = s1 + '_WIN64;'
|
|
if is_debug:
|
|
opt, defines, crt = 'Disabled', '_' + s1, 'Debug' + s2
|
|
else:
|
|
opt, defines, crt = 'Full', 'N' + s1, s2
|
|
outf.write(f1.format(opt, defines, crt))
|
|
|
|
def linker_options(outf):
|
|
|
|
f1 = r''' <Link>
|
|
<GenerateDebugInformation>true</GenerateDebugInformation>
|
|
<LargeAddressAware>true</LargeAddressAware>
|
|
</Link>
|
|
'''
|
|
outf.write(f1)
|
|
|
|
def vcx_pre_build(name, plat, outf):
|
|
|
|
f1 = r''' <PreBuildEvent>
|
|
<Command>cd ..\
|
|
prebuild {0:s} {1:s}
|
|
</Command>
|
|
</PreBuildEvent>
|
|
'''
|
|
outf.write(f1.format(name, plat))
|
|
|
|
def vcx_post_build(is_cpp, outf):
|
|
|
|
f1 = r'''
|
|
<PostBuildEvent>
|
|
<Command>cd ..\
|
|
postbuild "$(TargetPath)"
|
|
</Command>
|
|
</PostBuildEvent>
|
|
'''
|
|
|
|
outf.write(f1)
|
|
|
|
def vcx_tool_options(config, plat, is_dll, is_cpp, af_list, outf):
|
|
|
|
f1 = r''' <ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='{1:s}|{0:s}'">
|
|
'''
|
|
f2 = r''' </ItemDefinitionGroup>
|
|
'''
|
|
for pl in plat:
|
|
for is_debug in (False, True):
|
|
outf.write(f1.format(pl, 'Debug' if is_debug else 'Release'))
|
|
if add_prebuild and not is_cpp:
|
|
vcx_pre_build(config, pl, outf)
|
|
if af_list:
|
|
yasm_options(plat, is_dll, outf)
|
|
compiler_options(pl, is_dll, is_debug, outf)
|
|
if is_dll:
|
|
linker_options(outf)
|
|
vcx_post_build(is_cpp, outf)
|
|
outf.write(f2)
|
|
|
|
def vcx_hdr_items(hdr_list, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
'''
|
|
f2 = r''' <ClInclude Include="..\..\{0:s}" />
|
|
'''
|
|
f3 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for i in hdr_list:
|
|
outf.write(f2.format(i))
|
|
outf.write(f3)
|
|
|
|
def vcx_c_items(cf_list, plat, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
'''
|
|
f2 = r''' <ClCompile Include="..\..\{0[0]:s}{0[1]:s}" />
|
|
'''
|
|
f3 = r''' <ClCompile Include="..\..\{0[2]:s}\{0[0]:s}{0[1]:s}">
|
|
'''
|
|
f4 = r''' <ObjectFileName Condition="'$(Configuration)|$(Platform)'=='{0:s}|{1:s}'">$(IntDir){2:s}\</ObjectFileName>
|
|
'''
|
|
f5 = r''' </ClCompile>
|
|
'''
|
|
f6 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for nxd in cf_list:
|
|
if nxd[2] == '':
|
|
outf.write(f2.format(nxd))
|
|
else:
|
|
outf.write(f3.format(nxd))
|
|
for cf in ('Release', 'Debug'):
|
|
for pl in plat:
|
|
outf.write(f4.format(cf, pl, 'mpn' if nxd[2].endswith('generic') else nxd[2]))
|
|
outf.write(f5)
|
|
outf.write(f6)
|
|
|
|
def vcx_a_items(af_list, outf):
|
|
|
|
f1 = r''' <ItemGroup>
|
|
'''
|
|
f2 = r''' <YASM Include="..\..\{0[2]:s}\{0[0]:s}{0[1]:s}" />
|
|
'''
|
|
f3 = r''' </ItemGroup>
|
|
'''
|
|
outf.write(f1)
|
|
for nxd in af_list:
|
|
outf.write(f2.format(nxd))
|
|
outf.write(f3)
|
|
|
|
def gen_vcxproj(proj_name, file_name, guid, config, plat, is_dll, is_cpp, hf_list, cf_list, af_list):
|
|
|
|
f1 = r'''<?xml version="1.0" encoding="utf-8"?>
|
|
<Project DefaultTargets="Build" ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
|
'''
|
|
f2 = r''' <PropertyGroup Label="UserMacros" />
|
|
'''
|
|
f3 = r''' <Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
|
'''
|
|
f4 = r''' <ImportGroup Label="ExtensionTargets">
|
|
<Import Project="..\vsyasm.targets" />
|
|
</ImportGroup>
|
|
'''
|
|
|
|
f5 = r'''<ItemGroup>
|
|
<None Include="..\..\gmp-h.in" />
|
|
</ItemGroup>
|
|
</Project>
|
|
'''
|
|
|
|
with open(join(build_dir, file_name), 'w') as outf:
|
|
outf.write(f1)
|
|
vcx_proj_cfg(plat, outf)
|
|
vcx_globals(proj_name, guid, outf)
|
|
vcx_default_cpp_props(outf)
|
|
vcx_library_type(plat, is_dll, outf)
|
|
vcx_cpp_props(outf)
|
|
vcx_extensions(outf)
|
|
vcx_user_props(plat, outf)
|
|
outf.write(f2)
|
|
vcx_target_name_and_dirs(proj_name, plat, is_dll, outf)
|
|
vcx_tool_options(config, plat, is_dll, is_cpp, af_list, outf)
|
|
vcx_hdr_items(hf_list, outf)
|
|
vcx_c_items(cf_list, plat, outf)
|
|
vcx_a_items(af_list, outf)
|
|
outf.write(f3)
|
|
if af_list:
|
|
outf.write(f4)
|
|
outf.write(f5)
|
|
|
|
# add a project file to the solution
|
|
|
|
def add_proj_to_sln(proj_name, file_name, guid):
|
|
|
|
f6 = r'''Project("{0:s}") = "{1:s}", "{2:s}", "{3:s}"
|
|
'''
|
|
f7 = r'''EndProject
|
|
'''
|
|
re_guid = compile(r'Project\s*\(\s*\"\s*\{([^\}]+)\s*\}\s*\"\s*\)')
|
|
lines = open(join(build_dir, 'mpir.sln')).readlines()
|
|
s_guid = ''
|
|
i_pos = 0
|
|
for i, ln in enumerate(lines):
|
|
m = re_guid.search(ln)
|
|
if m:
|
|
if not s_guid:
|
|
s_guid = '{' + m.group(1) + '}'
|
|
if ln.find(proj_name) != -1:
|
|
i_pos = (i, i + 2)
|
|
break
|
|
if ln.find(r'Global') != -1:
|
|
i_pos = (i, i)
|
|
break
|
|
else:
|
|
print('error in updating the solution')
|
|
exit()
|
|
if i_pos and s_guid:
|
|
lines[i_pos[0]:i_pos[1]] = [f6.format(s_guid, proj_name, file_name, guid), f7]
|
|
open(join(build_dir, 'mpir.sln'), 'w').writelines(lines)
|
|
|
|
# compile list of C files
|
|
t = find_src(c_directories)
|
|
c_hdr_list = t[0]
|
|
c_src_list = t[1]
|
|
if t[2] or t[3]:
|
|
print('found C++ and/or assembler file(s) in a C directory')
|
|
if t[2]:
|
|
for f in t[2]:
|
|
print(f)
|
|
print()
|
|
if t[3]:
|
|
for f in t[3]:
|
|
print(f)
|
|
print()
|
|
|
|
# compile list of C++ files
|
|
t = find_src(['cxx'])
|
|
cc_hdr_list = t[0]
|
|
cc_src_list = t[2]
|
|
if t[1] or t[3]:
|
|
print('found C and/or assembler file(s) in a C++ directory')
|
|
if t[1]:
|
|
for f in t[1]:
|
|
print(f)
|
|
print()
|
|
if t[3]:
|
|
for f in cc_src_list:
|
|
print(f)
|
|
print()
|
|
|
|
# compile list of C files in mpn\generic
|
|
t = find_src([r'mpn\generic'])
|
|
gc_hdr_list = t[0]
|
|
gc_src_list = t[1]
|
|
if t[2] or t[3]:
|
|
print('found C++ and/or assembler file(s) in a C directory')
|
|
if t[2]:
|
|
for f in gc_hdr_list:
|
|
print(f)
|
|
print()
|
|
if t[3]:
|
|
for f in gc_src_list:
|
|
print(f)
|
|
print()
|
|
|
|
# prepare the generic C build
|
|
mpn_gc = dict((('gc',[ gc_hdr_list, gc_src_list, [], [] ]),))
|
|
|
|
# prepare the list of Win32 builds
|
|
mpn_32 = find_asm(mpir_dir + 'mpn/x86w', gc_src_list)
|
|
syms32 = file_symbols(mpn_32)
|
|
if len(argv) == 1 or int(argv[1]) & 1:
|
|
gen_have_lists(mpn_32, syms32)
|
|
del mpn_32['']
|
|
|
|
# prepare the list of x64 builds
|
|
mpn_64 = find_asm(mpir_dir + 'mpn/x86_64w', gc_src_list)
|
|
syms64 = file_symbols(mpn_64)
|
|
if len(argv) == 1 or int(argv[1]) & 1:
|
|
gen_have_lists(mpn_64, syms64)
|
|
del mpn_64['']
|
|
|
|
if len(argv) != 1 and not (int(argv[1]) & 2):
|
|
exit()
|
|
|
|
nd_gc = len(mpn_gc)
|
|
nd_32 = nd_gc + len(mpn_32)
|
|
nd_nd = nd_32 + len(mpn_64)
|
|
|
|
# now ask user which builds they wish to generate
|
|
|
|
while True:
|
|
cnt = 0
|
|
for v in sorted(mpn_gc):
|
|
cnt += 1
|
|
print('{0:2d}. {1:18s} '.format(cnt, v))
|
|
for v in sorted(mpn_32):
|
|
cnt += 1
|
|
print('{0:2d}. {1:18s} (win32)'.format(cnt, v))
|
|
for v in sorted(mpn_64):
|
|
cnt += 1
|
|
print('{0:2d}. {1:18s} (x64)'.format(cnt, v))
|
|
s = input('Space separated list of builds (1..{0:d}, 0 to exit)? '.format(cnt))
|
|
n_list = [int(c) for c in s.split()]
|
|
if 0 in n_list:
|
|
exit()
|
|
if any(n < 1 or n > nd_nd for n in n_list):
|
|
print('list contains invalid build numbers')
|
|
sleep(2)
|
|
else:
|
|
break
|
|
|
|
# multiple builds must each have their own prebuilds
|
|
if len(n_list) > 1:
|
|
add_prebuild = True
|
|
|
|
# now gnerate the requested builds
|
|
for n in n_list:
|
|
|
|
if 0 < n <= nd_gc:
|
|
config = sorted(mpn_gc)[n - 1]
|
|
mode = ('Win32', 'x64')
|
|
mpn_f = mpn_gc[config]
|
|
elif nd_gc < n <= nd_32:
|
|
config = sorted(mpn_32)[n - 1 - nd_gc]
|
|
mode = ('Win32', )
|
|
mpn_f = mpn_32[config]
|
|
elif nd_32 < n <= nd_nd:
|
|
config = sorted(mpn_64)[n - 1 - nd_32]
|
|
mode = ('x64', )
|
|
mpn_f = mpn_64[config]
|
|
else:
|
|
print('internal error')
|
|
exit()
|
|
|
|
if mode[0] == 'x64':
|
|
for l in mpn_f[1:]:
|
|
for t in l:
|
|
if t[0].startswith('preinv_'):
|
|
if ('x64' in mode and t[0] == 'preinv_divrem_1'):
|
|
l.remove(t)
|
|
|
|
print(config, mode)
|
|
|
|
if not add_prebuild:
|
|
# generate mpir.h and gmp.h from gmp_h.in
|
|
gmp_h = '''
|
|
#ifdef _WIN32
|
|
# ifdef _WIN64
|
|
# define _LONG_LONG_LIMB 1
|
|
# define GMP_LIMB_BITS 64
|
|
# else
|
|
# define GMP_LIMB_BITS 32
|
|
# endif
|
|
# define __GMP_BITS_PER_MP_LIMB GMP_LIMB_BITS
|
|
# define SIZEOF_MP_LIMB_T (GMP_LIMB_BITS >> 3)
|
|
# define GMP_NAIL_BITS 0
|
|
#endif
|
|
'''
|
|
|
|
try:
|
|
lines = open(join(mpir_dir, 'gmp-h.in'), 'r').readlines()
|
|
except IOError:
|
|
print('error attempting to read from gmp_h.in')
|
|
exit()
|
|
try:
|
|
tfile = join(mpir_dir, 'tmp.h')
|
|
with open(tfile, 'w') as outf:
|
|
first = True
|
|
for line in lines:
|
|
if search('@\w+@', line):
|
|
if first:
|
|
first = False
|
|
outf.writelines(gmp_h)
|
|
else:
|
|
outf.writelines([line])
|
|
|
|
# write result to mpir.h but only overwrite the existing
|
|
# version if this version is different (don't trigger an
|
|
# unnecessary rebuild)
|
|
write_f(tfile, join(mpir_dir, 'mpir.h'))
|
|
write_f(tfile, join(mpir_dir, 'gmp.h'))
|
|
unlink(tfile)
|
|
except IOError:
|
|
print('error attempting to create mpir.h from gmp-h.in')
|
|
exit()
|
|
|
|
# generate config.h
|
|
|
|
try:
|
|
tfile = join(mpir_dir, 'tmp.h')
|
|
with open(tfile, 'w') as outf:
|
|
for i in sorted(mpn_f[5] + mpn_f[6]):
|
|
outf.writelines(['#define HAVE_NATIVE_{0:s} 1\n'.format(i)])
|
|
|
|
append_f(join(build_dir, 'cfg.h'), tfile)
|
|
write_f(tfile, join(mpir_dir, 'config.h'))
|
|
unlink(tfile)
|
|
except IOError:
|
|
print('error attempting to write to {0:s}'.format(tfile))
|
|
exit()
|
|
|
|
# generate longlong.h and copy gmp-mparam.h
|
|
|
|
try:
|
|
li_file = None
|
|
for i in mpn_f[0]:
|
|
if i[0] == 'longlong_inc':
|
|
li_file = join(mpir_dir, join(i[2], r'longlong_inc.h'))
|
|
if i[0] == 'gmp-mparam':
|
|
write_f(join(mpir_dir, join(i[2], 'gmp-mparam.h')), join(mpir_dir, 'gmp-mparam.h'))
|
|
|
|
if not li_file or not exists(li_file):
|
|
print('error attempting to read {0:s}'.format(li_file))
|
|
exit()
|
|
|
|
tfile = join(mpir_dir, 'tmp.h')
|
|
write_f(join(mpir_dir, 'longlong_pre.h'), tfile)
|
|
append_f(li_file, tfile)
|
|
append_f(join(mpir_dir, 'longlong_post.h'), tfile)
|
|
write_f(tfile, join(mpir_dir, 'longlong.h'))
|
|
unlink(tfile)
|
|
except IOError:
|
|
print('error attempting to generate longlong.h')
|
|
exit()
|
|
|
|
# generate the vcxproj and the IDE filter files
|
|
# and add/replace project in the solution file
|
|
|
|
hf_list = ('config.h', 'gmp-impl.h', 'longlong.h', 'mpir.h', 'gmp-mparam.h')
|
|
af_list = sorted(mpn_f[2] + mpn_f[3])
|
|
|
|
proj_name = 'mpir'
|
|
cf = config.replace('\\', '_')
|
|
|
|
# set up DLL build
|
|
guid = '{' + str(uuid1()) + '}'
|
|
vcx_name = 'dll_mpir_' + cf
|
|
vcx_path = 'dll_mpir_' + cf + '\\' + vcx_name + '.vcxproj'
|
|
gen_filter(vcx_path + '.filters', hf_list, c_src_list + cc_src_list + mpn_f[1], af_list)
|
|
gen_vcxproj(proj_name, vcx_path, guid, config, mode, True, False, hf_list, c_src_list + cc_src_list + mpn_f[1], af_list)
|
|
add_proj_to_sln(vcx_name, vcx_path, guid)
|
|
|
|
# set up LIB build
|
|
guid = '{' + str(uuid1()) + '}'
|
|
vcx_name = 'lib_mpir_' + cf
|
|
vcx_path = 'lib_mpir_' + cf + '\\' + vcx_name + '.vcxproj'
|
|
gen_filter(vcx_path + '.filters', hf_list, c_src_list + mpn_f[1], af_list)
|
|
gen_vcxproj(proj_name, vcx_path, guid, config, mode, False, False, hf_list, c_src_list + mpn_f[1], af_list)
|
|
add_proj_to_sln(vcx_name, vcx_path, guid)
|
|
|
|
# C++ library build
|
|
|
|
if add_cpp_lib:
|
|
guid = '{' + str(uuid1()) + '}'
|
|
proj_name = 'mpirxx'
|
|
mode = ('Win32', 'x64')
|
|
vcx_name = 'lib_mpir_cxx'
|
|
vcx_path = 'lib_mpir_cxx\\' + vcx_name + '.vcxproj'
|
|
th = hf_list + ('mpirxx.h',)
|
|
gen_filter(vcx_path + '.filters', th, cc_src_list, '')
|
|
gen_vcxproj(proj_name, vcx_path, guid, config, mode, False, True, th, cc_src_list, '')
|
|
add_proj_to_sln(vcx_name, vcx_path, guid)
|
|
|
|
# the following code is for diagnostic purposes only
|
|
if debug:
|
|
|
|
for x in sorted(mpn_f[0] + mpn_f[1]):
|
|
print(x)
|
|
print()
|
|
for x in sorted(mpn_f[2] + mpn_f[3]):
|
|
print(x)
|
|
print()
|
|
|
|
# mpn_files = dict()
|
|
# mpn_files.update(mpn_32)
|
|
# mpn_files.update(mpn_64)
|
|
for x in mpn_f[config]:
|
|
print(x)
|
|
if False:
|
|
print('1:')
|
|
for y in mpn_files[x][0]:
|
|
print(y)
|
|
print('2:')
|
|
for y in mpn_files[x][1]:
|
|
print(y)
|
|
print('3:')
|
|
for y in mpn_files[x][2]:
|
|
print(y)
|
|
print('4:')
|
|
for y in mpn_files[x][3]:
|
|
print(y)
|
|
print()
|
|
for y in sorted(x[2] + x[3]):
|
|
print(y)
|
|
print()
|
|
print()
|
|
|
|
if debug:
|
|
|
|
mpn_dirs = ('mpn/generic', 'mpn/x86_64w', 'mpn/x86w' )
|
|
|
|
# compile a list of files in directories in 'dl' under root 'r' with extension 'p'
|
|
def findf(r, dl, p):
|
|
l = []
|
|
for d in dl:
|
|
for root, dirs, files in walk(r + d):
|
|
relp = relpath(root, r) # path relative to mpir root directory
|
|
if '.svn' in dirs:
|
|
dirs.remove('.svn') # ignore SVN directories
|
|
if d == '' or root.endswith(build_vc):
|
|
for d in reversed(dirs): # don't scan build.vc10 subdirectories
|
|
dirs.remove(d)
|
|
for f in files:
|
|
if f.endswith(p):
|
|
l += [(tuple(relp.split('\\')), f)]
|
|
return sorted(l)
|
|
|
|
hdr_list = findf(mpir_dir, c_directories, '.h')
|
|
for x in hdr_list:
|
|
print(x)
|
|
print()
|
|
|
|
src_list = findf(mpir_dir, c_directories, '.c')
|
|
for x in src_list:
|
|
print(x)
|
|
print()
|
|
|
|
cpp_list = findf(mpir_dir, ['cpp'], '.cc')
|
|
for x in cpp_list:
|
|
print(x)
|
|
print()
|
|
|
|
gnc_list = findf(mpir_dir + 'mpn/', ['generic'], '.c')
|
|
for x in gnc_list:
|
|
print(x)
|
|
print()
|
|
|
|
w32_list = findf(mpir_dir + 'mpn/', ['x86w'], '.asm')
|
|
for x in w32_list:
|
|
print(x)
|
|
print()
|
|
|
|
x64_list = findf(mpir_dir + 'mpn/', ['x86_64w'], '.asm')
|
|
for x in x64_list:
|
|
print(x)
|
|
print()
|
|
|
|
nd = dict([])
|
|
for d, f in gnc_list:
|
|
n, x = splitext(f)
|
|
nd[n] = nd.get(n, []) + [(d, 'c')]
|
|
for d, f in x64_list:
|
|
n, x = splitext(f)
|
|
nd[n] = nd.get(n, []) + [(d, 'asm')]
|
|
for d, f in w32_list:
|
|
n, x = splitext(f)
|
|
nd[n] = nd.get(n, []) + [(d, 'asm')]
|
|
for x in nd:
|
|
print(x, nd[x])
|