1df3bd252SSatish Balay#!/usr/bin/env python3 20ee81e68SLisandro Dalcin 30ee81e68SLisandro Dalcinimport os 40ee81e68SLisandro Dalcinimport sys 50ee81e68SLisandro Dalcinimport logging 60ee81e68SLisandro Dalcinsys.path.insert(0, os.path.abspath(os.path.dirname(__file__))) 77b8851e6SJed Brownfrom collections import defaultdict 80ee81e68SLisandro Dalcin 9b75c6efcSBarry SmithSKIPDIRS = set('benchmarks build mex-scripts tests tutorials'.split()) # Skip these during the build 10dc0529c6SBarry Smith 114bfab837SSebastian Grimbergdef pathsplit(pkg_dir, path): 12dc0529c6SBarry Smith """Recursively split a path, returns a tuple""" 13dc0529c6SBarry Smith stem, basename = os.path.split(path) 144bfab837SSebastian Grimberg if stem == '' or stem == pkg_dir: 15dc0529c6SBarry Smith return (basename,) 16dc0529c6SBarry Smith if stem == path: # fixed point, likely '/' 1751de3720SJed Brown return (None,) 184bfab837SSebastian Grimberg return pathsplit(pkg_dir, stem) + (basename,) 19dc0529c6SBarry Smith 20c0558f20SBarry Smithdef getlangext(name): 21c0558f20SBarry Smith """Returns everything after the first . in the filename, including the .""" 22c0558f20SBarry Smith file = os.path.basename(name) 23c0558f20SBarry Smith loc = file.find('.') 24c0558f20SBarry Smith if loc > -1: return file[loc:] 25c0558f20SBarry Smith else: return '' 26c0558f20SBarry Smith 27c0558f20SBarry Smithdef getlangsplit(name): 28c0558f20SBarry Smith """Returns everything before the first . in the filename, excluding the .""" 29c0558f20SBarry Smith file = os.path.basename(name) 30c0558f20SBarry Smith loc = file.find('.') 31c0558f20SBarry Smith if loc > -1: return os.path.join(os.path.dirname(name),file[:loc]) 32c0558f20SBarry Smith raise RuntimeError("No . in filename") 33c0558f20SBarry Smith 34dc0529c6SBarry Smithdef stripsplit(line): 35dc0529c6SBarry Smith return line[len('#requires'):].replace("'","").split() 36dc0529c6SBarry Smith 379bb2fd42SSatish Balaydef parse_makefile(fn, out=None): 389bb2fd42SSatish Balay if out is None: 399bb2fd42SSatish Balay out = {} 409bb2fd42SSatish Balay with open(fn) as f: 419bb2fd42SSatish Balay for l in f: 429bb2fd42SSatish Balay if "=" in l: 439bb2fd42SSatish Balay a,b = l.split("=", 1) 449bb2fd42SSatish Balay out[a.strip()] = b.strip() 459bb2fd42SSatish Balay return out 469bb2fd42SSatish Balay 4734b254c5SRichard Tran MillsPetscPKGS = 'sys vec mat dm ksp snes ts tao ml'.split() 48c0558f20SBarry Smith# the key is actually the language suffix, it won't work for suffixes such as 'kokkos.cxx' so use an _ and replace the _ as needed with . 49*d52a580bSJunchao ZhangLANGS = dict(kokkos_cxx='KOKKOS', hip_cxx='HIP', sycl_cxx='SYCL', raja_cxx='RAJA', c='C', cxx='CXX', cpp='CPP', cu='CU', F='F', F90='F90') 500ee81e68SLisandro Dalcin 510ee81e68SLisandro Dalcinclass debuglogger(object): 520ee81e68SLisandro Dalcin def __init__(self, log): 530ee81e68SLisandro Dalcin self._log = log 540ee81e68SLisandro Dalcin 550ee81e68SLisandro Dalcin def write(self, string): 560ee81e68SLisandro Dalcin self._log.debug(string) 570ee81e68SLisandro Dalcin 580ee81e68SLisandro Dalcinclass Petsc(object): 59a95bf149SBarry Smith def __init__(self, petsc_dir=None, petsc_arch=None, pkg_dir=None, pkg_name=None, pkg_arch=None, pkg_pkgs=None): 600ee81e68SLisandro Dalcin if petsc_dir is None: 610ee81e68SLisandro Dalcin petsc_dir = os.environ.get('PETSC_DIR') 620ee81e68SLisandro Dalcin if petsc_dir is None: 630ee81e68SLisandro Dalcin try: 64af0996ceSBarry Smith petsc_dir = parse_makefile(os.path.join('lib','petsc','conf', 'petscvariables')).get('PETSC_DIR') 650ee81e68SLisandro Dalcin finally: 660ee81e68SLisandro Dalcin if petsc_dir is None: 670ee81e68SLisandro Dalcin raise RuntimeError('Could not determine PETSC_DIR, please set in environment') 680ee81e68SLisandro Dalcin if petsc_arch is None: 690ee81e68SLisandro Dalcin petsc_arch = os.environ.get('PETSC_ARCH') 700ee81e68SLisandro Dalcin if petsc_arch is None: 710ee81e68SLisandro Dalcin try: 72af0996ceSBarry Smith petsc_arch = parse_makefile(os.path.join(petsc_dir, 'lib','petsc','conf', 'petscvariables')).get('PETSC_ARCH') 730ee81e68SLisandro Dalcin finally: 740ee81e68SLisandro Dalcin if petsc_arch is None: 750ee81e68SLisandro Dalcin raise RuntimeError('Could not determine PETSC_ARCH, please set in environment') 76add6df95SStefano Zampini self.petsc_dir = os.path.normpath(petsc_dir) 77add6df95SStefano Zampini self.petsc_arch = petsc_arch.rstrip(os.sep) 78add6df95SStefano Zampini self.pkg_dir = pkg_dir 79add6df95SStefano Zampini self.pkg_name = pkg_name 80c4bccdb5SStefano Zampini self.pkg_arch = pkg_arch 81add6df95SStefano Zampini if self.pkg_dir is None: 82add6df95SStefano Zampini self.pkg_dir = petsc_dir 83add6df95SStefano Zampini self.pkg_name = 'petsc' 84c4bccdb5SStefano Zampini self.pkg_arch = self.petsc_arch 85add6df95SStefano Zampini if self.pkg_name is None: 86add6df95SStefano Zampini self.pkg_name = os.path.basename(os.path.normpath(self.pkg_dir)) 87c4bccdb5SStefano Zampini if self.pkg_arch is None: 88c4bccdb5SStefano Zampini self.pkg_arch = self.petsc_arch 8947fd361eSStefano Zampini self.pkg_pkgs = PetscPKGS 9047fd361eSStefano Zampini if pkg_pkgs is not None: 918d5d1ea2SBarry Smith if pkg_pkgs.find(',') > 0: npkgs = set(pkg_pkgs.split(',')) 928d5d1ea2SBarry Smith else: npkgs = set(pkg_pkgs.split(' ')) 938d5d1ea2SBarry Smith self.pkg_pkgs += list(npkgs - set(self.pkg_pkgs)) 9447fd361eSStefano Zampini self.read_conf() 958e69c5ecSJed Brown try: 96add6df95SStefano Zampini logging.basicConfig(filename=self.pkg_arch_path('lib',self.pkg_name,'conf', 'gmake.log'), level=logging.DEBUG) 978e69c5ecSJed Brown except IOError: 988e69c5ecSJed Brown # Disable logging if path is not writeable (e.g., prefix install) 998e69c5ecSJed Brown logging.basicConfig(filename='/dev/null', level=logging.DEBUG) 1000ee81e68SLisandro Dalcin self.log = logging.getLogger('gmakegen') 1010ee81e68SLisandro Dalcin self.gendeps = [] 1020ee81e68SLisandro Dalcin 1030ee81e68SLisandro Dalcin def arch_path(self, *args): 1040ee81e68SLisandro Dalcin return os.path.join(self.petsc_dir, self.petsc_arch, *args) 1050ee81e68SLisandro Dalcin 106add6df95SStefano Zampini def pkg_arch_path(self, *args): 107c4bccdb5SStefano Zampini return os.path.join(self.pkg_dir, self.pkg_arch, *args) 108add6df95SStefano Zampini 1090ee81e68SLisandro Dalcin def read_conf(self): 1100ee81e68SLisandro Dalcin self.conf = dict() 111021a2b48SJed Brown with open(self.arch_path('include', 'petscconf.h')) as petscconf_h: 112021a2b48SJed Brown for line in petscconf_h: 1130ee81e68SLisandro Dalcin if line.startswith('#define '): 1140ee81e68SLisandro Dalcin define = line[len('#define '):] 1150ee81e68SLisandro Dalcin space = define.find(' ') 1160ee81e68SLisandro Dalcin key = define[:space] 1170ee81e68SLisandro Dalcin val = define[space+1:] 1180ee81e68SLisandro Dalcin self.conf[key] = val 119af0996ceSBarry Smith self.conf.update(parse_makefile(self.arch_path('lib','petsc','conf', 'petscvariables'))) 12047fd361eSStefano Zampini # allow parsing package additional configurations (if any) 12147fd361eSStefano Zampini if self.pkg_name != 'petsc' : 12247fd361eSStefano Zampini f = self.pkg_arch_path('include', self.pkg_name + 'conf.h') 12347fd361eSStefano Zampini if os.path.isfile(f): 124021a2b48SJed Brown with open(self.pkg_arch_path('include', self.pkg_name + 'conf.h')) as pkg_conf_h: 125021a2b48SJed Brown for line in pkg_conf_h: 12647fd361eSStefano Zampini if line.startswith('#define '): 12747fd361eSStefano Zampini define = line[len('#define '):] 12847fd361eSStefano Zampini space = define.find(' ') 12947fd361eSStefano Zampini key = define[:space] 13047fd361eSStefano Zampini val = define[space+1:] 13147fd361eSStefano Zampini self.conf[key] = val 13247fd361eSStefano Zampini f = self.pkg_arch_path('lib',self.pkg_name,'conf', self.pkg_name + 'variables') 13347fd361eSStefano Zampini if os.path.isfile(f): 13447fd361eSStefano Zampini self.conf.update(parse_makefile(self.pkg_arch_path('lib',self.pkg_name,'conf', self.pkg_name + 'variables'))) 135fbf9dbe5SBarry Smith self.have_fortran = int(self.conf.get('PETSC_USE_FORTRAN_BINDINGS', '0')) 1360ee81e68SLisandro Dalcin 1370ee81e68SLisandro Dalcin def inconf(self, key, val): 1380ee81e68SLisandro Dalcin if key in ['package', 'function', 'define']: 1390ee81e68SLisandro Dalcin return self.conf.get(val) 1400ee81e68SLisandro Dalcin elif key == 'precision': 1410ee81e68SLisandro Dalcin return val == self.conf['PETSC_PRECISION'] 1420ee81e68SLisandro Dalcin elif key == 'scalar': 1430ee81e68SLisandro Dalcin return val == self.conf['PETSC_SCALAR'] 1440ee81e68SLisandro Dalcin elif key == 'language': 1450ee81e68SLisandro Dalcin return val == self.conf['PETSC_LANGUAGE'] 1460ee81e68SLisandro Dalcin raise RuntimeError('Unknown conf check: %s %s' % (key, val)) 1470ee81e68SLisandro Dalcin 1480ee81e68SLisandro Dalcin def relpath(self, root, src): 149add6df95SStefano Zampini return os.path.relpath(os.path.join(root, src), self.pkg_dir) 1500ee81e68SLisandro Dalcin 1515a7ab478SBarry Smith def get_sources_from_files(self, files): 1520ee81e68SLisandro Dalcin """Return dict {lang: list_of_source_files}""" 1530ee81e68SLisandro Dalcin source = dict() 1540ee81e68SLisandro Dalcin for lang, sourcelang in LANGS.items(): 1555a7ab478SBarry Smith source[lang] = [f for f in files if f.endswith('.'+lang.replace('_','.'))] 1565a7ab478SBarry Smith files = [f for f in files if not f.endswith('.'+lang.replace('_','.'))] 1570ee81e68SLisandro Dalcin return source 1580ee81e68SLisandro Dalcin 159ce78bad3SBarry Smith def gen_pkg(self, pkg_arch, pkg): 16013d87ab8SBarry Smith from itertools import chain 1610ee81e68SLisandro Dalcin pkgsrcs = dict() 1620ee81e68SLisandro Dalcin for lang in LANGS: 1630ee81e68SLisandro Dalcin pkgsrcs[lang] = [] 164ce78bad3SBarry Smith for root, dirs, files in chain.from_iterable(os.walk(path) for path in [os.path.join(self.pkg_dir, 'src', pkg),os.path.join(self.pkg_dir, self.pkg_arch, 'ftn', pkg)]): 1654bfab837SSebastian Grimberg if SKIPDIRS.intersection(pathsplit(self.pkg_dir, root)): continue 1666dd63270SBarry Smith if not self.have_fortran and os.path.basename(root).find('ftn-') > -1: continue 16709a6cbfcSBernhard M. Wiedemann dirs.sort() 168a95bf149SBarry Smith dirs[:] = list(set(dirs).difference(SKIPDIRS)) 16909a6cbfcSBernhard M. Wiedemann files.sort() 1700ee81e68SLisandro Dalcin makefile = os.path.join(root,'makefile') 17113d87ab8SBarry Smith if os.path.isfile(makefile): 172e5ad84adSJunchao Zhang # Common case is one <key, value> pair per line, but we allow multi-values in format of '#requires<key> val1 val2 .. valn', e.g., '#requiresprecision single double' 173e5ad84adSJunchao Zhang # Conditions across lines are AND'ed together. 174e5ad84adSJunchao Zhang # Conditions within a line are OR'ed together, in other words, the condition is met if any value in <val1, ..., valn> is fulfilled. 175021a2b48SJed Brown with open(makefile) as mklines: 1760ee81e68SLisandro Dalcin conditions = set(tuple(stripsplit(line)) for line in mklines if line.startswith('#requires')) 177e5ad84adSJunchao Zhang allmet = True 178e5ad84adSJunchao Zhang for cond in conditions: 179e5ad84adSJunchao Zhang key, *vals = cond 180e5ad84adSJunchao Zhang if not any(self.inconf(key, val) for val in vals): # none of the values for this key is fulfilled 181e5ad84adSJunchao Zhang allmet = False 182e5ad84adSJunchao Zhang break 183e5ad84adSJunchao Zhang if not allmet: 1840ee81e68SLisandro Dalcin dirs[:] = [] 1850ee81e68SLisandro Dalcin continue 1860ee81e68SLisandro Dalcin allsource = [] 187ce78bad3SBarry Smith if root.find('/ftn/') > -1: nroot = ''.join(root.rsplit(pkg_arch + '/', 1)) 188ce78bad3SBarry Smith else: nroot = root 1890ee81e68SLisandro Dalcin def mkrel(src): 190ce78bad3SBarry Smith return self.relpath(nroot, src) 19113d87ab8SBarry Smith if files: 1925a7ab478SBarry Smith source = self.get_sources_from_files(files) 1930ee81e68SLisandro Dalcin for lang, s in source.items(): 1942b757757SJed Brown pkgsrcs[lang] += [mkrel(t) for t in s] 19513d87ab8SBarry Smith if os.path.isfile(makefile): self.gendeps.append(self.relpath(root, 'makefile')) 1960ee81e68SLisandro Dalcin return pkgsrcs 1970ee81e68SLisandro Dalcin 198ce78bad3SBarry Smith def gen_gnumake(self, pkg_arch,fd): 1990ee81e68SLisandro Dalcin def write(stem, srcs): 2000ee81e68SLisandro Dalcin for lang in LANGS: 20149316b25SLars Bilke fd.write('%(stem)s.%(lang)s := %(srcs)s\n' % dict(stem=stem, lang=lang.replace('_','.'), srcs=' '.join(sorted(srcs[lang])))) 20247fd361eSStefano Zampini for pkg in self.pkg_pkgs: 203ce78bad3SBarry Smith srcs = self.gen_pkg(pkg_arch,pkg) 204b0790570SJed Brown write('srcs-' + pkg, srcs) 2050ee81e68SLisandro Dalcin return self.gendeps 2060ee81e68SLisandro Dalcin 2070ee81e68SLisandro Dalcin def gen_ninja(self, fd): 2080ee81e68SLisandro Dalcin libobjs = [] 20947fd361eSStefano Zampini for pkg in self.pkg_pkgs: 2100ee81e68SLisandro Dalcin srcs = self.gen_pkg(pkg) 2110ee81e68SLisandro Dalcin for lang in LANGS: 2120ee81e68SLisandro Dalcin for src in srcs[lang]: 2130ee81e68SLisandro Dalcin obj = '$objdir/%s.o' % src 214add6df95SStefano Zampini fd.write('build %(obj)s : %(lang)s_COMPILE %(src)s\n' % dict(obj=obj, lang=lang.upper(), src=os.path.join(self.pkg_dir,src))) 2150ee81e68SLisandro Dalcin libobjs.append(obj) 2160ee81e68SLisandro Dalcin fd.write('\n') 2170ee81e68SLisandro Dalcin fd.write('build $libdir/libpetsc.so : %s_LINK_SHARED %s\n\n' % ('CF'[self.have_fortran], ' '.join(libobjs))) 2180ee81e68SLisandro Dalcin fd.write('build petsc : phony || $libdir/libpetsc.so\n\n') 2190ee81e68SLisandro Dalcin 220ce78bad3SBarry Smithdef WriteGnuMake(pkg_arch,petsc): 221add6df95SStefano Zampini arch_files = petsc.pkg_arch_path('lib',petsc.pkg_name,'conf', 'files') 222021a2b48SJed Brown with open(arch_files, 'w') as fd: 223ce78bad3SBarry Smith gendeps = petsc.gen_gnumake(pkg_arch,fd) 2240ee81e68SLisandro Dalcin fd.write('\n') 2250ee81e68SLisandro Dalcin fd.write('# Dependency to regenerate this file\n') 226add6df95SStefano Zampini fd.write('%s : %s %s\n' % (os.path.relpath(arch_files, petsc.pkg_dir), 227add6df95SStefano Zampini os.path.relpath(__file__, os.path.realpath(petsc.pkg_dir)), 2280ee81e68SLisandro Dalcin ' '.join(gendeps))) 2290ee81e68SLisandro Dalcin fd.write('\n') 2300ee81e68SLisandro Dalcin fd.write('# Dummy dependencies in case makefiles are removed\n') 2310ee81e68SLisandro Dalcin fd.write(''.join([dep + ':\n' for dep in gendeps])) 2320ee81e68SLisandro Dalcin 2330ee81e68SLisandro Dalcindef WriteNinja(petsc): 2340ee81e68SLisandro Dalcin conf = dict() 235af0996ceSBarry Smith parse_makefile(os.path.join(petsc.petsc_dir, 'lib', 'petsc','conf', 'variables'), conf) 236af0996ceSBarry Smith parse_makefile(petsc.arch_path('lib','petsc','conf', 'petscvariables'), conf) 2370ee81e68SLisandro Dalcin build_ninja = petsc.arch_path('build.ninja') 238021a2b48SJed Brown with open(build_ninja, 'w') as fd: 2390ee81e68SLisandro Dalcin fd.write('objdir = obj-ninja\n') 2400ee81e68SLisandro Dalcin fd.write('libdir = lib\n') 2410ee81e68SLisandro Dalcin fd.write('c_compile = %(PCC)s\n' % conf) 2420ee81e68SLisandro Dalcin fd.write('c_flags = %(PETSC_CC_INCLUDES)s %(PCC_FLAGS)s %(CCPPFLAGS)s\n' % conf) 2430ee81e68SLisandro Dalcin fd.write('c_link = %(PCC_LINKER)s\n' % conf) 2440ee81e68SLisandro Dalcin fd.write('c_link_flags = %(PCC_LINKER_FLAGS)s\n' % conf) 2450ee81e68SLisandro Dalcin if petsc.have_fortran: 2460ee81e68SLisandro Dalcin fd.write('f_compile = %(FC)s\n' % conf) 2470ee81e68SLisandro Dalcin fd.write('f_flags = %(PETSC_FC_INCLUDES)s %(FC_FLAGS)s %(FCPPFLAGS)s\n' % conf) 2480ee81e68SLisandro Dalcin fd.write('f_link = %(FC_LINKER)s\n' % conf) 2490ee81e68SLisandro Dalcin fd.write('f_link_flags = %(FC_LINKER_FLAGS)s\n' % conf) 2500ee81e68SLisandro Dalcin fd.write('petsc_external_lib = %(PETSC_EXTERNAL_LIB_BASIC)s\n' % conf) 2510ee81e68SLisandro Dalcin fd.write('python = %(PYTHON)s\n' % conf) 2520ee81e68SLisandro Dalcin fd.write('\n') 2530ee81e68SLisandro Dalcin fd.write('rule C_COMPILE\n' 2540ee81e68SLisandro Dalcin ' command = $c_compile -MMD -MF $out.d $c_flags -c $in -o $out\n' 2550ee81e68SLisandro Dalcin ' description = CC $out\n' 2560ee81e68SLisandro Dalcin ' depfile = $out.d\n' 2570ee81e68SLisandro Dalcin # ' deps = gcc\n') # 'gcc' is default, 'msvc' only recognized by newer versions of ninja 2580ee81e68SLisandro Dalcin '\n') 2590ee81e68SLisandro Dalcin fd.write('rule C_LINK_SHARED\n' 2600ee81e68SLisandro Dalcin ' command = $c_link $c_link_flags -shared -o $out $in $petsc_external_lib\n' 2610ee81e68SLisandro Dalcin ' description = CLINK_SHARED $out\n' 2620ee81e68SLisandro Dalcin '\n') 2630ee81e68SLisandro Dalcin if petsc.have_fortran: 2640ee81e68SLisandro Dalcin fd.write('rule F_COMPILE\n' 2650ee81e68SLisandro Dalcin ' command = $f_compile -MMD -MF $out.d $f_flags -c $in -o $out\n' 2660ee81e68SLisandro Dalcin ' description = FC $out\n' 2670ee81e68SLisandro Dalcin ' depfile = $out.d\n' 2680ee81e68SLisandro Dalcin '\n') 2690ee81e68SLisandro Dalcin fd.write('rule F_LINK_SHARED\n' 2700ee81e68SLisandro Dalcin ' command = $f_link $f_link_flags -shared -o $out $in $petsc_external_lib\n' 2710ee81e68SLisandro Dalcin ' description = FLINK_SHARED $out\n' 2720ee81e68SLisandro Dalcin '\n') 2730ee81e68SLisandro Dalcin fd.write('rule GEN_NINJA\n' 2740ee81e68SLisandro Dalcin ' command = $python $in --output=ninja\n' 2750ee81e68SLisandro Dalcin ' generator = 1\n' 2760ee81e68SLisandro Dalcin '\n') 2770ee81e68SLisandro Dalcin petsc.gen_ninja(fd) 2780ee81e68SLisandro Dalcin fd.write('\n') 2790ee81e68SLisandro Dalcin fd.write('build %s : GEN_NINJA | %s %s %s %s\n' % (build_ninja, 2800ee81e68SLisandro Dalcin os.path.abspath(__file__), 281af0996ceSBarry Smith os.path.join(petsc.petsc_dir, 'lib','petsc','conf', 'variables'), 282af0996ceSBarry Smith petsc.arch_path('lib','petsc','conf', 'petscvariables'), 283add6df95SStefano Zampini ' '.join(os.path.join(petsc.pkg_dir, dep) for dep in petsc.gendeps))) 2840ee81e68SLisandro Dalcin 285a95bf149SBarry Smithdef main(petsc_dir=None, petsc_arch=None, pkg_dir=None, pkg_name=None, pkg_arch=None, pkg_pkgs=None, output=None): 286a95bf149SBarry Smith petsc = Petsc(petsc_dir=petsc_dir, petsc_arch=petsc_arch, pkg_dir=pkg_dir, pkg_name=pkg_name, pkg_arch=pkg_arch, pkg_pkgs=pkg_pkgs) 287ce78bad3SBarry Smith # Use pkg_arch in case petsc_arch is empty (needed by SLEPc) 288ce78bad3SBarry Smith WriteGnuMake(petsc_arch if petsc_arch else pkg_arch,petsc) 2890ee81e68SLisandro Dalcin 2900ee81e68SLisandro Dalcinif __name__ == '__main__': 2910ee81e68SLisandro Dalcin import optparse 2920ee81e68SLisandro Dalcin parser = optparse.OptionParser() 2930ee81e68SLisandro Dalcin parser.add_option('--petsc-arch', help='Set PETSC_ARCH different from environment', default=os.environ.get('PETSC_ARCH')) 294add6df95SStefano Zampini parser.add_option('--pkg-dir', help='Set the directory of the package (different from PETSc) you want to generate the makefile rules for', default=None) 295add6df95SStefano Zampini parser.add_option('--pkg-name', help='Set the name of the package you want to generate the makefile rules for', default=None) 296c4bccdb5SStefano Zampini parser.add_option('--pkg-arch', help='Set the package arch name you want to generate the makefile rules for', default=None) 29747fd361eSStefano Zampini parser.add_option('--pkg-pkgs', help='Set the package folders (comma separated list, different from the usual sys,vec,mat etc) you want to generate the makefile rules for', default=None) 2980ee81e68SLisandro Dalcin parser.add_option('--output', help='Location to write output file', default=None) 2990ee81e68SLisandro Dalcin opts, extra_args = parser.parse_args() 3000ee81e68SLisandro Dalcin if extra_args: 3010ee81e68SLisandro Dalcin import sys 3020ee81e68SLisandro Dalcin sys.stderr.write('Unknown arguments: %s\n' % ' '.join(extra_args)) 3030ee81e68SLisandro Dalcin exit(1) 304a95bf149SBarry Smith main(petsc_arch=opts.petsc_arch, pkg_dir=opts.pkg_dir, pkg_name=opts.pkg_name, pkg_arch=opts.pkg_arch, pkg_pkgs=opts.pkg_pkgs, output=opts.output) 305