xref: /petsc/config/gmakegen.py (revision 9bb2fd428d7e9961226e662080eef94e9e67e4a6)
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
9b892d579SJed BrownAUTODIRS = set('ftn-auto ftn-custom f90-custom ftn-auto-interfaces'.split()) # Automatically recurse into these, if they exist
10b75c6efcSBarry SmithSKIPDIRS = set('benchmarks build mex-scripts tests tutorials'.split())       # Skip these during the build
11dc0529c6SBarry Smith
124bfab837SSebastian Grimbergdef pathsplit(pkg_dir, path):
13dc0529c6SBarry Smith    """Recursively split a path, returns a tuple"""
14dc0529c6SBarry Smith    stem, basename = os.path.split(path)
154bfab837SSebastian Grimberg    if stem == '' or stem == pkg_dir:
16dc0529c6SBarry Smith        return (basename,)
17dc0529c6SBarry Smith    if stem == path: # fixed point, likely '/'
1851de3720SJed Brown        return (None,)
194bfab837SSebastian Grimberg    return pathsplit(pkg_dir, stem) + (basename,)
20dc0529c6SBarry Smith
21c0558f20SBarry Smithdef getlangext(name):
22c0558f20SBarry Smith    """Returns everything after the first . in the filename, including the ."""
23c0558f20SBarry Smith    file = os.path.basename(name)
24c0558f20SBarry Smith    loc = file.find('.')
25c0558f20SBarry Smith    if loc > -1: return file[loc:]
26c0558f20SBarry Smith    else: return ''
27c0558f20SBarry Smith
28c0558f20SBarry Smithdef getlangsplit(name):
29c0558f20SBarry Smith    """Returns everything before the first . in the filename, excluding the ."""
30c0558f20SBarry Smith    file = os.path.basename(name)
31c0558f20SBarry Smith    loc = file.find('.')
32c0558f20SBarry Smith    if loc > -1: return os.path.join(os.path.dirname(name),file[:loc])
33c0558f20SBarry Smith    raise RuntimeError("No . in filename")
34c0558f20SBarry Smith
35dc0529c6SBarry Smithdef stripsplit(line):
36dc0529c6SBarry Smith  return line[len('#requires'):].replace("'","").split()
37dc0529c6SBarry Smith
38*9bb2fd42SSatish Balaydef parse_makefile(fn, out=None):
39*9bb2fd42SSatish Balay    if out is None:
40*9bb2fd42SSatish Balay        out = {}
41*9bb2fd42SSatish Balay    with open(fn) as f:
42*9bb2fd42SSatish Balay        for l in f:
43*9bb2fd42SSatish Balay            if "=" in l:
44*9bb2fd42SSatish Balay                a,b = l.split("=", 1)
45*9bb2fd42SSatish Balay                out[a.strip()] = b.strip()
46*9bb2fd42SSatish Balay    return out
47*9bb2fd42SSatish Balay
4847fd361eSStefano ZampiniPetscPKGS = 'sys vec mat dm ksp snes ts tao'.split()
49c0558f20SBarry 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 .
505a7ab478SBarry SmithLANGS = dict(kokkos_cxx='KOKKOS', hip_cpp='HIP', sycl_cxx='SYCL', raja_cxx='RAJA', c='C', cxx='CXX', cpp='CPP', cu='CU', F='F', F90='F90')
510ee81e68SLisandro Dalcin
520ee81e68SLisandro Dalcinclass debuglogger(object):
530ee81e68SLisandro Dalcin    def __init__(self, log):
540ee81e68SLisandro Dalcin        self._log = log
550ee81e68SLisandro Dalcin
560ee81e68SLisandro Dalcin    def write(self, string):
570ee81e68SLisandro Dalcin        self._log.debug(string)
580ee81e68SLisandro Dalcin
590ee81e68SLisandro Dalcinclass Petsc(object):
60a95bf149SBarry Smith    def __init__(self, petsc_dir=None, petsc_arch=None, pkg_dir=None, pkg_name=None, pkg_arch=None, pkg_pkgs=None):
610ee81e68SLisandro Dalcin        if petsc_dir is None:
620ee81e68SLisandro Dalcin            petsc_dir = os.environ.get('PETSC_DIR')
630ee81e68SLisandro Dalcin            if petsc_dir is None:
640ee81e68SLisandro Dalcin                try:
65af0996ceSBarry Smith                    petsc_dir = parse_makefile(os.path.join('lib','petsc','conf', 'petscvariables')).get('PETSC_DIR')
660ee81e68SLisandro Dalcin                finally:
670ee81e68SLisandro Dalcin                    if petsc_dir is None:
680ee81e68SLisandro Dalcin                        raise RuntimeError('Could not determine PETSC_DIR, please set in environment')
690ee81e68SLisandro Dalcin        if petsc_arch is None:
700ee81e68SLisandro Dalcin            petsc_arch = os.environ.get('PETSC_ARCH')
710ee81e68SLisandro Dalcin            if petsc_arch is None:
720ee81e68SLisandro Dalcin                try:
73af0996ceSBarry Smith                    petsc_arch = parse_makefile(os.path.join(petsc_dir, 'lib','petsc','conf', 'petscvariables')).get('PETSC_ARCH')
740ee81e68SLisandro Dalcin                finally:
750ee81e68SLisandro Dalcin                    if petsc_arch is None:
760ee81e68SLisandro Dalcin                        raise RuntimeError('Could not determine PETSC_ARCH, please set in environment')
77add6df95SStefano Zampini        self.petsc_dir = os.path.normpath(petsc_dir)
78add6df95SStefano Zampini        self.petsc_arch = petsc_arch.rstrip(os.sep)
79add6df95SStefano Zampini        self.pkg_dir = pkg_dir
80add6df95SStefano Zampini        self.pkg_name = pkg_name
81c4bccdb5SStefano Zampini        self.pkg_arch = pkg_arch
82add6df95SStefano Zampini        if self.pkg_dir is None:
83add6df95SStefano Zampini          self.pkg_dir = petsc_dir
84add6df95SStefano Zampini          self.pkg_name = 'petsc'
85c4bccdb5SStefano Zampini          self.pkg_arch = self.petsc_arch
86add6df95SStefano Zampini        if self.pkg_name is None:
87add6df95SStefano Zampini          self.pkg_name = os.path.basename(os.path.normpath(self.pkg_dir))
88c4bccdb5SStefano Zampini        if self.pkg_arch is None:
89c4bccdb5SStefano Zampini          self.pkg_arch = self.petsc_arch
9047fd361eSStefano Zampini        self.pkg_pkgs = PetscPKGS
9147fd361eSStefano Zampini        if pkg_pkgs is not None:
928d5d1ea2SBarry Smith          if pkg_pkgs.find(',') > 0: npkgs = set(pkg_pkgs.split(','))
938d5d1ea2SBarry Smith          else: npkgs = set(pkg_pkgs.split(' '))
948d5d1ea2SBarry Smith          self.pkg_pkgs += list(npkgs - set(self.pkg_pkgs))
9547fd361eSStefano Zampini        self.read_conf()
968e69c5ecSJed Brown        try:
97add6df95SStefano Zampini            logging.basicConfig(filename=self.pkg_arch_path('lib',self.pkg_name,'conf', 'gmake.log'), level=logging.DEBUG)
988e69c5ecSJed Brown        except IOError:
998e69c5ecSJed Brown            # Disable logging if path is not writeable (e.g., prefix install)
1008e69c5ecSJed Brown            logging.basicConfig(filename='/dev/null', level=logging.DEBUG)
1010ee81e68SLisandro Dalcin        self.log = logging.getLogger('gmakegen')
1020ee81e68SLisandro Dalcin        self.gendeps = []
1030ee81e68SLisandro Dalcin
1040ee81e68SLisandro Dalcin    def arch_path(self, *args):
1050ee81e68SLisandro Dalcin        return os.path.join(self.petsc_dir, self.petsc_arch, *args)
1060ee81e68SLisandro Dalcin
107add6df95SStefano Zampini    def pkg_arch_path(self, *args):
108c4bccdb5SStefano Zampini        return os.path.join(self.pkg_dir, self.pkg_arch, *args)
109add6df95SStefano Zampini
1100ee81e68SLisandro Dalcin    def read_conf(self):
1110ee81e68SLisandro Dalcin        self.conf = dict()
112021a2b48SJed Brown        with open(self.arch_path('include', 'petscconf.h')) as petscconf_h:
113021a2b48SJed Brown            for line in petscconf_h:
1140ee81e68SLisandro Dalcin                if line.startswith('#define '):
1150ee81e68SLisandro Dalcin                    define = line[len('#define '):]
1160ee81e68SLisandro Dalcin                    space = define.find(' ')
1170ee81e68SLisandro Dalcin                    key = define[:space]
1180ee81e68SLisandro Dalcin                    val = define[space+1:]
1190ee81e68SLisandro Dalcin                    self.conf[key] = val
120af0996ceSBarry Smith        self.conf.update(parse_makefile(self.arch_path('lib','petsc','conf', 'petscvariables')))
12147fd361eSStefano Zampini        # allow parsing package additional configurations (if any)
12247fd361eSStefano Zampini        if self.pkg_name != 'petsc' :
12347fd361eSStefano Zampini            f = self.pkg_arch_path('include', self.pkg_name + 'conf.h')
12447fd361eSStefano Zampini            if os.path.isfile(f):
125021a2b48SJed Brown                with open(self.pkg_arch_path('include', self.pkg_name + 'conf.h')) as pkg_conf_h:
126021a2b48SJed Brown                    for line in pkg_conf_h:
12747fd361eSStefano Zampini                        if line.startswith('#define '):
12847fd361eSStefano Zampini                            define = line[len('#define '):]
12947fd361eSStefano Zampini                            space = define.find(' ')
13047fd361eSStefano Zampini                            key = define[:space]
13147fd361eSStefano Zampini                            val = define[space+1:]
13247fd361eSStefano Zampini                            self.conf[key] = val
13347fd361eSStefano Zampini            f = self.pkg_arch_path('lib',self.pkg_name,'conf', self.pkg_name + 'variables')
13447fd361eSStefano Zampini            if os.path.isfile(f):
13547fd361eSStefano Zampini                self.conf.update(parse_makefile(self.pkg_arch_path('lib',self.pkg_name,'conf', self.pkg_name + 'variables')))
136fbf9dbe5SBarry Smith        self.have_fortran = int(self.conf.get('PETSC_USE_FORTRAN_BINDINGS', '0'))
1370ee81e68SLisandro Dalcin
1380ee81e68SLisandro Dalcin    def inconf(self, key, val):
1390ee81e68SLisandro Dalcin        if key in ['package', 'function', 'define']:
1400ee81e68SLisandro Dalcin            return self.conf.get(val)
1410ee81e68SLisandro Dalcin        elif key == 'precision':
1420ee81e68SLisandro Dalcin            return val == self.conf['PETSC_PRECISION']
1430ee81e68SLisandro Dalcin        elif key == 'scalar':
1440ee81e68SLisandro Dalcin            return val == self.conf['PETSC_SCALAR']
1450ee81e68SLisandro Dalcin        elif key == 'language':
1460ee81e68SLisandro Dalcin            return val == self.conf['PETSC_LANGUAGE']
1470ee81e68SLisandro Dalcin        raise RuntimeError('Unknown conf check: %s %s' % (key, val))
1480ee81e68SLisandro Dalcin
1490ee81e68SLisandro Dalcin    def relpath(self, root, src):
150add6df95SStefano Zampini        return os.path.relpath(os.path.join(root, src), self.pkg_dir)
1510ee81e68SLisandro Dalcin
1525a7ab478SBarry Smith    def get_sources_from_files(self, files):
1530ee81e68SLisandro Dalcin        """Return dict {lang: list_of_source_files}"""
1540ee81e68SLisandro Dalcin        source = dict()
1550ee81e68SLisandro Dalcin        for lang, sourcelang in LANGS.items():
1565a7ab478SBarry Smith            source[lang] = [f for f in files if f.endswith('.'+lang.replace('_','.'))]
1575a7ab478SBarry Smith            files = [f for f in files if not f.endswith('.'+lang.replace('_','.'))]
1580ee81e68SLisandro Dalcin        return source
1590ee81e68SLisandro Dalcin
1600ee81e68SLisandro Dalcin    def gen_pkg(self, pkg):
16113d87ab8SBarry Smith        from itertools import chain
1620ee81e68SLisandro Dalcin        pkgsrcs = dict()
1630ee81e68SLisandro Dalcin        for lang in LANGS:
1640ee81e68SLisandro Dalcin            pkgsrcs[lang] = []
165c729af5eSJose E. Roman        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, 'src', pkg)]):
1664bfab837SSebastian Grimberg            if SKIPDIRS.intersection(pathsplit(self.pkg_dir, root)): 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):
172021a2b48SJed Brown              with open(makefile) as mklines:
1730ee81e68SLisandro Dalcin                conditions = set(tuple(stripsplit(line)) for line in mklines if line.startswith('#requires'))
1740ee81e68SLisandro Dalcin              if not all(self.inconf(key, val) for key, val in conditions):
1750ee81e68SLisandro Dalcin                dirs[:] = []
1760ee81e68SLisandro Dalcin                continue
1770ee81e68SLisandro Dalcin            allsource = []
1780ee81e68SLisandro Dalcin            def mkrel(src):
1790ee81e68SLisandro Dalcin                return self.relpath(root, src)
18013d87ab8SBarry Smith            if files:
1815a7ab478SBarry Smith              source = self.get_sources_from_files(files)
1820ee81e68SLisandro Dalcin              for lang, s in source.items():
1832b757757SJed Brown                  pkgsrcs[lang] += [mkrel(t) for t in s]
18413d87ab8SBarry Smith              if os.path.isfile(makefile): self.gendeps.append(self.relpath(root, 'makefile'))
1850ee81e68SLisandro Dalcin        return pkgsrcs
1860ee81e68SLisandro Dalcin
187b0790570SJed Brown    def gen_gnumake(self, fd):
1880ee81e68SLisandro Dalcin        def write(stem, srcs):
1890ee81e68SLisandro Dalcin            for lang in LANGS:
19049316b25SLars Bilke                fd.write('%(stem)s.%(lang)s := %(srcs)s\n' % dict(stem=stem, lang=lang.replace('_','.'), srcs=' '.join(sorted(srcs[lang]))))
19147fd361eSStefano Zampini        for pkg in self.pkg_pkgs:
1920ee81e68SLisandro Dalcin            srcs = self.gen_pkg(pkg)
193b0790570SJed Brown            write('srcs-' + pkg, srcs)
1940ee81e68SLisandro Dalcin        return self.gendeps
1950ee81e68SLisandro Dalcin
1960ee81e68SLisandro Dalcin    def gen_ninja(self, fd):
1970ee81e68SLisandro Dalcin        libobjs = []
19847fd361eSStefano Zampini        for pkg in self.pkg_pkgs:
1990ee81e68SLisandro Dalcin            srcs = self.gen_pkg(pkg)
2000ee81e68SLisandro Dalcin            for lang in LANGS:
2010ee81e68SLisandro Dalcin                for src in srcs[lang]:
2020ee81e68SLisandro Dalcin                    obj = '$objdir/%s.o' % src
203add6df95SStefano 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)))
2040ee81e68SLisandro Dalcin                    libobjs.append(obj)
2050ee81e68SLisandro Dalcin        fd.write('\n')
2060ee81e68SLisandro Dalcin        fd.write('build $libdir/libpetsc.so : %s_LINK_SHARED %s\n\n' % ('CF'[self.have_fortran], ' '.join(libobjs)))
2070ee81e68SLisandro Dalcin        fd.write('build petsc : phony || $libdir/libpetsc.so\n\n')
2080ee81e68SLisandro Dalcin
2090ee81e68SLisandro Dalcindef WriteGnuMake(petsc):
210add6df95SStefano Zampini    arch_files = petsc.pkg_arch_path('lib',petsc.pkg_name,'conf', 'files')
211021a2b48SJed Brown    with open(arch_files, 'w') as fd:
2120ee81e68SLisandro Dalcin        gendeps = petsc.gen_gnumake(fd)
2130ee81e68SLisandro Dalcin        fd.write('\n')
2140ee81e68SLisandro Dalcin        fd.write('# Dependency to regenerate this file\n')
215add6df95SStefano Zampini        fd.write('%s : %s %s\n' % (os.path.relpath(arch_files, petsc.pkg_dir),
216add6df95SStefano Zampini                                   os.path.relpath(__file__, os.path.realpath(petsc.pkg_dir)),
2170ee81e68SLisandro Dalcin                                   ' '.join(gendeps)))
2180ee81e68SLisandro Dalcin        fd.write('\n')
2190ee81e68SLisandro Dalcin        fd.write('# Dummy dependencies in case makefiles are removed\n')
2200ee81e68SLisandro Dalcin        fd.write(''.join([dep + ':\n' for dep in gendeps]))
2210ee81e68SLisandro Dalcin
2220ee81e68SLisandro Dalcindef WriteNinja(petsc):
2230ee81e68SLisandro Dalcin    conf = dict()
224af0996ceSBarry Smith    parse_makefile(os.path.join(petsc.petsc_dir, 'lib', 'petsc','conf', 'variables'), conf)
225af0996ceSBarry Smith    parse_makefile(petsc.arch_path('lib','petsc','conf', 'petscvariables'), conf)
2260ee81e68SLisandro Dalcin    build_ninja = petsc.arch_path('build.ninja')
227021a2b48SJed Brown    with open(build_ninja, 'w') as fd:
2280ee81e68SLisandro Dalcin        fd.write('objdir = obj-ninja\n')
2290ee81e68SLisandro Dalcin        fd.write('libdir = lib\n')
2300ee81e68SLisandro Dalcin        fd.write('c_compile = %(PCC)s\n' % conf)
2310ee81e68SLisandro Dalcin        fd.write('c_flags = %(PETSC_CC_INCLUDES)s %(PCC_FLAGS)s %(CCPPFLAGS)s\n' % conf)
2320ee81e68SLisandro Dalcin        fd.write('c_link = %(PCC_LINKER)s\n' % conf)
2330ee81e68SLisandro Dalcin        fd.write('c_link_flags = %(PCC_LINKER_FLAGS)s\n' % conf)
2340ee81e68SLisandro Dalcin        if petsc.have_fortran:
2350ee81e68SLisandro Dalcin            fd.write('f_compile = %(FC)s\n' % conf)
2360ee81e68SLisandro Dalcin            fd.write('f_flags = %(PETSC_FC_INCLUDES)s %(FC_FLAGS)s %(FCPPFLAGS)s\n' % conf)
2370ee81e68SLisandro Dalcin            fd.write('f_link = %(FC_LINKER)s\n' % conf)
2380ee81e68SLisandro Dalcin            fd.write('f_link_flags = %(FC_LINKER_FLAGS)s\n' % conf)
2390ee81e68SLisandro Dalcin        fd.write('petsc_external_lib = %(PETSC_EXTERNAL_LIB_BASIC)s\n' % conf)
2400ee81e68SLisandro Dalcin        fd.write('python = %(PYTHON)s\n' % conf)
2410ee81e68SLisandro Dalcin        fd.write('\n')
2420ee81e68SLisandro Dalcin        fd.write('rule C_COMPILE\n'
2430ee81e68SLisandro Dalcin                 '  command = $c_compile -MMD -MF $out.d $c_flags -c $in -o $out\n'
2440ee81e68SLisandro Dalcin                 '  description = CC $out\n'
2450ee81e68SLisandro Dalcin                 '  depfile = $out.d\n'
2460ee81e68SLisandro Dalcin                 # '  deps = gcc\n') # 'gcc' is default, 'msvc' only recognized by newer versions of ninja
2470ee81e68SLisandro Dalcin                 '\n')
2480ee81e68SLisandro Dalcin        fd.write('rule C_LINK_SHARED\n'
2490ee81e68SLisandro Dalcin                 '  command = $c_link $c_link_flags -shared -o $out $in $petsc_external_lib\n'
2500ee81e68SLisandro Dalcin                 '  description = CLINK_SHARED $out\n'
2510ee81e68SLisandro Dalcin                 '\n')
2520ee81e68SLisandro Dalcin        if petsc.have_fortran:
2530ee81e68SLisandro Dalcin            fd.write('rule F_COMPILE\n'
2540ee81e68SLisandro Dalcin                     '  command = $f_compile -MMD -MF $out.d $f_flags -c $in -o $out\n'
2550ee81e68SLisandro Dalcin                     '  description = FC $out\n'
2560ee81e68SLisandro Dalcin                     '  depfile = $out.d\n'
2570ee81e68SLisandro Dalcin                     '\n')
2580ee81e68SLisandro Dalcin            fd.write('rule F_LINK_SHARED\n'
2590ee81e68SLisandro Dalcin                     '  command = $f_link $f_link_flags -shared -o $out $in $petsc_external_lib\n'
2600ee81e68SLisandro Dalcin                     '  description = FLINK_SHARED $out\n'
2610ee81e68SLisandro Dalcin                     '\n')
2620ee81e68SLisandro Dalcin        fd.write('rule GEN_NINJA\n'
2630ee81e68SLisandro Dalcin                 '  command = $python $in --output=ninja\n'
2640ee81e68SLisandro Dalcin                 '  generator = 1\n'
2650ee81e68SLisandro Dalcin                 '\n')
2660ee81e68SLisandro Dalcin        petsc.gen_ninja(fd)
2670ee81e68SLisandro Dalcin        fd.write('\n')
2680ee81e68SLisandro Dalcin        fd.write('build %s : GEN_NINJA | %s %s %s %s\n' % (build_ninja,
2690ee81e68SLisandro Dalcin                                                           os.path.abspath(__file__),
270af0996ceSBarry Smith                                                           os.path.join(petsc.petsc_dir, 'lib','petsc','conf', 'variables'),
271af0996ceSBarry Smith                                                           petsc.arch_path('lib','petsc','conf', 'petscvariables'),
272add6df95SStefano Zampini                                                       ' '.join(os.path.join(petsc.pkg_dir, dep) for dep in petsc.gendeps)))
2730ee81e68SLisandro Dalcin
274a95bf149SBarry Smithdef main(petsc_dir=None, petsc_arch=None, pkg_dir=None, pkg_name=None, pkg_arch=None, pkg_pkgs=None, output=None):
2750ee81e68SLisandro Dalcin    if output is None:
2760ee81e68SLisandro Dalcin        output = 'gnumake'
2770ee81e68SLisandro Dalcin    writer = dict(gnumake=WriteGnuMake, ninja=WriteNinja)
278a95bf149SBarry 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)
2790ee81e68SLisandro Dalcin    writer[output](petsc)
2800ee81e68SLisandro Dalcin
2810ee81e68SLisandro Dalcinif __name__ == '__main__':
2820ee81e68SLisandro Dalcin    import optparse
2830ee81e68SLisandro Dalcin    parser = optparse.OptionParser()
2840ee81e68SLisandro Dalcin    parser.add_option('--petsc-arch', help='Set PETSC_ARCH different from environment', default=os.environ.get('PETSC_ARCH'))
285add6df95SStefano 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)
286add6df95SStefano Zampini    parser.add_option('--pkg-name', help='Set the name of the package you want to generate the makefile rules for', default=None)
287c4bccdb5SStefano Zampini    parser.add_option('--pkg-arch', help='Set the package arch name you want to generate the makefile rules for', default=None)
28847fd361eSStefano 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)
2890ee81e68SLisandro Dalcin    parser.add_option('--output', help='Location to write output file', default=None)
2900ee81e68SLisandro Dalcin    opts, extra_args = parser.parse_args()
2910ee81e68SLisandro Dalcin    if extra_args:
2920ee81e68SLisandro Dalcin        import sys
2930ee81e68SLisandro Dalcin        sys.stderr.write('Unknown arguments: %s\n' % ' '.join(extra_args))
2940ee81e68SLisandro Dalcin        exit(1)
295a95bf149SBarry 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)
296