This makes waf compatible with Python 3.12 again. Also, apply modifications needed for MacOS and add as a patch file (see commitspull/953/head0f2e3b2anddc6c995). Signed-off-by: Nils Philippsen <nils@tiptoe.de>
| @@ -1,4 +1,4 @@ | |||||
| #!/usr/bin/python3 | |||||
| #!/usr/bin/env python | |||||
| # encoding: latin-1 | # encoding: latin-1 | ||||
| # Thomas Nagy, 2005-2018 | # Thomas Nagy, 2005-2018 | ||||
| # | # | ||||
| @@ -32,7 +32,7 @@ POSSIBILITY OF SUCH DAMAGE. | |||||
| import os, sys, inspect | import os, sys, inspect | ||||
| VERSION="2.0.12" | |||||
| VERSION="2.0.26" | |||||
| REVISION="x" | REVISION="x" | ||||
| GIT="x" | GIT="x" | ||||
| INSTALL="x" | INSTALL="x" | ||||
| @@ -142,6 +142,9 @@ def find_lib(): | |||||
| if name.endswith('waf-light'): | if name.endswith('waf-light'): | ||||
| w = test(base) | w = test(base) | ||||
| if w: return w | if w: return w | ||||
| for dir in sys.path: | |||||
| if test(dir): | |||||
| return dir | |||||
| err('waf-light requires waflib -> export WAFDIR=/folder') | err('waf-light requires waflib -> export WAFDIR=/folder') | ||||
| dirname = '%s-%s-%s' % (WAF, VERSION, REVISION) | dirname = '%s-%s-%s' % (WAF, VERSION, REVISION) | ||||
| @@ -0,0 +1,18 @@ | |||||
| diff --git a/waflib/Tools/ccroot.py b/waflib/Tools/ccroot.py | |||||
| index cfef8bf5..484846f5 100644 | |||||
| --- a/waflib/Tools/ccroot.py | |||||
| +++ b/waflib/Tools/ccroot.py | |||||
| @@ -575,12 +575,10 @@ def apply_vnum(self): | |||||
| cnum = getattr(self, 'cnum', str(nums[0])) | |||||
| cnums = cnum.split('.') | |||||
| - if len(cnums)>len(nums) or nums[0:len(cnums)] != cnums: | |||||
| - raise Errors.WafError('invalid compatibility version %s' % cnum) | |||||
| libname = node.name | |||||
| if libname.endswith('.dylib'): | |||||
| - name3 = libname.replace('.dylib', '.%s.dylib' % self.vnum) | |||||
| + name3 = libname.replace('.dylib', '.%s.dylib' % cnums[0]) | |||||
| name2 = libname.replace('.dylib', '.%s.dylib' % cnum) | |||||
| else: | |||||
| name3 = libname + '.' + self.vnum | |||||
| @@ -104,7 +104,7 @@ class BuildContext(Context.Context): | |||||
| """Amount of jobs to run in parallel""" | """Amount of jobs to run in parallel""" | ||||
| self.targets = Options.options.targets | self.targets = Options.options.targets | ||||
| """List of targets to build (default: \*)""" | |||||
| """List of targets to build (default: \\*)""" | |||||
| self.keep = Options.options.keep | self.keep = Options.options.keep | ||||
| """Whether the build should continue past errors""" | """Whether the build should continue past errors""" | ||||
| @@ -753,10 +753,12 @@ class BuildContext(Context.Context): | |||||
| else: | else: | ||||
| ln = self.launch_node() | ln = self.launch_node() | ||||
| if ln.is_child_of(self.bldnode): | if ln.is_child_of(self.bldnode): | ||||
| Logs.warn('Building from the build directory, forcing --targets=*') | |||||
| if Logs.verbose > 1: | |||||
| Logs.warn('Building from the build directory, forcing --targets=*') | |||||
| ln = self.srcnode | ln = self.srcnode | ||||
| elif not ln.is_child_of(self.srcnode): | elif not ln.is_child_of(self.srcnode): | ||||
| Logs.warn('CWD %s is not under %s, forcing --targets=* (run distclean?)', ln.abspath(), self.srcnode.abspath()) | |||||
| if Logs.verbose > 1: | |||||
| Logs.warn('CWD %s is not under %s, forcing --targets=* (run distclean?)', ln.abspath(), self.srcnode.abspath()) | |||||
| ln = self.srcnode | ln = self.srcnode | ||||
| def is_post(tg, ln): | def is_post(tg, ln): | ||||
| @@ -1054,7 +1056,7 @@ class inst(Task.Task): | |||||
| def get_install_path(self, destdir=True): | def get_install_path(self, destdir=True): | ||||
| """ | """ | ||||
| Returns the destination path where files will be installed, pre-pending `destdir`. | Returns the destination path where files will be installed, pre-pending `destdir`. | ||||
| Relative paths will be interpreted relative to `PREFIX` if no `destdir` is given. | Relative paths will be interpreted relative to `PREFIX` if no `destdir` is given. | ||||
| :rtype: string | :rtype: string | ||||
| @@ -1062,11 +1064,11 @@ class inst(Task.Task): | |||||
| if isinstance(self.install_to, Node.Node): | if isinstance(self.install_to, Node.Node): | ||||
| dest = self.install_to.abspath() | dest = self.install_to.abspath() | ||||
| else: | else: | ||||
| dest = Utils.subst_vars(self.install_to, self.env) | |||||
| dest = os.path.normpath(Utils.subst_vars(self.install_to, self.env)) | |||||
| if not os.path.isabs(dest): | if not os.path.isabs(dest): | ||||
| dest = os.path.join(self.env.PREFIX, dest) | |||||
| dest = os.path.join(self.env.PREFIX, dest) | |||||
| if destdir and Options.options.destdir: | if destdir and Options.options.destdir: | ||||
| dest = os.path.join(Options.options.destdir, os.path.splitdrive(dest)[1].lstrip(os.sep)) | |||||
| dest = Options.options.destdir.rstrip(os.sep) + os.sep + os.path.splitdrive(dest)[1].lstrip(os.sep) | |||||
| return dest | return dest | ||||
| def copy_fun(self, src, tgt): | def copy_fun(self, src, tgt): | ||||
| @@ -1160,11 +1162,19 @@ class inst(Task.Task): | |||||
| # same size and identical timestamps -> make no copy | # same size and identical timestamps -> make no copy | ||||
| if st1.st_mtime + 2 >= st2.st_mtime and st1.st_size == st2.st_size: | if st1.st_mtime + 2 >= st2.st_mtime and st1.st_size == st2.st_size: | ||||
| if not self.generator.bld.progress_bar: | if not self.generator.bld.progress_bar: | ||||
| Logs.info('- install %s (from %s)', tgt, lbl) | |||||
| c1 = Logs.colors.NORMAL | |||||
| c2 = Logs.colors.BLUE | |||||
| Logs.info('%s- install %s%s%s (from %s)', c1, c2, tgt, c1, lbl) | |||||
| return False | return False | ||||
| if not self.generator.bld.progress_bar: | if not self.generator.bld.progress_bar: | ||||
| Logs.info('+ install %s (from %s)', tgt, lbl) | |||||
| c1 = Logs.colors.NORMAL | |||||
| c2 = Logs.colors.BLUE | |||||
| Logs.info('%s+ install %s%s%s (from %s)', c1, c2, tgt, c1, lbl) | |||||
| # Give best attempt at making destination overwritable, | # Give best attempt at making destination overwritable, | ||||
| # like the 'install' utility used by 'make install' does. | # like the 'install' utility used by 'make install' does. | ||||
| @@ -1221,14 +1231,18 @@ class inst(Task.Task): | |||||
| """ | """ | ||||
| if os.path.islink(tgt) and os.readlink(tgt) == src: | if os.path.islink(tgt) and os.readlink(tgt) == src: | ||||
| if not self.generator.bld.progress_bar: | if not self.generator.bld.progress_bar: | ||||
| Logs.info('- symlink %s (to %s)', tgt, src) | |||||
| c1 = Logs.colors.NORMAL | |||||
| c2 = Logs.colors.BLUE | |||||
| Logs.info('%s- symlink %s%s%s (to %s)', c1, c2, tgt, c1, src) | |||||
| else: | else: | ||||
| try: | try: | ||||
| os.remove(tgt) | os.remove(tgt) | ||||
| except OSError: | except OSError: | ||||
| pass | pass | ||||
| if not self.generator.bld.progress_bar: | if not self.generator.bld.progress_bar: | ||||
| Logs.info('+ symlink %s (to %s)', tgt, src) | |||||
| c1 = Logs.colors.NORMAL | |||||
| c2 = Logs.colors.BLUE | |||||
| Logs.info('%s+ symlink %s%s%s (to %s)', c1, c2, tgt, c1, src) | |||||
| os.symlink(src, tgt) | os.symlink(src, tgt) | ||||
| self.fix_perms(tgt) | self.fix_perms(tgt) | ||||
| @@ -1237,7 +1251,9 @@ class inst(Task.Task): | |||||
| See :py:meth:`waflib.Build.inst.do_install` | See :py:meth:`waflib.Build.inst.do_install` | ||||
| """ | """ | ||||
| if not self.generator.bld.progress_bar: | if not self.generator.bld.progress_bar: | ||||
| Logs.info('- remove %s', tgt) | |||||
| c1 = Logs.colors.NORMAL | |||||
| c2 = Logs.colors.BLUE | |||||
| Logs.info('%s- remove %s%s%s', c1, c2, tgt, c1) | |||||
| #self.uninstall.append(tgt) | #self.uninstall.append(tgt) | ||||
| try: | try: | ||||
| @@ -1257,7 +1273,9 @@ class inst(Task.Task): | |||||
| """ | """ | ||||
| try: | try: | ||||
| if not self.generator.bld.progress_bar: | if not self.generator.bld.progress_bar: | ||||
| Logs.info('- remove %s', tgt) | |||||
| c1 = Logs.colors.NORMAL | |||||
| c2 = Logs.colors.BLUE | |||||
| Logs.info('%s- remove %s%s%s', c1, c2, tgt, c1) | |||||
| os.remove(tgt) | os.remove(tgt) | ||||
| except OSError: | except OSError: | ||||
| pass | pass | ||||
| @@ -1318,7 +1336,8 @@ class CleanContext(BuildContext): | |||||
| lst = [] | lst = [] | ||||
| for env in self.all_envs.values(): | for env in self.all_envs.values(): | ||||
| lst.extend(self.root.find_or_declare(f) for f in env[CFG_FILES]) | lst.extend(self.root.find_or_declare(f) for f in env[CFG_FILES]) | ||||
| for n in self.bldnode.ant_glob('**/*', excl='.lock* *conf_check_*/** config.log c4che/*', quiet=True): | |||||
| excluded_dirs = '.lock* *conf_check_*/** config.log %s/*' % CACHE_DIR | |||||
| for n in self.bldnode.ant_glob('**/*', excl=excluded_dirs, quiet=True): | |||||
| if n in lst: | if n in lst: | ||||
| continue | continue | ||||
| n.delete() | n.delete() | ||||
| @@ -11,7 +11,7 @@ The values put in :py:class:`ConfigSet` must be serializable (dicts, lists, stri | |||||
| import copy, re, os | import copy, re, os | ||||
| from waflib import Logs, Utils | from waflib import Logs, Utils | ||||
| re_imp = re.compile('^(#)*?([^#=]*?)\ =\ (.*?)$', re.M) | |||||
| re_imp = re.compile(r'^(#)*?([^#=]*?)\ =\ (.*?)$', re.M) | |||||
| class ConfigSet(object): | class ConfigSet(object): | ||||
| """ | """ | ||||
| @@ -125,7 +125,7 @@ class ConfigurationContext(Context.Context): | |||||
| self.bldnode.mkdir() | self.bldnode.mkdir() | ||||
| if not os.path.isdir(self.bldnode.abspath()): | if not os.path.isdir(self.bldnode.abspath()): | ||||
| conf.fatal('Could not create the build directory %s' % self.bldnode.abspath()) | |||||
| self.fatal('Could not create the build directory %s' % self.bldnode.abspath()) | |||||
| def execute(self): | def execute(self): | ||||
| """ | """ | ||||
| @@ -180,6 +180,7 @@ class ConfigurationContext(Context.Context): | |||||
| env.hash = self.hash | env.hash = self.hash | ||||
| env.files = self.files | env.files = self.files | ||||
| env.environ = dict(self.environ) | env.environ = dict(self.environ) | ||||
| env.launch_dir = Context.launch_dir | |||||
| if not (self.env.NO_LOCK_IN_RUN or env.environ.get('NO_LOCK_IN_RUN') or getattr(Options.options, 'no_lock_in_run')): | if not (self.env.NO_LOCK_IN_RUN or env.environ.get('NO_LOCK_IN_RUN') or getattr(Options.options, 'no_lock_in_run')): | ||||
| env.store(os.path.join(Context.run_dir, Options.lockfile)) | env.store(os.path.join(Context.run_dir, Options.lockfile)) | ||||
| @@ -438,7 +439,7 @@ def find_program(self, filename, **kw): | |||||
| var = kw.get('var', '') | var = kw.get('var', '') | ||||
| if not var: | if not var: | ||||
| var = re.sub(r'[-.]', '_', filename[0].upper()) | |||||
| var = re.sub(r'\W', '_', filename[0].upper()) | |||||
| path_list = kw.get('path_list', '') | path_list = kw.get('path_list', '') | ||||
| if path_list: | if path_list: | ||||
| @@ -507,23 +508,27 @@ def find_binary(self, filenames, exts, paths): | |||||
| @conf | @conf | ||||
| def run_build(self, *k, **kw): | def run_build(self, *k, **kw): | ||||
| """ | """ | ||||
| Create a temporary build context to execute a build. A reference to that build | |||||
| context is kept on self.test_bld for debugging purposes, and you should not rely | |||||
| on it too much (read the note on the cache below). | |||||
| The parameters given in the arguments to this function are passed as arguments for | |||||
| a single task generator created in the build. Only three parameters are obligatory: | |||||
| Create a temporary build context to execute a build. A temporary reference to that build | |||||
| context is kept on self.test_bld for debugging purposes. | |||||
| The arguments to this function are passed to a single task generator for that build. | |||||
| Only three parameters are mandatory: | |||||
| :param features: features to pass to a task generator created in the build | :param features: features to pass to a task generator created in the build | ||||
| :type features: list of string | :type features: list of string | ||||
| :param compile_filename: file to create for the compilation (default: *test.c*) | :param compile_filename: file to create for the compilation (default: *test.c*) | ||||
| :type compile_filename: string | :type compile_filename: string | ||||
| :param code: code to write in the filename to compile | |||||
| :param code: input file contents | |||||
| :type code: string | :type code: string | ||||
| Though this function returns *0* by default, the build may set an attribute named *retval* on the | |||||
| Though this function returns *0* by default, the build may bind attribute named *retval* on the | |||||
| build context object to return a particular value. See :py:func:`waflib.Tools.c_config.test_exec_fun` for example. | build context object to return a particular value. See :py:func:`waflib.Tools.c_config.test_exec_fun` for example. | ||||
| This function also provides a limited cache. To use it, provide the following option:: | |||||
| The temporary builds creates a temporary folder; the name of that folder is calculated | |||||
| by hashing input arguments to this function, with the exception of :py:class:`waflib.ConfigSet.ConfigSet` | |||||
| objects which are used for both reading and writing values. | |||||
| This function also features a cache which is disabled by default; that cache relies | |||||
| on the hash value calculated as indicated above:: | |||||
| def options(opt): | def options(opt): | ||||
| opt.add_option('--confcache', dest='confcache', default=0, | opt.add_option('--confcache', dest='confcache', default=0, | ||||
| @@ -534,10 +539,24 @@ def run_build(self, *k, **kw): | |||||
| $ waf configure --confcache | $ waf configure --confcache | ||||
| """ | """ | ||||
| lst = [str(v) for (p, v) in kw.items() if p != 'env'] | |||||
| h = Utils.h_list(lst) | |||||
| buf = [] | |||||
| for key in sorted(kw.keys()): | |||||
| v = kw[key] | |||||
| if isinstance(v, ConfigSet.ConfigSet): | |||||
| # values are being written to, so they are excluded from contributing to the hash | |||||
| continue | |||||
| elif hasattr(v, '__call__'): | |||||
| buf.append(Utils.h_fun(v)) | |||||
| else: | |||||
| buf.append(str(v)) | |||||
| h = Utils.h_list(buf) | |||||
| dir = self.bldnode.abspath() + os.sep + (not Utils.is_win32 and '.' or '') + 'conf_check_' + Utils.to_hex(h) | dir = self.bldnode.abspath() + os.sep + (not Utils.is_win32 and '.' or '') + 'conf_check_' + Utils.to_hex(h) | ||||
| cachemode = kw.get('confcache', getattr(Options.options, 'confcache', None)) | |||||
| if not cachemode and os.path.exists(dir): | |||||
| shutil.rmtree(dir) | |||||
| try: | try: | ||||
| os.makedirs(dir) | os.makedirs(dir) | ||||
| except OSError: | except OSError: | ||||
| @@ -548,7 +567,6 @@ def run_build(self, *k, **kw): | |||||
| except OSError: | except OSError: | ||||
| self.fatal('cannot use the configuration test folder %r' % dir) | self.fatal('cannot use the configuration test folder %r' % dir) | ||||
| cachemode = getattr(Options.options, 'confcache', None) | |||||
| if cachemode == 1: | if cachemode == 1: | ||||
| try: | try: | ||||
| proj = ConfigSet.ConfigSet(os.path.join(dir, 'cache_run_build')) | proj = ConfigSet.ConfigSet(os.path.join(dir, 'cache_run_build')) | ||||
| @@ -588,7 +606,7 @@ def run_build(self, *k, **kw): | |||||
| else: | else: | ||||
| ret = getattr(bld, 'retval', 0) | ret = getattr(bld, 'retval', 0) | ||||
| finally: | finally: | ||||
| if cachemode == 1: | |||||
| if cachemode: | |||||
| # cache the results each time | # cache the results each time | ||||
| proj = ConfigSet.ConfigSet() | proj = ConfigSet.ConfigSet() | ||||
| proj['cache_run_build'] = ret | proj['cache_run_build'] = ret | ||||
| @@ -6,20 +6,30 @@ | |||||
| Classes and functions enabling the command system | Classes and functions enabling the command system | ||||
| """ | """ | ||||
| import os, re, imp, sys | |||||
| import os, re, sys | |||||
| from waflib import Utils, Errors, Logs | from waflib import Utils, Errors, Logs | ||||
| import waflib.Node | import waflib.Node | ||||
| if sys.hexversion > 0x3040000: | |||||
| import types | |||||
| class imp(object): | |||||
| new_module = lambda x: types.ModuleType(x) | |||||
| else: | |||||
| import imp | |||||
| # the following 3 constants are updated on each new release (do not touch) | # the following 3 constants are updated on each new release (do not touch) | ||||
| HEXVERSION=0x2000c00 | |||||
| HEXVERSION=0x2001a00 | |||||
| """Constant updated on new releases""" | """Constant updated on new releases""" | ||||
| WAFVERSION="2.0.12" | |||||
| WAFVERSION="2.0.26" | |||||
| """Constant updated on new releases""" | """Constant updated on new releases""" | ||||
| WAFREVISION="54841218840ffa34fddf834680a5a17db69caa12" | |||||
| WAFREVISION="0fb985ce1932c6f3e7533f435e4ee209d673776e" | |||||
| """Git revision when the waf version is updated""" | """Git revision when the waf version is updated""" | ||||
| WAFNAME="waf" | |||||
| """Application name displayed on --help""" | |||||
| ABI = 20 | ABI = 20 | ||||
| """Version of the build data cache file format (used in :py:const:`waflib.Context.DBFILE`)""" | """Version of the build data cache file format (used in :py:const:`waflib.Context.DBFILE`)""" | ||||
| @@ -134,7 +144,7 @@ class Context(ctx): | |||||
| :type fun: string | :type fun: string | ||||
| .. inheritance-diagram:: waflib.Context.Context waflib.Build.BuildContext waflib.Build.InstallContext waflib.Build.UninstallContext waflib.Build.StepContext waflib.Build.ListContext waflib.Configure.ConfigurationContext waflib.Scripting.Dist waflib.Scripting.DistCheck waflib.Build.CleanContext | .. inheritance-diagram:: waflib.Context.Context waflib.Build.BuildContext waflib.Build.InstallContext waflib.Build.UninstallContext waflib.Build.StepContext waflib.Build.ListContext waflib.Configure.ConfigurationContext waflib.Scripting.Dist waflib.Scripting.DistCheck waflib.Build.CleanContext | ||||
| :top-classes: waflib.Context.Context | |||||
| """ | """ | ||||
| errors = Errors | errors = Errors | ||||
| @@ -613,7 +623,7 @@ class Context(ctx): | |||||
| is typically called once for a programming language group, see for | is typically called once for a programming language group, see for | ||||
| example :py:mod:`waflib.Tools.compiler_c` | example :py:mod:`waflib.Tools.compiler_c` | ||||
| :param var: glob expression, for example 'cxx\_\*.py' | |||||
| :param var: glob expression, for example 'cxx\\_\\*.py' | |||||
| :type var: string | :type var: string | ||||
| :param ban: list of exact file names to exclude | :param ban: list of exact file names to exclude | ||||
| :type ban: list of string | :type ban: list of string | ||||
| @@ -678,7 +688,7 @@ def load_module(path, encoding=None): | |||||
| def load_tool(tool, tooldir=None, ctx=None, with_sys_path=True): | def load_tool(tool, tooldir=None, ctx=None, with_sys_path=True): | ||||
| """ | """ | ||||
| Importx a Waf tool as a python module, and stores it in the dict :py:const:`waflib.Context.Context.tools` | |||||
| Imports a Waf tool as a python module, and stores it in the dict :py:const:`waflib.Context.Context.tools` | |||||
| :type tool: string | :type tool: string | ||||
| :param tool: Name of the tool | :param tool: Name of the tool | ||||
| @@ -237,7 +237,10 @@ class formatter(logging.Formatter): | |||||
| if rec.levelno >= logging.INFO: | if rec.levelno >= logging.INFO: | ||||
| # the goal of this is to format without the leading "Logs, hour" prefix | # the goal of this is to format without the leading "Logs, hour" prefix | ||||
| if rec.args: | if rec.args: | ||||
| return msg % rec.args | |||||
| try: | |||||
| return msg % rec.args | |||||
| except UnicodeDecodeError: | |||||
| return msg.encode('utf-8') % rec.args | |||||
| return msg | return msg | ||||
| rec.msg = msg | rec.msg = msg | ||||
| @@ -276,9 +279,9 @@ def error(*k, **kw): | |||||
| def warn(*k, **kw): | def warn(*k, **kw): | ||||
| """ | """ | ||||
| Wraps logging.warn | |||||
| Wraps logging.warning | |||||
| """ | """ | ||||
| log.warn(*k, **kw) | |||||
| log.warning(*k, **kw) | |||||
| def info(*k, **kw): | def info(*k, **kw): | ||||
| """ | """ | ||||
| @@ -73,7 +73,7 @@ def ant_matcher(s, ignorecase): | |||||
| if k == '**': | if k == '**': | ||||
| accu.append(k) | accu.append(k) | ||||
| else: | else: | ||||
| k = k.replace('.', '[.]').replace('*','.*').replace('?', '.').replace('+', '\\+') | |||||
| k = k.replace('.', '[.]').replace('*', '.*').replace('?', '.').replace('+', '\\+') | |||||
| k = '^%s$' % k | k = '^%s$' % k | ||||
| try: | try: | ||||
| exp = re.compile(k, flags=reflags) | exp = re.compile(k, flags=reflags) | ||||
| @@ -595,7 +595,6 @@ class Node(object): | |||||
| :rtype: iterator | :rtype: iterator | ||||
| """ | """ | ||||
| dircont = self.listdir() | dircont = self.listdir() | ||||
| dircont.sort() | |||||
| try: | try: | ||||
| lst = set(self.children.keys()) | lst = set(self.children.keys()) | ||||
| @@ -44,7 +44,7 @@ class opt_parser(optparse.OptionParser): | |||||
| """ | """ | ||||
| def __init__(self, ctx, allow_unknown=False): | def __init__(self, ctx, allow_unknown=False): | ||||
| optparse.OptionParser.__init__(self, conflict_handler='resolve', add_help_option=False, | optparse.OptionParser.__init__(self, conflict_handler='resolve', add_help_option=False, | ||||
| version='waf %s (%s)' % (Context.WAFVERSION, Context.WAFREVISION)) | |||||
| version='%s %s (%s)' % (Context.WAFNAME, Context.WAFVERSION, Context.WAFREVISION)) | |||||
| self.formatter.width = Logs.get_term_cols() | self.formatter.width = Logs.get_term_cols() | ||||
| self.ctx = ctx | self.ctx = ctx | ||||
| self.allow_unknown = allow_unknown | self.allow_unknown = allow_unknown | ||||
| @@ -62,6 +62,21 @@ class opt_parser(optparse.OptionParser): | |||||
| else: | else: | ||||
| self.error(str(e)) | self.error(str(e)) | ||||
| def _process_long_opt(self, rargs, values): | |||||
| # --custom-option=-ftxyz is interpreted as -f -t... see #2280 | |||||
| if self.allow_unknown: | |||||
| back = [] + rargs | |||||
| try: | |||||
| optparse.OptionParser._process_long_opt(self, rargs, values) | |||||
| except optparse.BadOptionError: | |||||
| while rargs: | |||||
| rargs.pop() | |||||
| rargs.extend(back) | |||||
| rargs.pop(0) | |||||
| raise | |||||
| else: | |||||
| optparse.OptionParser._process_long_opt(self, rargs, values) | |||||
| def print_usage(self, file=None): | def print_usage(self, file=None): | ||||
| return self.print_help(file) | return self.print_help(file) | ||||
| @@ -96,11 +111,11 @@ class opt_parser(optparse.OptionParser): | |||||
| lst.sort() | lst.sort() | ||||
| ret = '\n'.join(lst) | ret = '\n'.join(lst) | ||||
| return '''waf [commands] [options] | |||||
| return '''%s [commands] [options] | |||||
| Main commands (example: ./waf build -j4) | |||||
| Main commands (example: ./%s build -j4) | |||||
| %s | %s | ||||
| ''' % ret | |||||
| ''' % (Context.WAFNAME, Context.WAFNAME, ret) | |||||
| class OptionsContext(Context.Context): | class OptionsContext(Context.Context): | ||||
| @@ -141,9 +156,9 @@ class OptionsContext(Context.Context): | |||||
| gr.add_option('-o', '--out', action='store', default='', help='build dir for the project', dest='out') | gr.add_option('-o', '--out', action='store', default='', help='build dir for the project', dest='out') | ||||
| gr.add_option('-t', '--top', action='store', default='', help='src dir for the project', dest='top') | gr.add_option('-t', '--top', action='store', default='', help='src dir for the project', dest='top') | ||||
| gr.add_option('--no-lock-in-run', action='store_true', default='', help=optparse.SUPPRESS_HELP, dest='no_lock_in_run') | |||||
| gr.add_option('--no-lock-in-out', action='store_true', default='', help=optparse.SUPPRESS_HELP, dest='no_lock_in_out') | |||||
| gr.add_option('--no-lock-in-top', action='store_true', default='', help=optparse.SUPPRESS_HELP, dest='no_lock_in_top') | |||||
| gr.add_option('--no-lock-in-run', action='store_true', default=os.environ.get('NO_LOCK_IN_RUN', ''), help=optparse.SUPPRESS_HELP, dest='no_lock_in_run') | |||||
| gr.add_option('--no-lock-in-out', action='store_true', default=os.environ.get('NO_LOCK_IN_OUT', ''), help=optparse.SUPPRESS_HELP, dest='no_lock_in_out') | |||||
| gr.add_option('--no-lock-in-top', action='store_true', default=os.environ.get('NO_LOCK_IN_TOP', ''), help=optparse.SUPPRESS_HELP, dest='no_lock_in_top') | |||||
| default_prefix = getattr(Context.g_module, 'default_prefix', os.environ.get('PREFIX')) | default_prefix = getattr(Context.g_module, 'default_prefix', os.environ.get('PREFIX')) | ||||
| if not default_prefix: | if not default_prefix: | ||||
| @@ -282,6 +297,8 @@ class OptionsContext(Context.Context): | |||||
| elif arg != 'options': | elif arg != 'options': | ||||
| commands.append(arg) | commands.append(arg) | ||||
| if options.jobs < 1: | |||||
| options.jobs = 1 | |||||
| for name in 'top out destdir prefix bindir libdir'.split(): | for name in 'top out destdir prefix bindir libdir'.split(): | ||||
| # those paths are usually expanded from Context.launch_dir | # those paths are usually expanded from Context.launch_dir | ||||
| if getattr(options, name, None): | if getattr(options, name, None): | ||||
| @@ -37,6 +37,8 @@ class PriorityTasks(object): | |||||
| return len(self.lst) | return len(self.lst) | ||||
| def __iter__(self): | def __iter__(self): | ||||
| return iter(self.lst) | return iter(self.lst) | ||||
| def __str__(self): | |||||
| return 'PriorityTasks: [%s]' % '\n '.join(str(x) for x in self.lst) | |||||
| def clear(self): | def clear(self): | ||||
| self.lst = [] | self.lst = [] | ||||
| def append(self, task): | def append(self, task): | ||||
| @@ -69,7 +71,7 @@ class Consumer(Utils.threading.Thread): | |||||
| """Task to execute""" | """Task to execute""" | ||||
| self.spawner = spawner | self.spawner = spawner | ||||
| """Coordinator object""" | """Coordinator object""" | ||||
| self.setDaemon(1) | |||||
| self.daemon = True | |||||
| self.start() | self.start() | ||||
| def run(self): | def run(self): | ||||
| """ | """ | ||||
| @@ -96,7 +98,7 @@ class Spawner(Utils.threading.Thread): | |||||
| """:py:class:`waflib.Runner.Parallel` producer instance""" | """:py:class:`waflib.Runner.Parallel` producer instance""" | ||||
| self.sem = Utils.threading.Semaphore(master.numjobs) | self.sem = Utils.threading.Semaphore(master.numjobs) | ||||
| """Bounded semaphore that prevents spawning more than *n* concurrent consumers""" | """Bounded semaphore that prevents spawning more than *n* concurrent consumers""" | ||||
| self.setDaemon(1) | |||||
| self.daemon = True | |||||
| self.start() | self.start() | ||||
| def run(self): | def run(self): | ||||
| """ | """ | ||||
| @@ -181,10 +183,12 @@ class Parallel(object): | |||||
| The reverse dependency graph of dependencies obtained from Task.run_after | The reverse dependency graph of dependencies obtained from Task.run_after | ||||
| """ | """ | ||||
| self.spawner = Spawner(self) | |||||
| self.spawner = None | |||||
| """ | """ | ||||
| Coordinating daemon thread that spawns thread consumers | Coordinating daemon thread that spawns thread consumers | ||||
| """ | """ | ||||
| if self.numjobs > 1: | |||||
| self.spawner = Spawner(self) | |||||
| def get_next_task(self): | def get_next_task(self): | ||||
| """ | """ | ||||
| @@ -254,6 +258,8 @@ class Parallel(object): | |||||
| self.outstanding.append(x) | self.outstanding.append(x) | ||||
| break | break | ||||
| else: | else: | ||||
| if self.stop or self.error: | |||||
| break | |||||
| raise Errors.WafError('Broken revdeps detected on %r' % self.incomplete) | raise Errors.WafError('Broken revdeps detected on %r' % self.incomplete) | ||||
| else: | else: | ||||
| tasks = next(self.biter) | tasks = next(self.biter) | ||||
| @@ -331,11 +337,16 @@ class Parallel(object): | |||||
| if hasattr(tsk, 'semaphore'): | if hasattr(tsk, 'semaphore'): | ||||
| sem = tsk.semaphore | sem = tsk.semaphore | ||||
| sem.release(tsk) | |||||
| while sem.waiting and not sem.is_locked(): | |||||
| # take a frozen task, make it ready to run | |||||
| x = sem.waiting.pop() | |||||
| self._add_task(x) | |||||
| try: | |||||
| sem.release(tsk) | |||||
| except KeyError: | |||||
| # TODO | |||||
| pass | |||||
| else: | |||||
| while sem.waiting and not sem.is_locked(): | |||||
| # take a frozen task, make it ready to run | |||||
| x = sem.waiting.pop() | |||||
| self._add_task(x) | |||||
| def get_out(self): | def get_out(self): | ||||
| """ | """ | ||||
| @@ -216,7 +216,10 @@ def parse_options(): | |||||
| ctx = Context.create_context('options') | ctx = Context.create_context('options') | ||||
| ctx.execute() | ctx.execute() | ||||
| if not Options.commands: | if not Options.commands: | ||||
| Options.commands.append(default_cmd) | |||||
| if isinstance(default_cmd, list): | |||||
| Options.commands.extend(default_cmd) | |||||
| else: | |||||
| Options.commands.append(default_cmd) | |||||
| if Options.options.whelp: | if Options.options.whelp: | ||||
| ctx.parser.print_help() | ctx.parser.print_help() | ||||
| sys.exit(0) | sys.exit(0) | ||||
| @@ -280,7 +283,7 @@ def distclean_dir(dirname): | |||||
| pass | pass | ||||
| try: | try: | ||||
| shutil.rmtree('c4che') | |||||
| shutil.rmtree(Build.CACHE_DIR) | |||||
| except OSError: | except OSError: | ||||
| pass | pass | ||||
| @@ -303,7 +306,7 @@ def distclean(ctx): | |||||
| # remove a build folder, if any | # remove a build folder, if any | ||||
| cur = '.' | cur = '.' | ||||
| if ctx.options.no_lock_in_top: | |||||
| if os.environ.get('NO_LOCK_IN_TOP') or ctx.options.no_lock_in_top: | |||||
| cur = ctx.options.out | cur = ctx.options.out | ||||
| try: | try: | ||||
| @@ -329,7 +332,12 @@ def distclean(ctx): | |||||
| else: | else: | ||||
| remove_and_log(env.out_dir, shutil.rmtree) | remove_and_log(env.out_dir, shutil.rmtree) | ||||
| for k in (env.out_dir, env.top_dir, env.run_dir): | |||||
| env_dirs = [env.out_dir] | |||||
| if not (os.environ.get('NO_LOCK_IN_TOP') or ctx.options.no_lock_in_top): | |||||
| env_dirs.append(env.top_dir) | |||||
| if not (os.environ.get('NO_LOCK_IN_RUN') or ctx.options.no_lock_in_run): | |||||
| env_dirs.append(env.run_dir) | |||||
| for k in env_dirs: | |||||
| p = os.path.join(k, Options.lockfile) | p = os.path.join(k, Options.lockfile) | ||||
| remove_and_log(p, os.remove) | remove_and_log(p, os.remove) | ||||
| @@ -380,7 +388,11 @@ class Dist(Context.Context): | |||||
| for x in files: | for x in files: | ||||
| archive_name = self.get_base_name() + '/' + x.path_from(self.base_path) | archive_name = self.get_base_name() + '/' + x.path_from(self.base_path) | ||||
| zip.write(x.abspath(), archive_name, zipfile.ZIP_DEFLATED) | |||||
| if os.environ.get('SOURCE_DATE_EPOCH'): | |||||
| # TODO: parse that timestamp | |||||
| zip.writestr(zipfile.ZipInfo(archive_name), x.read(), zipfile.ZIP_DEFLATED) | |||||
| else: | |||||
| zip.write(x.abspath(), archive_name, zipfile.ZIP_DEFLATED) | |||||
| zip.close() | zip.close() | ||||
| else: | else: | ||||
| self.fatal('Valid algo types are tar.bz2, tar.gz, tar.xz or zip') | self.fatal('Valid algo types are tar.bz2, tar.gz, tar.xz or zip') | ||||
| @@ -417,6 +429,8 @@ class Dist(Context.Context): | |||||
| tinfo.gid = 0 | tinfo.gid = 0 | ||||
| tinfo.uname = 'root' | tinfo.uname = 'root' | ||||
| tinfo.gname = 'root' | tinfo.gname = 'root' | ||||
| if os.environ.get('SOURCE_DATE_EPOCH'): | |||||
| tinfo.mtime = int(os.environ.get('SOURCE_DATE_EPOCH')) | |||||
| if os.path.isfile(p): | if os.path.isfile(p): | ||||
| with open(p, 'rb') as f: | with open(p, 'rb') as f: | ||||
| @@ -598,12 +612,15 @@ def autoconfigure(execute_method): | |||||
| cmd = env.config_cmd or 'configure' | cmd = env.config_cmd or 'configure' | ||||
| if Configure.autoconfig == 'clobber': | if Configure.autoconfig == 'clobber': | ||||
| tmp = Options.options.__dict__ | tmp = Options.options.__dict__ | ||||
| launch_dir_tmp = Context.launch_dir | |||||
| if env.options: | if env.options: | ||||
| Options.options.__dict__ = env.options | Options.options.__dict__ = env.options | ||||
| Context.launch_dir = env.launch_dir | |||||
| try: | try: | ||||
| run_command(cmd) | run_command(cmd) | ||||
| finally: | finally: | ||||
| Options.options.__dict__ = tmp | Options.options.__dict__ = tmp | ||||
| Context.launch_dir = launch_dir_tmp | |||||
| else: | else: | ||||
| run_command(cmd) | run_command(cmd) | ||||
| run_command(self.cmd) | run_command(self.cmd) | ||||
| @@ -163,10 +163,10 @@ class Task(evil): | |||||
| """File extensions that objects of this task class may create""" | """File extensions that objects of this task class may create""" | ||||
| before = [] | before = [] | ||||
| """List of task class names to execute before instances of this class""" | |||||
| """The instances of this class are executed before the instances of classes whose names are in this list""" | |||||
| after = [] | after = [] | ||||
| """List of task class names to execute after instances of this class""" | |||||
| """The instances of this class are executed after the instances of classes whose names are in this list""" | |||||
| hcode = Utils.SIG_NIL | hcode = Utils.SIG_NIL | ||||
| """String representing an additional hash for the class representation""" | """String representing an additional hash for the class representation""" | ||||
| @@ -306,25 +306,31 @@ class Task(evil): | |||||
| if hasattr(self, 'stderr'): | if hasattr(self, 'stderr'): | ||||
| kw['stderr'] = self.stderr | kw['stderr'] = self.stderr | ||||
| # workaround for command line length limit: | |||||
| # http://support.microsoft.com/kb/830473 | |||||
| if not isinstance(cmd, str) and (len(repr(cmd)) >= 8192 if Utils.is_win32 else len(cmd) > 200000): | |||||
| cmd, args = self.split_argfile(cmd) | |||||
| try: | |||||
| (fd, tmp) = tempfile.mkstemp() | |||||
| os.write(fd, '\r\n'.join(args).encode()) | |||||
| os.close(fd) | |||||
| if Logs.verbose: | |||||
| Logs.debug('argfile: @%r -> %r', tmp, args) | |||||
| return self.generator.bld.exec_command(cmd + ['@' + tmp], **kw) | |||||
| finally: | |||||
| if not isinstance(cmd, str): | |||||
| if Utils.is_win32: | |||||
| # win32 compares the resulting length http://support.microsoft.com/kb/830473 | |||||
| too_long = sum([len(arg) for arg in cmd]) + len(cmd) > 8192 | |||||
| else: | |||||
| # non-win32 counts the amount of arguments (200k) | |||||
| too_long = len(cmd) > 200000 | |||||
| if too_long and getattr(self, 'allow_argsfile', True): | |||||
| # Shunt arguments to a temporary file if the command is too long. | |||||
| cmd, args = self.split_argfile(cmd) | |||||
| try: | try: | ||||
| os.remove(tmp) | |||||
| except OSError: | |||||
| # anti-virus and indexers can keep files open -_- | |||||
| pass | |||||
| else: | |||||
| return self.generator.bld.exec_command(cmd, **kw) | |||||
| (fd, tmp) = tempfile.mkstemp() | |||||
| os.write(fd, '\r\n'.join(args).encode()) | |||||
| os.close(fd) | |||||
| if Logs.verbose: | |||||
| Logs.debug('argfile: @%r -> %r', tmp, args) | |||||
| return self.generator.bld.exec_command(cmd + ['@' + tmp], **kw) | |||||
| finally: | |||||
| try: | |||||
| os.remove(tmp) | |||||
| except OSError: | |||||
| # anti-virus and indexers can keep files open -_- | |||||
| pass | |||||
| return self.generator.bld.exec_command(cmd, **kw) | |||||
| def process(self): | def process(self): | ||||
| """ | """ | ||||
| @@ -1044,7 +1050,7 @@ def funex(c): | |||||
| exec(c, dc) | exec(c, dc) | ||||
| return dc['f'] | return dc['f'] | ||||
| re_cond = re.compile('(?P<var>\w+)|(?P<or>\|)|(?P<and>&)') | |||||
| re_cond = re.compile(r'(?P<var>\w+)|(?P<or>\|)|(?P<and>&)') | |||||
| re_novar = re.compile(r'^(SRC|TGT)\W+.*?$') | re_novar = re.compile(r'^(SRC|TGT)\W+.*?$') | ||||
| reg_act = re.compile(r'(?P<backslash>\\)|(?P<dollar>\$\$)|(?P<subst>\$\{(?P<var>\w+)(?P<code>.*?)\})', re.M) | reg_act = re.compile(r'(?P<backslash>\\)|(?P<dollar>\$\$)|(?P<subst>\$\{(?P<var>\w+)(?P<code>.*?)\})', re.M) | ||||
| def compile_fun_shell(line): | def compile_fun_shell(line): | ||||
| @@ -74,7 +74,7 @@ class task_gen(object): | |||||
| else: | else: | ||||
| self.bld = kw['bld'] | self.bld = kw['bld'] | ||||
| self.env = self.bld.env.derive() | self.env = self.bld.env.derive() | ||||
| self.path = self.bld.path # emulate chdir when reading scripts | |||||
| self.path = kw.get('path', self.bld.path) # by default, emulate chdir when reading scripts | |||||
| # Provide a unique index per folder | # Provide a unique index per folder | ||||
| # This is part of a measure to prevent output file name collisions | # This is part of a measure to prevent output file name collisions | ||||
| @@ -400,7 +400,7 @@ def feature(*k): | |||||
| Decorator that registers a task generator method that will be executed when the | Decorator that registers a task generator method that will be executed when the | ||||
| object attribute ``feature`` contains the corresponding key(s):: | object attribute ``feature`` contains the corresponding key(s):: | ||||
| from waflib.Task import feature | |||||
| from waflib.TaskGen import feature | |||||
| @feature('myfeature') | @feature('myfeature') | ||||
| def myfunction(self): | def myfunction(self): | ||||
| print('that is my feature!') | print('that is my feature!') | ||||
| @@ -631,12 +631,8 @@ def process_rule(self): | |||||
| cls.scan = self.scan | cls.scan = self.scan | ||||
| elif has_deps: | elif has_deps: | ||||
| def scan(self): | def scan(self): | ||||
| nodes = [] | |||||
| for x in self.generator.to_list(getattr(self.generator, 'deps', None)): | |||||
| node = self.generator.path.find_resource(x) | |||||
| if not node: | |||||
| self.generator.bld.fatal('Could not find %r (was it declared?)' % x) | |||||
| nodes.append(node) | |||||
| deps = getattr(self.generator, 'deps', None) | |||||
| nodes = self.generator.to_nodes(deps) | |||||
| return [nodes, []] | return [nodes, []] | ||||
| cls.scan = scan | cls.scan = scan | ||||
| @@ -727,7 +723,7 @@ def sequence_order(self): | |||||
| self.bld.prev = self | self.bld.prev = self | ||||
| re_m4 = re.compile('@(\w+)@', re.M) | |||||
| re_m4 = re.compile(r'@(\w+)@', re.M) | |||||
| class subst_pc(Task.Task): | class subst_pc(Task.Task): | ||||
| """ | """ | ||||
| @@ -905,7 +901,7 @@ def process_subst(self): | |||||
| # paranoid safety measure for the general case foo.in->foo.h with ambiguous dependencies | # paranoid safety measure for the general case foo.in->foo.h with ambiguous dependencies | ||||
| for xt in HEADER_EXTS: | for xt in HEADER_EXTS: | ||||
| if b.name.endswith(xt): | if b.name.endswith(xt): | ||||
| tsk.ext_in = tsk.ext_in + ['.h'] | |||||
| tsk.ext_out = tsk.ext_out + ['.h'] | |||||
| break | break | ||||
| inst_to = getattr(self, 'install_path', None) | inst_to = getattr(self, 'install_path', None) | ||||
| @@ -38,7 +38,7 @@ def sniff_features(**kw): | |||||
| :return: the list of features for a task generator processing the source files | :return: the list of features for a task generator processing the source files | ||||
| :rtype: list of string | :rtype: list of string | ||||
| """ | """ | ||||
| exts = get_extensions(kw['source']) | |||||
| exts = get_extensions(kw.get('source', [])) | |||||
| typ = kw['typ'] | typ = kw['typ'] | ||||
| feats = [] | feats = [] | ||||
| @@ -47,10 +47,12 @@ def sniff_features(**kw): | |||||
| if x in exts: | if x in exts: | ||||
| feats.append('cxx') | feats.append('cxx') | ||||
| break | break | ||||
| if 'c' in exts or 'vala' in exts or 'gs' in exts: | if 'c' in exts or 'vala' in exts or 'gs' in exts: | ||||
| feats.append('c') | feats.append('c') | ||||
| if 's' in exts or 'S' in exts: | |||||
| feats.append('asm') | |||||
| for x in 'f f90 F F90 for FOR'.split(): | for x in 'f f90 F F90 for FOR'.split(): | ||||
| if x in exts: | if x in exts: | ||||
| feats.append('fc') | feats.append('fc') | ||||
| @@ -66,11 +68,11 @@ def sniff_features(**kw): | |||||
| if typ in ('program', 'shlib', 'stlib'): | if typ in ('program', 'shlib', 'stlib'): | ||||
| will_link = False | will_link = False | ||||
| for x in feats: | for x in feats: | ||||
| if x in ('cxx', 'd', 'fc', 'c'): | |||||
| if x in ('cxx', 'd', 'fc', 'c', 'asm'): | |||||
| feats.append(x + typ) | feats.append(x + typ) | ||||
| will_link = True | will_link = True | ||||
| if not will_link and not kw.get('features', []): | if not will_link and not kw.get('features', []): | ||||
| raise Errors.WafError('Cannot link from %r, try passing eg: features="c cprogram"?' % kw) | |||||
| raise Errors.WafError('Unable to determine how to link %r, try adding eg: features="c cshlib"?' % kw) | |||||
| return feats | return feats | ||||
| def set_features(kw, typ): | def set_features(kw, typ): | ||||
| @@ -68,6 +68,8 @@ MACRO_TO_DEST_CPU = { | |||||
| '__s390__' : 's390', | '__s390__' : 's390', | ||||
| '__sh__' : 'sh', | '__sh__' : 'sh', | ||||
| '__xtensa__' : 'xtensa', | '__xtensa__' : 'xtensa', | ||||
| '__e2k__' : 'e2k', | |||||
| '__riscv' : 'riscv', | |||||
| } | } | ||||
| @conf | @conf | ||||
| @@ -86,6 +88,10 @@ def parse_flags(self, line, uselib_store, env=None, force_static=False, posix=No | |||||
| :type uselib_store: string | :type uselib_store: string | ||||
| :param env: config set or conf.env by default | :param env: config set or conf.env by default | ||||
| :type env: :py:class:`waflib.ConfigSet.ConfigSet` | :type env: :py:class:`waflib.ConfigSet.ConfigSet` | ||||
| :param force_static: force usage of static libraries | |||||
| :type force_static: bool default False | |||||
| :param posix: usage of POSIX mode for shlex lexical analiysis library | |||||
| :type posix: bool default True | |||||
| """ | """ | ||||
| assert(isinstance(line, str)) | assert(isinstance(line, str)) | ||||
| @@ -103,6 +109,8 @@ def parse_flags(self, line, uselib_store, env=None, force_static=False, posix=No | |||||
| lex.commenters = '' | lex.commenters = '' | ||||
| lst = list(lex) | lst = list(lex) | ||||
| so_re = re.compile(r"\.so(?:\.[0-9]+)*$") | |||||
| # append_unique is not always possible | # append_unique is not always possible | ||||
| # for example, apple flags may require both -arch i386 and -arch ppc | # for example, apple flags may require both -arch i386 and -arch ppc | ||||
| uselib = uselib_store | uselib = uselib_store | ||||
| @@ -144,7 +152,7 @@ def parse_flags(self, line, uselib_store, env=None, force_static=False, posix=No | |||||
| elif x.startswith('-std='): | elif x.startswith('-std='): | ||||
| prefix = 'CXXFLAGS' if '++' in x else 'CFLAGS' | prefix = 'CXXFLAGS' if '++' in x else 'CFLAGS' | ||||
| app(prefix, x) | app(prefix, x) | ||||
| elif x.startswith('+') or x in ('-pthread', '-fPIC', '-fpic', '-fPIE', '-fpie'): | |||||
| elif x.startswith('+') or x in ('-pthread', '-fPIC', '-fpic', '-fPIE', '-fpie', '-flto', '-fno-lto'): | |||||
| app('CFLAGS', x) | app('CFLAGS', x) | ||||
| app('CXXFLAGS', x) | app('CXXFLAGS', x) | ||||
| app('LINKFLAGS', x) | app('LINKFLAGS', x) | ||||
| @@ -180,7 +188,7 @@ def parse_flags(self, line, uselib_store, env=None, force_static=False, posix=No | |||||
| app('CFLAGS', tmp) | app('CFLAGS', tmp) | ||||
| app('CXXFLAGS', tmp) | app('CXXFLAGS', tmp) | ||||
| app('LINKFLAGS', tmp) | app('LINKFLAGS', tmp) | ||||
| elif x.endswith(('.a', '.so', '.dylib', '.lib')): | |||||
| elif x.endswith(('.a', '.dylib', '.lib')) or so_re.search(x): | |||||
| appu('LINKFLAGS', x) # not cool, #762 | appu('LINKFLAGS', x) # not cool, #762 | ||||
| else: | else: | ||||
| self.to_log('Unhandled flag %r' % x) | self.to_log('Unhandled flag %r' % x) | ||||
| @@ -246,13 +254,15 @@ def exec_cfg(self, kw): | |||||
| * if modversion is given, then return the module version | * if modversion is given, then return the module version | ||||
| * else, execute the *-config* program with the *args* and *variables* given, and set the flags on the *conf.env.FLAGS_name* variable | * else, execute the *-config* program with the *args* and *variables* given, and set the flags on the *conf.env.FLAGS_name* variable | ||||
| :param path: the **-config program to use** | |||||
| :type path: list of string | |||||
| :param atleast_pkgconfig_version: minimum pkg-config version to use (disable other tests) | :param atleast_pkgconfig_version: minimum pkg-config version to use (disable other tests) | ||||
| :type atleast_pkgconfig_version: string | :type atleast_pkgconfig_version: string | ||||
| :param package: package name, for example *gtk+-2.0* | :param package: package name, for example *gtk+-2.0* | ||||
| :type package: string | :type package: string | ||||
| :param uselib_store: if the test is successful, define HAVE\_*name*. It is also used to define *conf.env.FLAGS_name* variables. | |||||
| :param uselib_store: if the test is successful, define HAVE\\_*name*. It is also used to define *conf.env.FLAGS_name* variables. | |||||
| :type uselib_store: string | :type uselib_store: string | ||||
| :param modversion: if provided, return the version of the given module and define *name*\_VERSION | |||||
| :param modversion: if provided, return the version of the given module and define *name*\\_VERSION | |||||
| :type modversion: string | :type modversion: string | ||||
| :param args: arguments to give to *package* when retrieving flags | :param args: arguments to give to *package* when retrieving flags | ||||
| :type args: list of string | :type args: list of string | ||||
| @@ -260,6 +270,12 @@ def exec_cfg(self, kw): | |||||
| :type variables: list of string | :type variables: list of string | ||||
| :param define_variable: additional variables to define (also in conf.env.PKG_CONFIG_DEFINES) | :param define_variable: additional variables to define (also in conf.env.PKG_CONFIG_DEFINES) | ||||
| :type define_variable: dict(string: string) | :type define_variable: dict(string: string) | ||||
| :param pkg_config_path: paths where pkg-config should search for .pc config files (overrides env.PKG_CONFIG_PATH if exists) | |||||
| :type pkg_config_path: string, list of directories separated by colon | |||||
| :param force_static: force usage of static libraries | |||||
| :type force_static: bool default False | |||||
| :param posix: usage of POSIX mode for shlex lexical analiysis library | |||||
| :type posix: bool default True | |||||
| """ | """ | ||||
| path = Utils.to_list(kw['path']) | path = Utils.to_list(kw['path']) | ||||
| @@ -334,6 +350,7 @@ def check_cfg(self, *k, **kw): | |||||
| """ | """ | ||||
| Checks for configuration flags using a **-config**-like program (pkg-config, sdl-config, etc). | Checks for configuration flags using a **-config**-like program (pkg-config, sdl-config, etc). | ||||
| This wraps internal calls to :py:func:`waflib.Tools.c_config.validate_cfg` and :py:func:`waflib.Tools.c_config.exec_cfg` | This wraps internal calls to :py:func:`waflib.Tools.c_config.validate_cfg` and :py:func:`waflib.Tools.c_config.exec_cfg` | ||||
| so check exec_cfg parameters descriptions for more details on kw passed | |||||
| A few examples:: | A few examples:: | ||||
| @@ -659,20 +676,21 @@ class test_exec(Task.Task): | |||||
| """ | """ | ||||
| color = 'PINK' | color = 'PINK' | ||||
| def run(self): | def run(self): | ||||
| cmd = [self.inputs[0].abspath()] + getattr(self.generator, 'test_args', []) | |||||
| if getattr(self.generator, 'rpath', None): | if getattr(self.generator, 'rpath', None): | ||||
| if getattr(self.generator, 'define_ret', False): | if getattr(self.generator, 'define_ret', False): | ||||
| self.generator.bld.retval = self.generator.bld.cmd_and_log([self.inputs[0].abspath()]) | |||||
| self.generator.bld.retval = self.generator.bld.cmd_and_log(cmd) | |||||
| else: | else: | ||||
| self.generator.bld.retval = self.generator.bld.exec_command([self.inputs[0].abspath()]) | |||||
| self.generator.bld.retval = self.generator.bld.exec_command(cmd) | |||||
| else: | else: | ||||
| env = self.env.env or {} | env = self.env.env or {} | ||||
| env.update(dict(os.environ)) | env.update(dict(os.environ)) | ||||
| for var in ('LD_LIBRARY_PATH', 'DYLD_LIBRARY_PATH', 'PATH'): | for var in ('LD_LIBRARY_PATH', 'DYLD_LIBRARY_PATH', 'PATH'): | ||||
| env[var] = self.inputs[0].parent.abspath() + os.path.pathsep + env.get(var, '') | env[var] = self.inputs[0].parent.abspath() + os.path.pathsep + env.get(var, '') | ||||
| if getattr(self.generator, 'define_ret', False): | if getattr(self.generator, 'define_ret', False): | ||||
| self.generator.bld.retval = self.generator.bld.cmd_and_log([self.inputs[0].abspath()], env=env) | |||||
| self.generator.bld.retval = self.generator.bld.cmd_and_log(cmd, env=env) | |||||
| else: | else: | ||||
| self.generator.bld.retval = self.generator.bld.exec_command([self.inputs[0].abspath()], env=env) | |||||
| self.generator.bld.retval = self.generator.bld.exec_command(cmd, env=env) | |||||
| @feature('test_exec') | @feature('test_exec') | ||||
| @after_method('apply_link') | @after_method('apply_link') | ||||
| @@ -1266,10 +1284,11 @@ def multicheck(self, *k, **kw): | |||||
| tasks = [] | tasks = [] | ||||
| id_to_task = {} | id_to_task = {} | ||||
| for dct in k: | |||||
| for counter, dct in enumerate(k): | |||||
| x = Task.classes['cfgtask'](bld=bld, env=None) | x = Task.classes['cfgtask'](bld=bld, env=None) | ||||
| tasks.append(x) | tasks.append(x) | ||||
| x.args = dct | x.args = dct | ||||
| x.args['multicheck_counter'] = counter | |||||
| x.bld = bld | x.bld = bld | ||||
| x.conf = self | x.conf = self | ||||
| x.args = dct | x.args = dct | ||||
| @@ -75,13 +75,13 @@ re_lines = re.compile( | |||||
| re.IGNORECASE | re.MULTILINE) | re.IGNORECASE | re.MULTILINE) | ||||
| """Match #include lines""" | """Match #include lines""" | ||||
| re_mac = re.compile("^[a-zA-Z_]\w*") | |||||
| re_mac = re.compile(r"^[a-zA-Z_]\w*") | |||||
| """Match macro definitions""" | """Match macro definitions""" | ||||
| re_fun = re.compile('^[a-zA-Z_][a-zA-Z0-9_]*[(]') | re_fun = re.compile('^[a-zA-Z_][a-zA-Z0-9_]*[(]') | ||||
| """Match macro functions""" | """Match macro functions""" | ||||
| re_pragma_once = re.compile('^\s*once\s*', re.IGNORECASE) | |||||
| re_pragma_once = re.compile(r'^\s*once\s*', re.IGNORECASE) | |||||
| """Match #pragma once statements""" | """Match #pragma once statements""" | ||||
| re_nl = re.compile('\\\\\r*\n', re.MULTILINE) | re_nl = re.compile('\\\\\r*\n', re.MULTILINE) | ||||
| @@ -660,7 +660,7 @@ def extract_macro(txt): | |||||
| # empty define, assign an empty token | # empty define, assign an empty token | ||||
| return (v, [[], [('T','')]]) | return (v, [[], [('T','')]]) | ||||
| re_include = re.compile('^\s*(<(?:.*)>|"(?:.*)")') | |||||
| re_include = re.compile(r'^\s*(<(?:.*)>|"(?:.*)")') | |||||
| def extract_include(txt, defs): | def extract_include(txt, defs): | ||||
| """ | """ | ||||
| Process a line in the form:: | Process a line in the form:: | ||||
| @@ -180,9 +180,15 @@ def check_large_file(self, **kw): | |||||
| ######################################################################################## | ######################################################################################## | ||||
| ENDIAN_FRAGMENT = ''' | ENDIAN_FRAGMENT = ''' | ||||
| #ifdef _MSC_VER | |||||
| #define testshlib_EXPORT __declspec(dllexport) | |||||
| #else | |||||
| #define testshlib_EXPORT | |||||
| #endif | |||||
| short int ascii_mm[] = { 0x4249, 0x4765, 0x6E44, 0x6961, 0x6E53, 0x7953, 0 }; | short int ascii_mm[] = { 0x4249, 0x4765, 0x6E44, 0x6961, 0x6E53, 0x7953, 0 }; | ||||
| short int ascii_ii[] = { 0x694C, 0x5454, 0x656C, 0x6E45, 0x6944, 0x6E61, 0 }; | short int ascii_ii[] = { 0x694C, 0x5454, 0x656C, 0x6E45, 0x6944, 0x6E61, 0 }; | ||||
| int use_ascii (int i) { | |||||
| int testshlib_EXPORT use_ascii (int i) { | |||||
| return ascii_mm[i] + ascii_ii[i]; | return ascii_mm[i] + ascii_ii[i]; | ||||
| } | } | ||||
| short int ebcdic_ii[] = { 0x89D3, 0xE3E3, 0x8593, 0x95C5, 0x89C4, 0x9581, 0 }; | short int ebcdic_ii[] = { 0x89D3, 0xE3E3, 0x8593, 0x95C5, 0x89C4, 0x9581, 0 }; | ||||
| @@ -208,12 +214,12 @@ class grep_for_endianness(Task.Task): | |||||
| return -1 | return -1 | ||||
| @feature('grep_for_endianness') | @feature('grep_for_endianness') | ||||
| @after_method('process_source') | |||||
| @after_method('apply_link') | |||||
| def grep_for_endianness_fun(self): | def grep_for_endianness_fun(self): | ||||
| """ | """ | ||||
| Used by the endianness configuration test | Used by the endianness configuration test | ||||
| """ | """ | ||||
| self.create_task('grep_for_endianness', self.compiled_tasks[0].outputs[0]) | |||||
| self.create_task('grep_for_endianness', self.link_task.outputs[0]) | |||||
| @conf | @conf | ||||
| def check_endianness(self): | def check_endianness(self): | ||||
| @@ -223,7 +229,9 @@ def check_endianness(self): | |||||
| tmp = [] | tmp = [] | ||||
| def check_msg(self): | def check_msg(self): | ||||
| return tmp[0] | return tmp[0] | ||||
| self.check(fragment=ENDIAN_FRAGMENT, features='c grep_for_endianness', | |||||
| msg='Checking for endianness', define='ENDIANNESS', tmp=tmp, okmsg=check_msg) | |||||
| self.check(fragment=ENDIAN_FRAGMENT, features='c cshlib grep_for_endianness', | |||||
| msg='Checking for endianness', define='ENDIANNESS', tmp=tmp, | |||||
| okmsg=check_msg, confcache=None) | |||||
| return tmp[0] | return tmp[0] | ||||
| @@ -111,7 +111,7 @@ def apply_incpaths(self): | |||||
| tg = bld(features='includes', includes='.') | tg = bld(features='includes', includes='.') | ||||
| The folders only need to be relative to the current directory, the equivalent build directory is | The folders only need to be relative to the current directory, the equivalent build directory is | ||||
| added automatically (for headers created in the build directory). This enable using a build directory | |||||
| added automatically (for headers created in the build directory). This enables using a build directory | |||||
| or not (``top == out``). | or not (``top == out``). | ||||
| This method will add a list of nodes read by :py:func:`waflib.Tools.ccroot.to_incnodes` in ``tg.env.INCPATHS``, | This method will add a list of nodes read by :py:func:`waflib.Tools.ccroot.to_incnodes` in ``tg.env.INCPATHS``, | ||||
| @@ -128,6 +128,7 @@ class link_task(Task.Task): | |||||
| Base class for all link tasks. A task generator is supposed to have at most one link task bound in the attribute *link_task*. See :py:func:`waflib.Tools.ccroot.apply_link`. | Base class for all link tasks. A task generator is supposed to have at most one link task bound in the attribute *link_task*. See :py:func:`waflib.Tools.ccroot.apply_link`. | ||||
| .. inheritance-diagram:: waflib.Tools.ccroot.stlink_task waflib.Tools.c.cprogram waflib.Tools.c.cshlib waflib.Tools.cxx.cxxstlib waflib.Tools.cxx.cxxprogram waflib.Tools.cxx.cxxshlib waflib.Tools.d.dprogram waflib.Tools.d.dshlib waflib.Tools.d.dstlib waflib.Tools.ccroot.fake_shlib waflib.Tools.ccroot.fake_stlib waflib.Tools.asm.asmprogram waflib.Tools.asm.asmshlib waflib.Tools.asm.asmstlib | .. inheritance-diagram:: waflib.Tools.ccroot.stlink_task waflib.Tools.c.cprogram waflib.Tools.c.cshlib waflib.Tools.cxx.cxxstlib waflib.Tools.cxx.cxxprogram waflib.Tools.cxx.cxxshlib waflib.Tools.d.dprogram waflib.Tools.d.dshlib waflib.Tools.d.dstlib waflib.Tools.ccroot.fake_shlib waflib.Tools.ccroot.fake_stlib waflib.Tools.asm.asmprogram waflib.Tools.asm.asmshlib waflib.Tools.asm.asmstlib | ||||
| :top-classes: waflib.Tools.ccroot.link_task | |||||
| """ | """ | ||||
| color = 'YELLOW' | color = 'YELLOW' | ||||
| @@ -238,6 +239,17 @@ def rm_tgt(cls): | |||||
| setattr(cls, 'run', wrap) | setattr(cls, 'run', wrap) | ||||
| rm_tgt(stlink_task) | rm_tgt(stlink_task) | ||||
| @feature('skip_stlib_link_deps') | |||||
| @before_method('process_use') | |||||
| def apply_skip_stlib_link_deps(self): | |||||
| """ | |||||
| This enables an optimization in the :py:func:wafilb.Tools.ccroot.processes_use: method that skips dependency and | |||||
| link flag optimizations for targets that generate static libraries (via the :py:class:Tools.ccroot.stlink_task task). | |||||
| The actual behavior is implemented in :py:func:wafilb.Tools.ccroot.processes_use: method so this feature only tells waf | |||||
| to enable the new behavior. | |||||
| """ | |||||
| self.env.SKIP_STLIB_LINK_DEPS = True | |||||
| @feature('c', 'cxx', 'd', 'fc', 'asm') | @feature('c', 'cxx', 'd', 'fc', 'asm') | ||||
| @after_method('process_source') | @after_method('process_source') | ||||
| def apply_link(self): | def apply_link(self): | ||||
| @@ -386,7 +398,11 @@ def process_use(self): | |||||
| y = self.bld.get_tgen_by_name(x) | y = self.bld.get_tgen_by_name(x) | ||||
| var = y.tmp_use_var | var = y.tmp_use_var | ||||
| if var and link_task: | if var and link_task: | ||||
| if var == 'LIB' or y.tmp_use_stlib or x in names: | |||||
| if self.env.SKIP_STLIB_LINK_DEPS and isinstance(link_task, stlink_task): | |||||
| # If the skip_stlib_link_deps feature is enabled then we should | |||||
| # avoid adding lib deps to the stlink_task instance. | |||||
| pass | |||||
| elif var == 'LIB' or y.tmp_use_stlib or x in names: | |||||
| self.env.append_value(var, [y.target[y.target.rfind(os.sep) + 1:]]) | self.env.append_value(var, [y.target[y.target.rfind(os.sep) + 1:]]) | ||||
| self.link_task.dep_nodes.extend(y.link_task.outputs) | self.link_task.dep_nodes.extend(y.link_task.outputs) | ||||
| tmp_path = y.link_task.outputs[0].parent.path_from(self.get_cwd()) | tmp_path = y.link_task.outputs[0].parent.path_from(self.get_cwd()) | ||||
| @@ -36,18 +36,19 @@ from waflib import Utils | |||||
| from waflib.Logs import debug | from waflib.Logs import debug | ||||
| c_compiler = { | c_compiler = { | ||||
| 'win32': ['msvc', 'gcc', 'clang'], | |||||
| 'cygwin': ['gcc'], | |||||
| 'darwin': ['clang', 'gcc'], | |||||
| 'aix': ['xlc', 'gcc', 'clang'], | |||||
| 'linux': ['gcc', 'clang', 'icc'], | |||||
| 'sunos': ['suncc', 'gcc'], | |||||
| 'irix': ['gcc', 'irixcc'], | |||||
| 'hpux': ['gcc'], | |||||
| 'osf1V': ['gcc'], | |||||
| 'gnu': ['gcc', 'clang'], | |||||
| 'java': ['gcc', 'msvc', 'clang', 'icc'], | |||||
| 'default':['clang', 'gcc'], | |||||
| 'win32': ['msvc', 'gcc', 'clang'], | |||||
| 'cygwin': ['gcc', 'clang'], | |||||
| 'darwin': ['clang', 'gcc'], | |||||
| 'aix': ['xlc', 'gcc', 'clang'], | |||||
| 'linux': ['gcc', 'clang', 'icc'], | |||||
| 'sunos': ['suncc', 'gcc'], | |||||
| 'irix': ['gcc', 'irixcc'], | |||||
| 'hpux': ['gcc'], | |||||
| 'osf1V': ['gcc'], | |||||
| 'gnu': ['gcc', 'clang'], | |||||
| 'java': ['gcc', 'msvc', 'clang', 'icc'], | |||||
| 'gnukfreebsd': ['gcc', 'clang'], | |||||
| 'default': ['clang', 'gcc'], | |||||
| } | } | ||||
| """ | """ | ||||
| Dict mapping platform names to Waf tools finding specific C compilers:: | Dict mapping platform names to Waf tools finding specific C compilers:: | ||||
| @@ -37,18 +37,19 @@ from waflib import Utils | |||||
| from waflib.Logs import debug | from waflib.Logs import debug | ||||
| cxx_compiler = { | cxx_compiler = { | ||||
| 'win32': ['msvc', 'g++', 'clang++'], | |||||
| 'cygwin': ['g++'], | |||||
| 'darwin': ['clang++', 'g++'], | |||||
| 'aix': ['xlc++', 'g++', 'clang++'], | |||||
| 'linux': ['g++', 'clang++', 'icpc'], | |||||
| 'sunos': ['sunc++', 'g++'], | |||||
| 'irix': ['g++'], | |||||
| 'hpux': ['g++'], | |||||
| 'osf1V': ['g++'], | |||||
| 'gnu': ['g++', 'clang++'], | |||||
| 'java': ['g++', 'msvc', 'clang++', 'icpc'], | |||||
| 'default': ['clang++', 'g++'] | |||||
| 'win32': ['msvc', 'g++', 'clang++'], | |||||
| 'cygwin': ['g++', 'clang++'], | |||||
| 'darwin': ['clang++', 'g++'], | |||||
| 'aix': ['xlc++', 'g++', 'clang++'], | |||||
| 'linux': ['g++', 'clang++', 'icpc'], | |||||
| 'sunos': ['sunc++', 'g++'], | |||||
| 'irix': ['g++'], | |||||
| 'hpux': ['g++'], | |||||
| 'osf1V': ['g++'], | |||||
| 'gnu': ['g++', 'clang++'], | |||||
| 'java': ['g++', 'msvc', 'clang++', 'icpc'], | |||||
| 'gnukfreebsd': ['g++', 'clang++'], | |||||
| 'default': ['clang++', 'g++'] | |||||
| } | } | ||||
| """ | """ | ||||
| Dict mapping the platform names to Waf tools finding specific C++ compilers:: | Dict mapping the platform names to Waf tools finding specific C++ compilers:: | ||||
| @@ -13,22 +13,11 @@ from waflib.Configure import conf | |||||
| @conf | @conf | ||||
| def find_irixcc(conf): | def find_irixcc(conf): | ||||
| v = conf.env | v = conf.env | ||||
| cc = None | |||||
| if v.CC: | |||||
| cc = v.CC | |||||
| elif 'CC' in conf.environ: | |||||
| cc = conf.environ['CC'] | |||||
| if not cc: | |||||
| cc = conf.find_program('cc', var='CC') | |||||
| if not cc: | |||||
| conf.fatal('irixcc was not found') | |||||
| cc = conf.find_program('cc', var='CC') | |||||
| try: | try: | ||||
| conf.cmd_and_log(cc + ['-version']) | conf.cmd_and_log(cc + ['-version']) | ||||
| except Errors.WafError: | except Errors.WafError: | ||||
| conf.fatal('%r -version could not be executed' % cc) | conf.fatal('%r -version could not be executed' % cc) | ||||
| v.CC = cc | |||||
| v.CC_NAME = 'irix' | v.CC_NAME = 'irix' | ||||
| @conf | @conf | ||||
| @@ -57,7 +46,6 @@ def irixcc_common_flags(conf): | |||||
| def configure(conf): | def configure(conf): | ||||
| conf.find_irixcc() | conf.find_irixcc() | ||||
| conf.find_cpp() | |||||
| conf.find_ar() | conf.find_ar() | ||||
| conf.irixcc_common_flags() | conf.irixcc_common_flags() | ||||
| conf.cc_load_tools() | conf.cc_load_tools() | ||||
| @@ -99,10 +99,31 @@ all_icl_platforms = [ ('intel64', 'amd64'), ('em64t', 'amd64'), ('ia32', 'x86'), | |||||
| """List of icl platforms""" | """List of icl platforms""" | ||||
| def options(opt): | def options(opt): | ||||
| opt.add_option('--msvc_version', type='string', help = 'msvc version, eg: "msvc 10.0,msvc 9.0"', default='') | |||||
| default_ver = '' | |||||
| vsver = os.getenv('VSCMD_VER') | |||||
| if vsver: | |||||
| m = re.match(r'(^\d+\.\d+).*', vsver) | |||||
| if m: | |||||
| default_ver = 'msvc %s' % m.group(1) | |||||
| opt.add_option('--msvc_version', type='string', help = 'msvc version, eg: "msvc 10.0,msvc 9.0"', default=default_ver) | |||||
| opt.add_option('--msvc_targets', type='string', help = 'msvc targets, eg: "x64,arm"', default='') | opt.add_option('--msvc_targets', type='string', help = 'msvc targets, eg: "x64,arm"', default='') | ||||
| opt.add_option('--no-msvc-lazy', action='store_false', help = 'lazily check msvc target environments', default=True, dest='msvc_lazy') | opt.add_option('--no-msvc-lazy', action='store_false', help = 'lazily check msvc target environments', default=True, dest='msvc_lazy') | ||||
| class MSVCVersion(object): | |||||
| def __init__(self, ver): | |||||
| m = re.search(r'^(.*)\s+(\d+[.]\d+)', ver) | |||||
| if m: | |||||
| self.name = m.group(1) | |||||
| self.number = float(m.group(2)) | |||||
| else: | |||||
| self.name = ver | |||||
| self.number = 0. | |||||
| def __lt__(self, other): | |||||
| if self.number == other.number: | |||||
| return self.name < other.name | |||||
| return self.number < other.number | |||||
| @conf | @conf | ||||
| def setup_msvc(conf, versiondict): | def setup_msvc(conf, versiondict): | ||||
| """ | """ | ||||
| @@ -119,7 +140,7 @@ def setup_msvc(conf, versiondict): | |||||
| platforms=Utils.to_list(conf.env.MSVC_TARGETS) or [i for i,j in all_msvc_platforms+all_icl_platforms+all_wince_platforms] | platforms=Utils.to_list(conf.env.MSVC_TARGETS) or [i for i,j in all_msvc_platforms+all_icl_platforms+all_wince_platforms] | ||||
| desired_versions = getattr(Options.options, 'msvc_version', '').split(',') | desired_versions = getattr(Options.options, 'msvc_version', '').split(',') | ||||
| if desired_versions == ['']: | if desired_versions == ['']: | ||||
| desired_versions = conf.env.MSVC_VERSIONS or list(reversed(sorted(versiondict.keys()))) | |||||
| desired_versions = conf.env.MSVC_VERSIONS or list(sorted(versiondict.keys(), key=MSVCVersion, reverse=True)) | |||||
| # Override lazy detection by evaluating after the fact. | # Override lazy detection by evaluating after the fact. | ||||
| lazy_detect = getattr(Options.options, 'msvc_lazy', True) | lazy_detect = getattr(Options.options, 'msvc_lazy', True) | ||||
| @@ -187,7 +208,7 @@ echo PATH=%%PATH%% | |||||
| echo INCLUDE=%%INCLUDE%% | echo INCLUDE=%%INCLUDE%% | ||||
| echo LIB=%%LIB%%;%%LIBPATH%% | echo LIB=%%LIB%%;%%LIBPATH%% | ||||
| """ % (vcvars,target)) | """ % (vcvars,target)) | ||||
| sout = conf.cmd_and_log(['cmd.exe', '/E:on', '/V:on', '/C', batfile.abspath()]) | |||||
| sout = conf.cmd_and_log(['cmd.exe', '/E:on', '/V:on', '/C', batfile.abspath()], stdin=getattr(Utils.subprocess, 'DEVNULL', None)) | |||||
| lines = sout.splitlines() | lines = sout.splitlines() | ||||
| if not lines[0]: | if not lines[0]: | ||||
| @@ -281,7 +302,7 @@ def gather_wince_supported_platforms(): | |||||
| def gather_msvc_detected_versions(): | def gather_msvc_detected_versions(): | ||||
| #Detected MSVC versions! | #Detected MSVC versions! | ||||
| version_pattern = re.compile('^(\d\d?\.\d\d?)(Exp)?$') | |||||
| version_pattern = re.compile(r'^(\d\d?\.\d\d?)(Exp)?$') | |||||
| detected_versions = [] | detected_versions = [] | ||||
| for vcver,vcvar in (('VCExpress','Exp'), ('VisualStudio','')): | for vcver,vcvar in (('VCExpress','Exp'), ('VisualStudio','')): | ||||
| prefix = 'SOFTWARE\\Wow6432node\\Microsoft\\' + vcver | prefix = 'SOFTWARE\\Wow6432node\\Microsoft\\' + vcver | ||||
| @@ -367,7 +388,7 @@ def gather_wsdk_versions(conf, versions): | |||||
| :param versions: list to modify | :param versions: list to modify | ||||
| :type versions: list | :type versions: list | ||||
| """ | """ | ||||
| version_pattern = re.compile('^v..?.?\...?.?') | |||||
| version_pattern = re.compile(r'^v..?.?\...?.?') | |||||
| try: | try: | ||||
| all_versions = Utils.winreg.OpenKey(Utils.winreg.HKEY_LOCAL_MACHINE, 'SOFTWARE\\Wow6432node\\Microsoft\\Microsoft SDKs\\Windows') | all_versions = Utils.winreg.OpenKey(Utils.winreg.HKEY_LOCAL_MACHINE, 'SOFTWARE\\Wow6432node\\Microsoft\\Microsoft SDKs\\Windows') | ||||
| except OSError: | except OSError: | ||||
| @@ -525,7 +546,7 @@ def gather_icl_versions(conf, versions): | |||||
| :param versions: list to modify | :param versions: list to modify | ||||
| :type versions: list | :type versions: list | ||||
| """ | """ | ||||
| version_pattern = re.compile('^...?.?\....?.?') | |||||
| version_pattern = re.compile(r'^...?.?\....?.?') | |||||
| try: | try: | ||||
| all_versions = Utils.winreg.OpenKey(Utils.winreg.HKEY_LOCAL_MACHINE, 'SOFTWARE\\Wow6432node\\Intel\\Compilers\\C++') | all_versions = Utils.winreg.OpenKey(Utils.winreg.HKEY_LOCAL_MACHINE, 'SOFTWARE\\Wow6432node\\Intel\\Compilers\\C++') | ||||
| except OSError: | except OSError: | ||||
| @@ -579,7 +600,7 @@ def gather_intel_composer_versions(conf, versions): | |||||
| :param versions: list to modify | :param versions: list to modify | ||||
| :type versions: list | :type versions: list | ||||
| """ | """ | ||||
| version_pattern = re.compile('^...?.?\...?.?.?') | |||||
| version_pattern = re.compile(r'^...?.?\...?.?.?') | |||||
| try: | try: | ||||
| all_versions = Utils.winreg.OpenKey(Utils.winreg.HKEY_LOCAL_MACHINE, 'SOFTWARE\\Wow6432node\\Intel\\Suites') | all_versions = Utils.winreg.OpenKey(Utils.winreg.HKEY_LOCAL_MACHINE, 'SOFTWARE\\Wow6432node\\Intel\\Suites') | ||||
| except OSError: | except OSError: | ||||
| @@ -683,7 +704,7 @@ def find_lt_names_msvc(self, libname, is_static=False): | |||||
| if not is_static and ltdict.get('library_names', ''): | if not is_static and ltdict.get('library_names', ''): | ||||
| dllnames=ltdict['library_names'].split() | dllnames=ltdict['library_names'].split() | ||||
| dll=dllnames[0].lower() | dll=dllnames[0].lower() | ||||
| dll=re.sub('\.dll$', '', dll) | |||||
| dll=re.sub(r'\.dll$', '', dll) | |||||
| return (lt_libdir, dll, False) | return (lt_libdir, dll, False) | ||||
| elif ltdict.get('old_library', ''): | elif ltdict.get('old_library', ''): | ||||
| olib=ltdict['old_library'] | olib=ltdict['old_library'] | ||||
| @@ -700,7 +721,7 @@ def find_lt_names_msvc(self, libname, is_static=False): | |||||
| @conf | @conf | ||||
| def libname_msvc(self, libname, is_static=False): | def libname_msvc(self, libname, is_static=False): | ||||
| lib = libname.lower() | lib = libname.lower() | ||||
| lib = re.sub('\.lib$','',lib) | |||||
| lib = re.sub(r'\.lib$','',lib) | |||||
| if lib in g_msvc_systemlibs: | if lib in g_msvc_systemlibs: | ||||
| return lib | return lib | ||||
| @@ -747,11 +768,11 @@ def libname_msvc(self, libname, is_static=False): | |||||
| for libn in libnames: | for libn in libnames: | ||||
| if os.path.exists(os.path.join(path, libn)): | if os.path.exists(os.path.join(path, libn)): | ||||
| Logs.debug('msvc: lib found: %s', os.path.join(path,libn)) | Logs.debug('msvc: lib found: %s', os.path.join(path,libn)) | ||||
| return re.sub('\.lib$', '',libn) | |||||
| return re.sub(r'\.lib$', '',libn) | |||||
| #if no lib can be found, just return the libname as msvc expects it | #if no lib can be found, just return the libname as msvc expects it | ||||
| self.fatal('The library %r could not be found' % libname) | self.fatal('The library %r could not be found' % libname) | ||||
| return re.sub('\.lib$', '', libname) | |||||
| return re.sub(r'\.lib$', '', libname) | |||||
| @conf | @conf | ||||
| def check_lib_msvc(self, libname, is_static=False, uselib_store=None): | def check_lib_msvc(self, libname, is_static=False, uselib_store=None): | ||||
| @@ -969,7 +990,7 @@ def apply_flags_msvc(self): | |||||
| if not is_static: | if not is_static: | ||||
| for f in self.env.LINKFLAGS: | for f in self.env.LINKFLAGS: | ||||
| d = f.lower() | d = f.lower() | ||||
| if d[1:] == 'debug': | |||||
| if d[1:] in ('debug', 'debug:full', 'debug:fastlink'): | |||||
| pdbnode = self.link_task.outputs[0].change_ext('.pdb') | pdbnode = self.link_task.outputs[0].change_ext('.pdb') | ||||
| self.link_task.outputs.append(pdbnode) | self.link_task.outputs.append(pdbnode) | ||||
| @@ -97,6 +97,7 @@ def make_interpreted_test(self): | |||||
| if isinstance(v, str): | if isinstance(v, str): | ||||
| v = v.split(os.pathsep) | v = v.split(os.pathsep) | ||||
| self.ut_env[k] = os.pathsep.join(p + v) | self.ut_env[k] = os.pathsep.join(p + v) | ||||
| self.env.append_value('UT_DEPS', ['%r%r' % (key, self.ut_env[key]) for key in self.ut_env]) | |||||
| @feature('test') | @feature('test') | ||||
| @after_method('apply_link', 'process_use') | @after_method('apply_link', 'process_use') | ||||
| @@ -108,7 +109,8 @@ def make_test(self): | |||||
| tsk = self.create_task('utest', self.link_task.outputs) | tsk = self.create_task('utest', self.link_task.outputs) | ||||
| if getattr(self, 'ut_str', None): | if getattr(self, 'ut_str', None): | ||||
| self.ut_run, lst = Task.compile_fun(self.ut_str, shell=getattr(self, 'ut_shell', False)) | self.ut_run, lst = Task.compile_fun(self.ut_str, shell=getattr(self, 'ut_shell', False)) | ||||
| tsk.vars = lst + tsk.vars | |||||
| tsk.vars = tsk.vars + lst | |||||
| self.env.append_value('UT_DEPS', self.ut_str) | |||||
| self.handle_ut_cwd('ut_cwd') | self.handle_ut_cwd('ut_cwd') | ||||
| @@ -139,6 +141,10 @@ def make_test(self): | |||||
| if not hasattr(self, 'ut_cmd'): | if not hasattr(self, 'ut_cmd'): | ||||
| self.ut_cmd = getattr(Options.options, 'testcmd', False) | self.ut_cmd = getattr(Options.options, 'testcmd', False) | ||||
| self.env.append_value('UT_DEPS', str(self.ut_cmd)) | |||||
| self.env.append_value('UT_DEPS', self.ut_paths) | |||||
| self.env.append_value('UT_DEPS', ['%r%r' % (key, self.ut_env[key]) for key in self.ut_env]) | |||||
| @taskgen_method | @taskgen_method | ||||
| def add_test_results(self, tup): | def add_test_results(self, tup): | ||||
| """Override and return tup[1] to interrupt the build immediately if a test does not run""" | """Override and return tup[1] to interrupt the build immediately if a test does not run""" | ||||
| @@ -159,7 +165,7 @@ class utest(Task.Task): | |||||
| """ | """ | ||||
| color = 'PINK' | color = 'PINK' | ||||
| after = ['vnum', 'inst'] | after = ['vnum', 'inst'] | ||||
| vars = [] | |||||
| vars = ['UT_DEPS'] | |||||
| def runnable_status(self): | def runnable_status(self): | ||||
| """ | """ | ||||
| @@ -200,7 +206,7 @@ class utest(Task.Task): | |||||
| self.ut_exec = getattr(self.generator, 'ut_exec', [self.inputs[0].abspath()]) | self.ut_exec = getattr(self.generator, 'ut_exec', [self.inputs[0].abspath()]) | ||||
| ut_cmd = getattr(self.generator, 'ut_cmd', False) | ut_cmd = getattr(self.generator, 'ut_cmd', False) | ||||
| if ut_cmd: | if ut_cmd: | ||||
| self.ut_exec = shlex.split(ut_cmd % ' '.join(self.ut_exec)) | |||||
| self.ut_exec = shlex.split(ut_cmd % Utils.shell_escape(self.ut_exec)) | |||||
| return self.exec_command(self.ut_exec) | return self.exec_command(self.ut_exec) | ||||
| @@ -214,7 +220,7 @@ class utest(Task.Task): | |||||
| 'cmd': cmd | 'cmd': cmd | ||||
| } | } | ||||
| script_file = self.inputs[0].abspath() + '_run.py' | script_file = self.inputs[0].abspath() + '_run.py' | ||||
| Utils.writef(script_file, script_code) | |||||
| Utils.writef(script_file, script_code, encoding='utf-8') | |||||
| os.chmod(script_file, Utils.O755) | os.chmod(script_file, Utils.O755) | ||||
| if Logs.verbose > 1: | if Logs.verbose > 1: | ||||
| Logs.info('Test debug file written as %r' % script_file) | Logs.info('Test debug file written as %r' % script_file) | ||||
| @@ -11,7 +11,7 @@ through Python versions 2.5 to 3.X and across different platforms (win32, linux, | |||||
| from __future__ import with_statement | from __future__ import with_statement | ||||
| import atexit, os, sys, errno, inspect, re, datetime, platform, base64, signal, functools, time | |||||
| import atexit, os, sys, errno, inspect, re, datetime, platform, base64, signal, functools, time, shlex | |||||
| try: | try: | ||||
| import cPickle | import cPickle | ||||
| @@ -49,10 +49,16 @@ try: | |||||
| from hashlib import md5 | from hashlib import md5 | ||||
| except ImportError: | except ImportError: | ||||
| try: | try: | ||||
| from md5 import md5 | |||||
| from hashlib import sha1 as md5 | |||||
| except ImportError: | except ImportError: | ||||
| # never fail to enable fixes from another module | |||||
| # never fail to enable potential fixes from another module | |||||
| pass | pass | ||||
| else: | |||||
| try: | |||||
| md5().digest() | |||||
| except ValueError: | |||||
| # Fips? #2213 | |||||
| from hashlib import sha1 as md5 | |||||
| try: | try: | ||||
| import threading | import threading | ||||
| @@ -202,7 +208,7 @@ class lazy_generator(object): | |||||
| next = __next__ | next = __next__ | ||||
| is_win32 = os.sep == '\\' or sys.platform == 'win32' # msys2 | |||||
| is_win32 = os.sep == '\\' or sys.platform == 'win32' or os.name == 'nt' # msys2 | |||||
| """ | """ | ||||
| Whether this system is a Windows series | Whether this system is a Windows series | ||||
| """ | """ | ||||
| @@ -446,6 +452,8 @@ def console_encoding(): | |||||
| pass | pass | ||||
| else: | else: | ||||
| if codepage: | if codepage: | ||||
| if 65001 == codepage and sys.version_info < (3, 3): | |||||
| return 'utf-8' | |||||
| return 'cp%d' % codepage | return 'cp%d' % codepage | ||||
| return sys.stdout.encoding or ('cp1252' if is_win32 else 'latin-1') | return sys.stdout.encoding or ('cp1252' if is_win32 else 'latin-1') | ||||
| @@ -484,7 +492,9 @@ def split_path_msys(path): | |||||
| if sys.platform == 'cygwin': | if sys.platform == 'cygwin': | ||||
| split_path = split_path_cygwin | split_path = split_path_cygwin | ||||
| elif is_win32: | elif is_win32: | ||||
| if os.environ.get('MSYSTEM'): | |||||
| # Consider this an MSYSTEM environment if $MSYSTEM is set and python | |||||
| # reports is executable from a unix like path on a windows host. | |||||
| if os.environ.get('MSYSTEM') and sys.executable.startswith('/'): | |||||
| split_path = split_path_msys | split_path = split_path_msys | ||||
| else: | else: | ||||
| split_path = split_path_win32 | split_path = split_path_win32 | ||||
| @@ -569,10 +579,13 @@ def quote_define_name(s): | |||||
| fu = fu.upper() | fu = fu.upper() | ||||
| return fu | return fu | ||||
| re_sh = re.compile('\\s|\'|"') | |||||
| """ | |||||
| Regexp used for shell_escape below | |||||
| """ | |||||
| # shlex.quote didn't exist until python 3.3. Prior to that it was a non-documented | |||||
| # function in pipes. | |||||
| try: | |||||
| shell_quote = shlex.quote | |||||
| except AttributeError: | |||||
| import pipes | |||||
| shell_quote = pipes.quote | |||||
| def shell_escape(cmd): | def shell_escape(cmd): | ||||
| """ | """ | ||||
| @@ -581,7 +594,7 @@ def shell_escape(cmd): | |||||
| """ | """ | ||||
| if isinstance(cmd, str): | if isinstance(cmd, str): | ||||
| return cmd | return cmd | ||||
| return ' '.join(repr(x) if re_sh.search(x) else x for x in cmd) | |||||
| return ' '.join(shell_quote(x) for x in cmd) | |||||
| def h_list(lst): | def h_list(lst): | ||||
| """ | """ | ||||
| @@ -596,6 +609,12 @@ def h_list(lst): | |||||
| """ | """ | ||||
| return md5(repr(lst).encode()).digest() | return md5(repr(lst).encode()).digest() | ||||
| if sys.hexversion < 0x3000000: | |||||
| def h_list_python2(lst): | |||||
| return md5(repr(lst)).digest() | |||||
| h_list_python2.__doc__ = h_list.__doc__ | |||||
| h_list = h_list_python2 | |||||
| def h_fun(fun): | def h_fun(fun): | ||||
| """ | """ | ||||
| Hash functions | Hash functions | ||||
| @@ -615,7 +634,7 @@ def h_fun(fun): | |||||
| # | # | ||||
| # The sorting result outcome will be consistent because: | # The sorting result outcome will be consistent because: | ||||
| # 1. tuples are compared in order of their elements | # 1. tuples are compared in order of their elements | ||||
| # 2. optional argument names are unique | |||||
| # 2. optional argument namess are unique | |||||
| code.extend(sorted(fun.keywords.items())) | code.extend(sorted(fun.keywords.items())) | ||||
| code.append(h_fun(fun.func)) | code.append(h_fun(fun.func)) | ||||
| fun.code = h_list(code) | fun.code = h_list(code) | ||||
| @@ -730,7 +749,7 @@ def unversioned_sys_platform(): | |||||
| if s == 'cli' and os.name == 'nt': | if s == 'cli' and os.name == 'nt': | ||||
| # ironpython is only on windows as far as we know | # ironpython is only on windows as far as we know | ||||
| return 'win32' | return 'win32' | ||||
| return re.split('\d+$', s)[0] | |||||
| return re.split(r'\d+$', s)[0] | |||||
| def nada(*k, **kw): | def nada(*k, **kw): | ||||
| """ | """ | ||||
| @@ -851,6 +870,19 @@ def lib64(): | |||||
| return '64' | return '64' | ||||
| return '' | return '' | ||||
| def loose_version(ver_str): | |||||
| # private for the time being! | |||||
| # see #2402 | |||||
| lst = re.split(r'([.]|\\d+|[a-zA-Z])', ver_str) | |||||
| ver = [] | |||||
| for i, val in enumerate(lst): | |||||
| try: | |||||
| ver.append(int(val)) | |||||
| except ValueError: | |||||
| if val != '.': | |||||
| ver.append(val) | |||||
| return ver | |||||
| def sane_path(p): | def sane_path(p): | ||||
| # private function for the time being! | # private function for the time being! | ||||
| return os.path.abspath(os.path.expanduser(p)) | return os.path.abspath(os.path.expanduser(p)) | ||||
| @@ -871,13 +903,13 @@ def get_process(): | |||||
| except IndexError: | except IndexError: | ||||
| filepath = os.path.dirname(os.path.abspath(__file__)) + os.sep + 'processor.py' | filepath = os.path.dirname(os.path.abspath(__file__)) + os.sep + 'processor.py' | ||||
| cmd = [sys.executable, '-c', readf(filepath)] | cmd = [sys.executable, '-c', readf(filepath)] | ||||
| return subprocess.Popen(cmd, stdout=subprocess.PIPE, stdin=subprocess.PIPE, bufsize=0) | |||||
| return subprocess.Popen(cmd, stdout=subprocess.PIPE, stdin=subprocess.PIPE, bufsize=0, close_fds=not is_win32) | |||||
| def run_prefork_process(cmd, kwargs, cargs): | def run_prefork_process(cmd, kwargs, cargs): | ||||
| """ | """ | ||||
| Delegates process execution to a pre-forked process instance. | Delegates process execution to a pre-forked process instance. | ||||
| """ | """ | ||||
| if not 'env' in kwargs: | |||||
| if not kwargs.get('env'): | |||||
| kwargs['env'] = dict(os.environ) | kwargs['env'] = dict(os.environ) | ||||
| try: | try: | ||||
| obj = base64.b64encode(cPickle.dumps([cmd, kwargs, cargs])) | obj = base64.b64encode(cPickle.dumps([cmd, kwargs, cargs])) | ||||
| @@ -264,7 +264,7 @@ else: | |||||
| 'u': pop_cursor, | 'u': pop_cursor, | ||||
| } | } | ||||
| # Match either the escape sequence or text not containing escape sequence | # Match either the escape sequence or text not containing escape sequence | ||||
| ansi_tokens = re.compile('(?:\x1b\[([0-9?;]*)([a-zA-Z])|([^\x1b]+))') | |||||
| ansi_tokens = re.compile(r'(?:\x1b\[([0-9?;]*)([a-zA-Z])|([^\x1b]+))') | |||||
| def write(self, text): | def write(self, text): | ||||
| try: | try: | ||||
| wlock.acquire() | wlock.acquire() | ||||
| @@ -0,0 +1,92 @@ | |||||
| #!/usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Krzysztof Kosiński 2014 | |||||
| # DragoonX6 2018 | |||||
| """ | |||||
| Detect the Clang C compiler | |||||
| This version is an attempt at supporting the -target and -sysroot flag of Clang. | |||||
| """ | |||||
| from waflib.Tools import ccroot, ar, gcc | |||||
| from waflib.Configure import conf | |||||
| import waflib.Context | |||||
| import waflib.extras.clang_cross_common | |||||
| def options(opt): | |||||
| """ | |||||
| Target triplet for clang:: | |||||
| $ waf configure --clang-target-triple=x86_64-pc-linux-gnu | |||||
| """ | |||||
| cc_compiler_opts = opt.add_option_group('Configuration options') | |||||
| cc_compiler_opts.add_option('--clang-target-triple', default=None, | |||||
| help='Target triple for clang', | |||||
| dest='clang_target_triple') | |||||
| cc_compiler_opts.add_option('--clang-sysroot', default=None, | |||||
| help='Sysroot for clang', | |||||
| dest='clang_sysroot') | |||||
| @conf | |||||
| def find_clang(conf): | |||||
| """ | |||||
| Finds the program clang and executes it to ensure it really is clang | |||||
| """ | |||||
| import os | |||||
| cc = conf.find_program('clang', var='CC') | |||||
| if conf.options.clang_target_triple != None: | |||||
| conf.env.append_value('CC', ['-target', conf.options.clang_target_triple]) | |||||
| if conf.options.clang_sysroot != None: | |||||
| sysroot = str() | |||||
| if os.path.isabs(conf.options.clang_sysroot): | |||||
| sysroot = conf.options.clang_sysroot | |||||
| else: | |||||
| sysroot = os.path.normpath(os.path.join(os.getcwd(), conf.options.clang_sysroot)) | |||||
| conf.env.append_value('CC', ['--sysroot', sysroot]) | |||||
| conf.get_cc_version(cc, clang=True) | |||||
| conf.env.CC_NAME = 'clang' | |||||
| @conf | |||||
| def clang_modifier_x86_64_w64_mingw32(conf): | |||||
| conf.gcc_modifier_win32() | |||||
| @conf | |||||
| def clang_modifier_i386_w64_mingw32(conf): | |||||
| conf.gcc_modifier_win32() | |||||
| @conf | |||||
| def clang_modifier_x86_64_windows_msvc(conf): | |||||
| conf.clang_modifier_msvc() | |||||
| # Allow the user to override any flags if they so desire. | |||||
| clang_modifier_user_func = getattr(conf, 'clang_modifier_x86_64_windows_msvc_user', None) | |||||
| if clang_modifier_user_func: | |||||
| clang_modifier_user_func() | |||||
| @conf | |||||
| def clang_modifier_i386_windows_msvc(conf): | |||||
| conf.clang_modifier_msvc() | |||||
| # Allow the user to override any flags if they so desire. | |||||
| clang_modifier_user_func = getattr(conf, 'clang_modifier_i386_windows_msvc_user', None) | |||||
| if clang_modifier_user_func: | |||||
| clang_modifier_user_func() | |||||
| def configure(conf): | |||||
| conf.find_clang() | |||||
| conf.find_program(['llvm-ar', 'ar'], var='AR') | |||||
| conf.find_ar() | |||||
| conf.gcc_common_flags() | |||||
| # Allow the user to provide flags for the target platform. | |||||
| conf.gcc_modifier_platform() | |||||
| # And allow more fine grained control based on the compiler's triplet. | |||||
| conf.clang_modifier_target_triple() | |||||
| conf.cc_load_tools() | |||||
| conf.cc_add_flags() | |||||
| conf.link_add_flags() | |||||
| @@ -0,0 +1,113 @@ | |||||
| #!/usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # DragoonX6 2018 | |||||
| """ | |||||
| Common routines for cross_clang.py and cross_clangxx.py | |||||
| """ | |||||
| from waflib.Configure import conf | |||||
| import waflib.Context | |||||
| def normalize_target_triple(target_triple): | |||||
| target_triple = target_triple[:-1] | |||||
| normalized_triple = target_triple.replace('--', '-unknown-') | |||||
| if normalized_triple.startswith('-'): | |||||
| normalized_triple = 'unknown' + normalized_triple | |||||
| if normalized_triple.endswith('-'): | |||||
| normalized_triple += 'unknown' | |||||
| # Normalize MinGW builds to *arch*-w64-mingw32 | |||||
| if normalized_triple.endswith('windows-gnu'): | |||||
| normalized_triple = normalized_triple[:normalized_triple.index('-')] + '-w64-mingw32' | |||||
| # Strip the vendor when doing msvc builds, since it's unused anyway. | |||||
| if normalized_triple.endswith('windows-msvc'): | |||||
| normalized_triple = normalized_triple[:normalized_triple.index('-')] + '-windows-msvc' | |||||
| return normalized_triple.replace('-', '_') | |||||
| @conf | |||||
| def clang_modifier_msvc(conf): | |||||
| import os | |||||
| """ | |||||
| Really basic setup to use clang in msvc mode. | |||||
| We actually don't really want to do a lot, even though clang is msvc compatible | |||||
| in this mode, that doesn't mean we're actually using msvc. | |||||
| It's probably the best to leave it to the user, we can assume msvc mode if the user | |||||
| uses the clang-cl frontend, but this module only concerns itself with the gcc-like frontend. | |||||
| """ | |||||
| v = conf.env | |||||
| v.cprogram_PATTERN = '%s.exe' | |||||
| v.cshlib_PATTERN = '%s.dll' | |||||
| v.implib_PATTERN = '%s.lib' | |||||
| v.IMPLIB_ST = '-Wl,-IMPLIB:%s' | |||||
| v.SHLIB_MARKER = [] | |||||
| v.CFLAGS_cshlib = [] | |||||
| v.LINKFLAGS_cshlib = ['-Wl,-DLL'] | |||||
| v.cstlib_PATTERN = '%s.lib' | |||||
| v.STLIB_MARKER = [] | |||||
| del(v.AR) | |||||
| conf.find_program(['llvm-lib', 'lib'], var='AR') | |||||
| v.ARFLAGS = ['-nologo'] | |||||
| v.AR_TGT_F = ['-out:'] | |||||
| # Default to the linker supplied with llvm instead of link.exe or ld | |||||
| v.LINK_CC = v.CC + ['-fuse-ld=lld', '-nostdlib'] | |||||
| v.CCLNK_TGT_F = ['-o'] | |||||
| v.def_PATTERN = '-Wl,-def:%s' | |||||
| v.LINKFLAGS = [] | |||||
| v.LIB_ST = '-l%s' | |||||
| v.LIBPATH_ST = '-Wl,-LIBPATH:%s' | |||||
| v.STLIB_ST = '-l%s' | |||||
| v.STLIBPATH_ST = '-Wl,-LIBPATH:%s' | |||||
| CFLAGS_CRT_COMMON = [ | |||||
| '-Xclang', '--dependent-lib=oldnames', | |||||
| '-Xclang', '-fno-rtti-data', | |||||
| '-D_MT' | |||||
| ] | |||||
| v.CFLAGS_CRT_MULTITHREADED = CFLAGS_CRT_COMMON + [ | |||||
| '-Xclang', '-flto-visibility-public-std', | |||||
| '-Xclang', '--dependent-lib=libcmt', | |||||
| ] | |||||
| v.CXXFLAGS_CRT_MULTITHREADED = v.CFLAGS_CRT_MULTITHREADED | |||||
| v.CFLAGS_CRT_MULTITHREADED_DBG = CFLAGS_CRT_COMMON + [ | |||||
| '-D_DEBUG', | |||||
| '-Xclang', '-flto-visibility-public-std', | |||||
| '-Xclang', '--dependent-lib=libcmtd', | |||||
| ] | |||||
| v.CXXFLAGS_CRT_MULTITHREADED_DBG = v.CFLAGS_CRT_MULTITHREADED_DBG | |||||
| v.CFLAGS_CRT_MULTITHREADED_DLL = CFLAGS_CRT_COMMON + [ | |||||
| '-D_DLL', | |||||
| '-Xclang', '--dependent-lib=msvcrt' | |||||
| ] | |||||
| v.CXXFLAGS_CRT_MULTITHREADED_DLL = v.CFLAGS_CRT_MULTITHREADED_DLL | |||||
| v.CFLAGS_CRT_MULTITHREADED_DLL_DBG = CFLAGS_CRT_COMMON + [ | |||||
| '-D_DLL', | |||||
| '-D_DEBUG', | |||||
| '-Xclang', '--dependent-lib=msvcrtd', | |||||
| ] | |||||
| v.CXXFLAGS_CRT_MULTITHREADED_DLL_DBG = v.CFLAGS_CRT_MULTITHREADED_DLL_DBG | |||||
| @conf | |||||
| def clang_modifier_target_triple(conf, cpp=False): | |||||
| compiler = conf.env.CXX if cpp else conf.env.CC | |||||
| output = conf.cmd_and_log(compiler + ['-dumpmachine'], output=waflib.Context.STDOUT) | |||||
| modifier = ('clangxx' if cpp else 'clang') + '_modifier_' | |||||
| clang_modifier_func = getattr(conf, modifier + normalize_target_triple(output), None) | |||||
| if clang_modifier_func: | |||||
| clang_modifier_func() | |||||
| @@ -0,0 +1,106 @@ | |||||
| #!/usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Thomas Nagy 2009-2018 (ita) | |||||
| # DragoonX6 2018 | |||||
| """ | |||||
| Detect the Clang++ C++ compiler | |||||
| This version is an attempt at supporting the -target and -sysroot flag of Clang++. | |||||
| """ | |||||
| from waflib.Tools import ccroot, ar, gxx | |||||
| from waflib.Configure import conf | |||||
| import waflib.extras.clang_cross_common | |||||
| def options(opt): | |||||
| """ | |||||
| Target triplet for clang++:: | |||||
| $ waf configure --clangxx-target-triple=x86_64-pc-linux-gnu | |||||
| """ | |||||
| cxx_compiler_opts = opt.add_option_group('Configuration options') | |||||
| cxx_compiler_opts.add_option('--clangxx-target-triple', default=None, | |||||
| help='Target triple for clang++', | |||||
| dest='clangxx_target_triple') | |||||
| cxx_compiler_opts.add_option('--clangxx-sysroot', default=None, | |||||
| help='Sysroot for clang++', | |||||
| dest='clangxx_sysroot') | |||||
| @conf | |||||
| def find_clangxx(conf): | |||||
| """ | |||||
| Finds the program clang++, and executes it to ensure it really is clang++ | |||||
| """ | |||||
| import os | |||||
| cxx = conf.find_program('clang++', var='CXX') | |||||
| if conf.options.clangxx_target_triple != None: | |||||
| conf.env.append_value('CXX', ['-target', conf.options.clangxx_target_triple]) | |||||
| if conf.options.clangxx_sysroot != None: | |||||
| sysroot = str() | |||||
| if os.path.isabs(conf.options.clangxx_sysroot): | |||||
| sysroot = conf.options.clangxx_sysroot | |||||
| else: | |||||
| sysroot = os.path.normpath(os.path.join(os.getcwd(), conf.options.clangxx_sysroot)) | |||||
| conf.env.append_value('CXX', ['--sysroot', sysroot]) | |||||
| conf.get_cc_version(cxx, clang=True) | |||||
| conf.env.CXX_NAME = 'clang' | |||||
| @conf | |||||
| def clangxx_modifier_x86_64_w64_mingw32(conf): | |||||
| conf.gcc_modifier_win32() | |||||
| @conf | |||||
| def clangxx_modifier_i386_w64_mingw32(conf): | |||||
| conf.gcc_modifier_win32() | |||||
| @conf | |||||
| def clangxx_modifier_msvc(conf): | |||||
| v = conf.env | |||||
| v.cxxprogram_PATTERN = v.cprogram_PATTERN | |||||
| v.cxxshlib_PATTERN = v.cshlib_PATTERN | |||||
| v.CXXFLAGS_cxxshlib = [] | |||||
| v.LINKFLAGS_cxxshlib = v.LINKFLAGS_cshlib | |||||
| v.cxxstlib_PATTERN = v.cstlib_PATTERN | |||||
| v.LINK_CXX = v.CXX + ['-fuse-ld=lld', '-nostdlib'] | |||||
| v.CXXLNK_TGT_F = v.CCLNK_TGT_F | |||||
| @conf | |||||
| def clangxx_modifier_x86_64_windows_msvc(conf): | |||||
| conf.clang_modifier_msvc() | |||||
| conf.clangxx_modifier_msvc() | |||||
| # Allow the user to override any flags if they so desire. | |||||
| clang_modifier_user_func = getattr(conf, 'clangxx_modifier_x86_64_windows_msvc_user', None) | |||||
| if clang_modifier_user_func: | |||||
| clang_modifier_user_func() | |||||
| @conf | |||||
| def clangxx_modifier_i386_windows_msvc(conf): | |||||
| conf.clang_modifier_msvc() | |||||
| conf.clangxx_modifier_msvc() | |||||
| # Allow the user to override any flags if they so desire. | |||||
| clang_modifier_user_func = getattr(conf, 'clangxx_modifier_i386_windows_msvc_user', None) | |||||
| if clang_modifier_user_func: | |||||
| clang_modifier_user_func() | |||||
| def configure(conf): | |||||
| conf.find_clangxx() | |||||
| conf.find_program(['llvm-ar', 'ar'], var='AR') | |||||
| conf.find_ar() | |||||
| conf.gxx_common_flags() | |||||
| # Allow the user to provide flags for the target platform. | |||||
| conf.gxx_modifier_platform() | |||||
| # And allow more fine grained control based on the compiler's triplet. | |||||
| conf.clang_modifier_target_triple(cpp=True) | |||||
| conf.cxx_load_tools() | |||||
| conf.cxx_add_flags() | |||||
| conf.link_add_flags() | |||||
| @@ -0,0 +1,68 @@ | |||||
| #!/usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Thomas Nagy, 2021 (ita) | |||||
| from waflib import Utils, Runner | |||||
| """ | |||||
| Re-enable the classic threading system from waf 1.x | |||||
| def configure(conf): | |||||
| conf.load('classic_runner') | |||||
| """ | |||||
| class TaskConsumer(Utils.threading.Thread): | |||||
| """ | |||||
| Task consumers belong to a pool of workers | |||||
| They wait for tasks in the queue and then use ``task.process(...)`` | |||||
| """ | |||||
| def __init__(self, spawner): | |||||
| Utils.threading.Thread.__init__(self) | |||||
| """ | |||||
| Obtain :py:class:`waflib.Task.TaskBase` instances from this queue. | |||||
| """ | |||||
| self.spawner = spawner | |||||
| self.daemon = True | |||||
| self.start() | |||||
| def run(self): | |||||
| """ | |||||
| Loop over the tasks to execute | |||||
| """ | |||||
| try: | |||||
| self.loop() | |||||
| except Exception: | |||||
| pass | |||||
| def loop(self): | |||||
| """ | |||||
| Obtain tasks from :py:attr:`waflib.Runner.TaskConsumer.ready` and call | |||||
| :py:meth:`waflib.Task.TaskBase.process`. If the object is a function, execute it. | |||||
| """ | |||||
| master = self.spawner.master | |||||
| while 1: | |||||
| if not master.stop: | |||||
| try: | |||||
| tsk = master.ready.get() | |||||
| if tsk: | |||||
| tsk.log_display(tsk.generator.bld) | |||||
| master.process_task(tsk) | |||||
| else: | |||||
| break | |||||
| finally: | |||||
| master.out.put(tsk) | |||||
| class Spawner(object): | |||||
| """ | |||||
| Daemon thread that consumes tasks from :py:class:`waflib.Runner.Parallel` producer and | |||||
| spawns a consuming thread :py:class:`waflib.Runner.Consumer` for each | |||||
| :py:class:`waflib.Task.Task` instance. | |||||
| """ | |||||
| def __init__(self, master): | |||||
| self.master = master | |||||
| """:py:class:`waflib.Runner.Parallel` producer instance""" | |||||
| self.pool = [TaskConsumer(self) for i in range(master.numjobs)] | |||||
| Runner.Spawner = Spawner | |||||
| @@ -0,0 +1,59 @@ | |||||
| #!/usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Replaces the default formatter by one which understands MSVC output and colorizes it. | |||||
| # Modified from color_gcc.py | |||||
| __author__ = __maintainer__ = "Alibek Omarov <a1ba.omarov@gmail.com>" | |||||
| __copyright__ = "Alibek Omarov, 2019" | |||||
| import sys | |||||
| from waflib import Logs | |||||
| class ColorMSVCFormatter(Logs.formatter): | |||||
| def __init__(self, colors): | |||||
| self.colors = colors | |||||
| Logs.formatter.__init__(self) | |||||
| def parseMessage(self, line, color): | |||||
| # Split messaage from 'disk:filepath: type: message' | |||||
| arr = line.split(':', 3) | |||||
| if len(arr) < 4: | |||||
| return line | |||||
| colored = self.colors.BOLD + arr[0] + ':' + arr[1] + ':' + self.colors.NORMAL | |||||
| colored += color + arr[2] + ':' + self.colors.NORMAL | |||||
| colored += arr[3] | |||||
| return colored | |||||
| def format(self, rec): | |||||
| frame = sys._getframe() | |||||
| while frame: | |||||
| func = frame.f_code.co_name | |||||
| if func == 'exec_command': | |||||
| cmd = frame.f_locals.get('cmd') | |||||
| if isinstance(cmd, list): | |||||
| # Fix file case, it may be CL.EXE or cl.exe | |||||
| argv0 = cmd[0].lower() | |||||
| if 'cl.exe' in argv0: | |||||
| lines = [] | |||||
| # This will not work with "localized" versions | |||||
| # of MSVC | |||||
| for line in rec.msg.splitlines(): | |||||
| if ': warning ' in line: | |||||
| lines.append(self.parseMessage(line, self.colors.YELLOW)) | |||||
| elif ': error ' in line: | |||||
| lines.append(self.parseMessage(line, self.colors.RED)) | |||||
| elif ': fatal error ' in line: | |||||
| lines.append(self.parseMessage(line, self.colors.RED + self.colors.BOLD)) | |||||
| elif ': note: ' in line: | |||||
| lines.append(self.parseMessage(line, self.colors.CYAN)) | |||||
| else: | |||||
| lines.append(line) | |||||
| rec.msg = "\n".join(lines) | |||||
| frame = frame.f_back | |||||
| return Logs.formatter.format(self, rec) | |||||
| def options(opt): | |||||
| Logs.log.handlers[0].setFormatter(ColorMSVCFormatter(Logs.colors)) | |||||
| @@ -0,0 +1,52 @@ | |||||
| #! /usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Detection of the Fujitsu Fortran compiler for ARM64FX | |||||
| import re | |||||
| from waflib.Tools import fc,fc_config,fc_scan | |||||
| from waflib.Configure import conf | |||||
| from waflib.Tools.compiler_fc import fc_compiler | |||||
| fc_compiler['linux'].append('fc_fujitsu') | |||||
| @conf | |||||
| def find_fujitsu(conf): | |||||
| fc=conf.find_program(['frtpx'],var='FC') | |||||
| conf.get_fujitsu_version(fc) | |||||
| conf.env.FC_NAME='FUJITSU' | |||||
| conf.env.FC_MOD_CAPITALIZATION='lower' | |||||
| @conf | |||||
| def fujitsu_flags(conf): | |||||
| v=conf.env | |||||
| v['_FCMODOUTFLAGS']=[] | |||||
| v['FCFLAGS_DEBUG']=[] | |||||
| v['FCFLAGS_fcshlib']=[] | |||||
| v['LINKFLAGS_fcshlib']=[] | |||||
| v['FCSTLIB_MARKER']='' | |||||
| v['FCSHLIB_MARKER']='' | |||||
| @conf | |||||
| def get_fujitsu_version(conf,fc): | |||||
| version_re=re.compile(r"frtpx\s*\(FRT\)\s*(?P<major>\d+)\.(?P<minor>\d+)\.",re.I).search | |||||
| cmd=fc+['--version'] | |||||
| out,err=fc_config.getoutput(conf,cmd,stdin=False) | |||||
| if out: | |||||
| match=version_re(out) | |||||
| else: | |||||
| match=version_re(err) | |||||
| if not match: | |||||
| return(False) | |||||
| conf.fatal('Could not determine the Fujitsu FRT Fortran compiler version.') | |||||
| else: | |||||
| k=match.groupdict() | |||||
| conf.env['FC_VERSION']=(k['major'],k['minor']) | |||||
| def configure(conf): | |||||
| conf.find_fujitsu() | |||||
| conf.find_program('ar',var='AR') | |||||
| conf.add_os_flags('ARFLAGS') | |||||
| if not conf.env.ARFLAGS: | |||||
| conf.env.ARFLAGS=['rcs'] | |||||
| conf.fc_flags() | |||||
| conf.fc_add_flags() | |||||
| conf.fujitsu_flags() | |||||
| @@ -0,0 +1,52 @@ | |||||
| #! /usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Detection of the NEC Fortran compiler for Aurora Tsubasa | |||||
| import re | |||||
| from waflib.Tools import fc,fc_config,fc_scan | |||||
| from waflib.Configure import conf | |||||
| from waflib.Tools.compiler_fc import fc_compiler | |||||
| fc_compiler['linux'].append('fc_nfort') | |||||
| @conf | |||||
| def find_nfort(conf): | |||||
| fc=conf.find_program(['nfort'],var='FC') | |||||
| conf.get_nfort_version(fc) | |||||
| conf.env.FC_NAME='NFORT' | |||||
| conf.env.FC_MOD_CAPITALIZATION='lower' | |||||
| @conf | |||||
| def nfort_flags(conf): | |||||
| v=conf.env | |||||
| v['_FCMODOUTFLAGS']=[] | |||||
| v['FCFLAGS_DEBUG']=[] | |||||
| v['FCFLAGS_fcshlib']=[] | |||||
| v['LINKFLAGS_fcshlib']=[] | |||||
| v['FCSTLIB_MARKER']='' | |||||
| v['FCSHLIB_MARKER']='' | |||||
| @conf | |||||
| def get_nfort_version(conf,fc): | |||||
| version_re=re.compile(r"nfort\s*\(NFORT\)\s*(?P<major>\d+)\.(?P<minor>\d+)\.",re.I).search | |||||
| cmd=fc+['--version'] | |||||
| out,err=fc_config.getoutput(conf,cmd,stdin=False) | |||||
| if out: | |||||
| match=version_re(out) | |||||
| else: | |||||
| match=version_re(err) | |||||
| if not match: | |||||
| return(False) | |||||
| conf.fatal('Could not determine the NEC NFORT Fortran compiler version.') | |||||
| else: | |||||
| k=match.groupdict() | |||||
| conf.env['FC_VERSION']=(k['major'],k['minor']) | |||||
| def configure(conf): | |||||
| conf.find_nfort() | |||||
| conf.find_program('nar',var='AR') | |||||
| conf.add_os_flags('ARFLAGS') | |||||
| if not conf.env.ARFLAGS: | |||||
| conf.env.ARFLAGS=['rcs'] | |||||
| conf.fc_flags() | |||||
| conf.fc_add_flags() | |||||
| conf.nfort_flags() | |||||
| @@ -0,0 +1,194 @@ | |||||
| import os | |||||
| import pipes | |||||
| import subprocess | |||||
| import sys | |||||
| from waflib import Logs, Task, Context | |||||
| from waflib.Tools.c_preproc import scan as scan_impl | |||||
| # ^-- Note: waflib.extras.gccdeps.scan does not work for us, | |||||
| # due to its current implementation: | |||||
| # The -MD flag is injected into the {C,CXX}FLAGS environment variable and | |||||
| # dependencies are read out in a separate step after compiling by reading | |||||
| # the .d file saved alongside the object file. | |||||
| # As the genpybind task refers to a header file that is never compiled itself, | |||||
| # gccdeps will not be able to extract the list of dependencies. | |||||
| from waflib.TaskGen import feature, before_method | |||||
| def join_args(args): | |||||
| return " ".join(pipes.quote(arg) for arg in args) | |||||
| def configure(cfg): | |||||
| cfg.load("compiler_cxx") | |||||
| cfg.load("python") | |||||
| cfg.check_python_version(minver=(2, 7)) | |||||
| if not cfg.env.LLVM_CONFIG: | |||||
| cfg.find_program("llvm-config", var="LLVM_CONFIG") | |||||
| if not cfg.env.GENPYBIND: | |||||
| cfg.find_program("genpybind", var="GENPYBIND") | |||||
| # find clang reasource dir for builtin headers | |||||
| cfg.env.GENPYBIND_RESOURCE_DIR = os.path.join( | |||||
| cfg.cmd_and_log(cfg.env.LLVM_CONFIG + ["--libdir"]).strip(), | |||||
| "clang", | |||||
| cfg.cmd_and_log(cfg.env.LLVM_CONFIG + ["--version"]).strip()) | |||||
| if os.path.exists(cfg.env.GENPYBIND_RESOURCE_DIR): | |||||
| cfg.msg("Checking clang resource dir", cfg.env.GENPYBIND_RESOURCE_DIR) | |||||
| else: | |||||
| cfg.fatal("Clang resource dir not found") | |||||
| @feature("genpybind") | |||||
| @before_method("process_source") | |||||
| def generate_genpybind_source(self): | |||||
| """ | |||||
| Run genpybind on the headers provided in `source` and compile/link the | |||||
| generated code instead. This works by generating the code on the fly and | |||||
| swapping the source node before `process_source` is run. | |||||
| """ | |||||
| # name of module defaults to name of target | |||||
| module = getattr(self, "module", self.target) | |||||
| # create temporary source file in build directory to hold generated code | |||||
| out = "genpybind-%s.%d.cpp" % (module, self.idx) | |||||
| out = self.path.get_bld().find_or_declare(out) | |||||
| task = self.create_task("genpybind", self.to_nodes(self.source), out) | |||||
| # used to detect whether CFLAGS or CXXFLAGS should be passed to genpybind | |||||
| task.features = self.features | |||||
| task.module = module | |||||
| # can be used to select definitions to include in the current module | |||||
| # (when header files are shared by more than one module) | |||||
| task.genpybind_tags = self.to_list(getattr(self, "genpybind_tags", [])) | |||||
| # additional include directories | |||||
| task.includes = self.to_list(getattr(self, "includes", [])) | |||||
| task.genpybind = self.env.GENPYBIND | |||||
| # Tell waf to compile/link the generated code instead of the headers | |||||
| # originally passed-in via the `source` parameter. (see `process_source`) | |||||
| self.source = [out] | |||||
| class genpybind(Task.Task): # pylint: disable=invalid-name | |||||
| """ | |||||
| Runs genpybind on headers provided as input to this task. | |||||
| Generated code will be written to the first (and only) output node. | |||||
| """ | |||||
| quiet = True | |||||
| color = "PINK" | |||||
| scan = scan_impl | |||||
| @staticmethod | |||||
| def keyword(): | |||||
| return "Analyzing" | |||||
| def run(self): | |||||
| if not self.inputs: | |||||
| return | |||||
| args = self.find_genpybind() + self._arguments( | |||||
| resource_dir=self.env.GENPYBIND_RESOURCE_DIR) | |||||
| output = self.run_genpybind(args) | |||||
| # For debugging / log output | |||||
| pasteable_command = join_args(args) | |||||
| # write generated code to file in build directory | |||||
| # (will be compiled during process_source stage) | |||||
| (output_node,) = self.outputs | |||||
| output_node.write("// {}\n{}\n".format( | |||||
| pasteable_command.replace("\n", "\n// "), output)) | |||||
| def find_genpybind(self): | |||||
| return self.genpybind | |||||
| def run_genpybind(self, args): | |||||
| bld = self.generator.bld | |||||
| kwargs = dict(cwd=bld.variant_dir) | |||||
| if hasattr(bld, "log_command"): | |||||
| bld.log_command(args, kwargs) | |||||
| else: | |||||
| Logs.debug("runner: {!r}".format(args)) | |||||
| proc = subprocess.Popen( | |||||
| args, stdout=subprocess.PIPE, stderr=subprocess.PIPE, **kwargs) | |||||
| stdout, stderr = proc.communicate() | |||||
| if not isinstance(stdout, str): | |||||
| stdout = stdout.decode(sys.stdout.encoding, errors="replace") | |||||
| if not isinstance(stderr, str): | |||||
| stderr = stderr.decode(sys.stderr.encoding, errors="replace") | |||||
| if proc.returncode != 0: | |||||
| bld.fatal( | |||||
| "genpybind returned {code} during the following call:" | |||||
| "\n{command}\n\n{stdout}\n\n{stderr}".format( | |||||
| code=proc.returncode, | |||||
| command=join_args(args), | |||||
| stdout=stdout, | |||||
| stderr=stderr, | |||||
| )) | |||||
| if stderr.strip(): | |||||
| Logs.debug("non-fatal warnings during genpybind run:\n{}".format(stderr)) | |||||
| return stdout | |||||
| def _include_paths(self): | |||||
| return self.generator.to_incnodes(self.includes + self.env.INCLUDES) | |||||
| def _inputs_as_relative_includes(self): | |||||
| include_paths = self._include_paths() | |||||
| relative_includes = [] | |||||
| for node in self.inputs: | |||||
| for inc in include_paths: | |||||
| if node.is_child_of(inc): | |||||
| relative_includes.append(node.path_from(inc)) | |||||
| break | |||||
| else: | |||||
| self.generator.bld.fatal("could not resolve {}".format(node)) | |||||
| return relative_includes | |||||
| def _arguments(self, genpybind_parse=None, resource_dir=None): | |||||
| args = [] | |||||
| relative_includes = self._inputs_as_relative_includes() | |||||
| is_cxx = "cxx" in self.features | |||||
| # options for genpybind | |||||
| args.extend(["--genpybind-module", self.module]) | |||||
| if self.genpybind_tags: | |||||
| args.extend(["--genpybind-tag"] + self.genpybind_tags) | |||||
| if relative_includes: | |||||
| args.extend(["--genpybind-include"] + relative_includes) | |||||
| if genpybind_parse: | |||||
| args.extend(["--genpybind-parse", genpybind_parse]) | |||||
| args.append("--") | |||||
| # headers to be processed by genpybind | |||||
| args.extend(node.abspath() for node in self.inputs) | |||||
| args.append("--") | |||||
| # options for clang/genpybind-parse | |||||
| args.append("-D__GENPYBIND__") | |||||
| args.append("-xc++" if is_cxx else "-xc") | |||||
| has_std_argument = False | |||||
| for flag in self.env["CXXFLAGS" if is_cxx else "CFLAGS"]: | |||||
| flag = flag.replace("-std=gnu", "-std=c") | |||||
| if flag.startswith("-std=c"): | |||||
| has_std_argument = True | |||||
| args.append(flag) | |||||
| if not has_std_argument: | |||||
| args.append("-std=c++14") | |||||
| args.extend("-I{}".format(n.abspath()) for n in self._include_paths()) | |||||
| args.extend("-D{}".format(p) for p in self.env.DEFINES) | |||||
| # point to clang resource dir, if specified | |||||
| if resource_dir: | |||||
| args.append("-resource-dir={}".format(resource_dir)) | |||||
| return args | |||||
| @@ -0,0 +1,154 @@ | |||||
| import re | |||||
| from waflib import Utils, Task, Errors, Logs | |||||
| from waflib.Configure import conf | |||||
| from waflib.TaskGen import extension, taskgen_method | |||||
| HAXE_COMPILERS = { | |||||
| 'JS': {'tgt': '--js', 'ext_out': ['.js']}, | |||||
| 'LUA': {'tgt': '--lua', 'ext_out': ['.lua']}, | |||||
| 'SWF': {'tgt': '--swf', 'ext_out': ['.swf']}, | |||||
| 'NEKO': {'tgt': '--neko', 'ext_out': ['.n']}, | |||||
| 'PHP': {'tgt': '--php', 'ext_out': ['.php']}, | |||||
| 'CPP': {'tgt': '--cpp', 'ext_out': ['.h', '.cpp']}, | |||||
| 'CPPIA': {'tgt': '--cppia', 'ext_out': ['.cppia']}, | |||||
| 'CS': {'tgt': '--cs', 'ext_out': ['.cs']}, | |||||
| 'JAVA': {'tgt': '--java', 'ext_out': ['.java']}, | |||||
| 'JVM': {'tgt': '--jvm', 'ext_out': ['.jar']}, | |||||
| 'PYTHON': {'tgt': '--python', 'ext_out': ['.py']}, | |||||
| 'HL': {'tgt': '--hl', 'ext_out': ['.hl']}, | |||||
| 'HLC': {'tgt': '--hl', 'ext_out': ['.h', '.c']}, | |||||
| } | |||||
| @conf | |||||
| def check_haxe_pkg(self, **kw): | |||||
| self.find_program('haxelib') | |||||
| libs = kw.get('libs') | |||||
| if not libs or not (type(libs) == str or (type(libs) == list and all(isinstance(s, str) for s in libs))): | |||||
| self.fatal('Specify correct libs value in ensure call') | |||||
| return | |||||
| fetch = kw.get('fetch') | |||||
| if not fetch is None and not type(fetch) == bool: | |||||
| self.fatal('Specify correct fetch value in ensure call') | |||||
| libs = [libs] if type(libs) == str else libs | |||||
| halt = False | |||||
| for lib in libs: | |||||
| try: | |||||
| self.start_msg('Checking for library %s' % lib) | |||||
| output = self.cmd_and_log(self.env.HAXELIB + ['list', lib]) | |||||
| except Errors.WafError: | |||||
| self.end_msg(False) | |||||
| self.fatal('Can\'t run haxelib list, ensuring halted') | |||||
| return | |||||
| if lib in output: | |||||
| self.end_msg(lib in output) | |||||
| else: | |||||
| if not fetch: | |||||
| self.end_msg(False) | |||||
| halt = True | |||||
| continue | |||||
| try: | |||||
| status = self.exec_command(self.env.HAXELIB + ['install', lib]) | |||||
| if status: | |||||
| self.end_msg(False) | |||||
| self.fatal('Can\'t get %s with haxelib, ensuring halted' % lib) | |||||
| return | |||||
| else: | |||||
| self.end_msg('downloaded', color='YELLOW') | |||||
| except Errors.WafError: | |||||
| self.end_msg(False) | |||||
| self.fatal('Can\'t run haxelib install, ensuring halted') | |||||
| return | |||||
| postfix = kw.get('uselib_store') or lib.upper() | |||||
| self.env.append_unique('LIB_' + postfix, lib) | |||||
| if halt: | |||||
| self.fatal('Can\'t find libraries in haxelib list, ensuring halted') | |||||
| return | |||||
| class haxe(Task.Task): | |||||
| vars = ['HAXE_VERSION', 'HAXE_FLAGS'] | |||||
| ext_in = ['.hx'] | |||||
| def run(self): | |||||
| cmd = self.env.HAXE + self.env.HAXE_FLAGS_DEFAULT + self.env.HAXE_FLAGS | |||||
| return self.exec_command(cmd) | |||||
| for COMP in HAXE_COMPILERS: | |||||
| # create runners for each compile target | |||||
| type("haxe_" + COMP, (haxe,), {'ext_out': HAXE_COMPILERS[COMP]['ext_out']}) | |||||
| @taskgen_method | |||||
| def init_haxe(self): | |||||
| errmsg = '%s not found, specify correct value' | |||||
| try: | |||||
| compiler = HAXE_COMPILERS[self.compiler] | |||||
| comp_tgt = compiler['tgt'] | |||||
| comp_mod = '/main.c' if self.compiler == 'HLC' else '' | |||||
| except (AttributeError, KeyError): | |||||
| self.bld.fatal(errmsg % 'COMPILER' + ': ' + ', '.join(HAXE_COMPILERS.keys())) | |||||
| return | |||||
| self.env.append_value( | |||||
| 'HAXE_FLAGS', | |||||
| [comp_tgt, self.path.get_bld().make_node(self.target + comp_mod).abspath()]) | |||||
| if hasattr(self, 'use'): | |||||
| if not (type(self.use) == str or type(self.use) == list): | |||||
| self.bld.fatal(errmsg % 'USE') | |||||
| return | |||||
| self.use = [self.use] if type(self.use) == str else self.use | |||||
| for dep in self.use: | |||||
| if self.env['LIB_' + dep]: | |||||
| for lib in self.env['LIB_' + dep]: | |||||
| self.env.append_value('HAXE_FLAGS', ['-lib', lib]) | |||||
| if hasattr(self, 'res'): | |||||
| if not type(self.res) == str: | |||||
| self.bld.fatal(errmsg % 'RES') | |||||
| return | |||||
| self.env.append_value('HAXE_FLAGS', ['-D', 'resourcesPath=%s' % self.res]) | |||||
| @extension('.hx') | |||||
| def haxe_hook(self, node): | |||||
| if len(self.source) > 1: | |||||
| self.bld.fatal('Use separate task generators for multiple files') | |||||
| return | |||||
| src = node | |||||
| tgt = self.path.get_bld().find_or_declare(self.target) | |||||
| self.init_haxe() | |||||
| self.create_task('haxe_' + self.compiler, src, tgt) | |||||
| @conf | |||||
| def check_haxe(self, mini=None, maxi=None): | |||||
| self.start_msg('Checking for haxe version') | |||||
| try: | |||||
| curr = re.search( | |||||
| r'(\d+.?)+', | |||||
| self.cmd_and_log(self.env.HAXE + ['-version'])).group() | |||||
| except Errors.WafError: | |||||
| self.end_msg(False) | |||||
| self.fatal('Can\'t get haxe version') | |||||
| return | |||||
| if mini and Utils.num2ver(curr) < Utils.num2ver(mini): | |||||
| self.end_msg('wrong', color='RED') | |||||
| self.fatal('%s is too old, need >= %s' % (curr, mini)) | |||||
| return | |||||
| if maxi and Utils.num2ver(curr) > Utils.num2ver(maxi): | |||||
| self.end_msg('wrong', color='RED') | |||||
| self.fatal('%s is too new, need <= %s' % (curr, maxi)) | |||||
| return | |||||
| self.end_msg(curr, color='GREEN') | |||||
| self.env.HAXE_VERSION = curr | |||||
| def configure(self): | |||||
| self.env.append_value( | |||||
| 'HAXE_FLAGS_DEFAULT', | |||||
| ['-D', 'no-compilation', '-cp', self.path.abspath()]) | |||||
| Logs.warn('Default flags: %s' % ' '.join(self.env.HAXE_FLAGS_DEFAULT)) | |||||
| self.find_program('haxe') | |||||
| @@ -0,0 +1,46 @@ | |||||
| #!/usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Rafaël Kooi 2019 | |||||
| from waflib import TaskGen | |||||
| @TaskGen.feature('c', 'cxx', 'fc') | |||||
| @TaskGen.after_method('propagate_uselib_vars') | |||||
| def add_pdb_per_object(self): | |||||
| """For msvc/fortran, specify a unique compile pdb per object, to work | |||||
| around LNK4099. Flags are updated with a unique /Fd flag based on the | |||||
| task output name. This is separate from the link pdb. | |||||
| """ | |||||
| if not hasattr(self, 'compiled_tasks'): | |||||
| return | |||||
| link_task = getattr(self, 'link_task', None) | |||||
| for task in self.compiled_tasks: | |||||
| if task.inputs and task.inputs[0].name.lower().endswith('.rc'): | |||||
| continue | |||||
| add_pdb = False | |||||
| for flagname in ('CFLAGS', 'CXXFLAGS', 'FCFLAGS'): | |||||
| # several languages may be used at once | |||||
| for flag in task.env[flagname]: | |||||
| if flag[1:].lower() == 'zi': | |||||
| add_pdb = True | |||||
| break | |||||
| if add_pdb: | |||||
| node = task.outputs[0].change_ext('.pdb') | |||||
| pdb_flag = '/Fd:' + node.abspath() | |||||
| for flagname in ('CFLAGS', 'CXXFLAGS', 'FCFLAGS'): | |||||
| buf = [pdb_flag] | |||||
| for flag in task.env[flagname]: | |||||
| if flag[1:3] == 'Fd' or flag[1:].lower() == 'fs' or flag[1:].lower() == 'mp': | |||||
| continue | |||||
| buf.append(flag) | |||||
| task.env[flagname] = buf | |||||
| if link_task and not node in link_task.dep_nodes: | |||||
| link_task.dep_nodes.append(node) | |||||
| if not node in task.outputs: | |||||
| task.outputs.append(node) | |||||
| @@ -0,0 +1,120 @@ | |||||
| """Support for Sphinx documentation | |||||
| This is a wrapper for sphinx-build program. Please note that sphinx-build supports only | |||||
| one output format at a time, but the tool can create multiple tasks to handle more. | |||||
| The output formats can be passed via the sphinx_output_format, which is an array of | |||||
| strings. For backwards compatibility if only one output is needed, it can be passed | |||||
| as a single string. | |||||
| The default output format is html. | |||||
| Specific formats can be installed in different directories by specifying the | |||||
| install_path_<FORMAT> attribute. If not defined, the standard install_path | |||||
| will be used instead. | |||||
| Example wscript: | |||||
| def configure(cnf): | |||||
| conf.load('sphinx') | |||||
| def build(bld): | |||||
| bld( | |||||
| features='sphinx', | |||||
| sphinx_source='sources', # path to source directory | |||||
| sphinx_options='-a -v', # sphinx-build program additional options | |||||
| sphinx_output_format=['html', 'man'], # output format of sphinx documentation | |||||
| install_path_man='${DOCDIR}/man' # put man pages in a specific directory | |||||
| ) | |||||
| """ | |||||
| from waflib.Node import Node | |||||
| from waflib import Utils | |||||
| from waflib import Task | |||||
| from waflib.TaskGen import feature, after_method | |||||
| def configure(cnf): | |||||
| """Check if sphinx-build program is available and loads gnu_dirs tool.""" | |||||
| cnf.find_program('sphinx-build', var='SPHINX_BUILD', mandatory=False) | |||||
| cnf.load('gnu_dirs') | |||||
| @feature('sphinx') | |||||
| def build_sphinx(self): | |||||
| """Builds sphinx sources. | |||||
| """ | |||||
| if not self.env.SPHINX_BUILD: | |||||
| self.bld.fatal('Program SPHINX_BUILD not defined.') | |||||
| if not getattr(self, 'sphinx_source', None): | |||||
| self.bld.fatal('Attribute sphinx_source not defined.') | |||||
| if not isinstance(self.sphinx_source, Node): | |||||
| self.sphinx_source = self.path.find_node(self.sphinx_source) | |||||
| if not self.sphinx_source: | |||||
| self.bld.fatal('Can\'t find sphinx_source: %r' % self.sphinx_source) | |||||
| # In the taskgen we have the complete list of formats | |||||
| Utils.def_attrs(self, sphinx_output_format='html') | |||||
| self.sphinx_output_format = Utils.to_list(self.sphinx_output_format) | |||||
| self.env.SPHINX_OPTIONS = getattr(self, 'sphinx_options', []) | |||||
| for source_file in self.sphinx_source.ant_glob('**/*'): | |||||
| self.bld.add_manual_dependency(self.sphinx_source, source_file) | |||||
| for cfmt in self.sphinx_output_format: | |||||
| sphinx_build_task = self.create_task('SphinxBuildingTask') | |||||
| sphinx_build_task.set_inputs(self.sphinx_source) | |||||
| # In task we keep the specific format this task is generating | |||||
| sphinx_build_task.env.SPHINX_OUTPUT_FORMAT = cfmt | |||||
| # the sphinx-build results are in <build + output_format> directory | |||||
| sphinx_build_task.sphinx_output_directory = self.path.get_bld().make_node(cfmt) | |||||
| sphinx_build_task.set_outputs(sphinx_build_task.sphinx_output_directory) | |||||
| sphinx_build_task.sphinx_output_directory.mkdir() | |||||
| Utils.def_attrs(sphinx_build_task, install_path=getattr(self, 'install_path_' + cfmt, getattr(self, 'install_path', get_install_path(sphinx_build_task)))) | |||||
| def get_install_path(object): | |||||
| if object.env.SPHINX_OUTPUT_FORMAT == 'man': | |||||
| return object.env.MANDIR | |||||
| elif object.env.SPHINX_OUTPUT_FORMAT == 'info': | |||||
| return object.env.INFODIR | |||||
| else: | |||||
| return object.env.DOCDIR | |||||
| class SphinxBuildingTask(Task.Task): | |||||
| color = 'BOLD' | |||||
| run_str = '${SPHINX_BUILD} -M ${SPHINX_OUTPUT_FORMAT} ${SRC} ${TGT} -d ${TGT[0].bld_dir()}/doctrees-${SPHINX_OUTPUT_FORMAT} ${SPHINX_OPTIONS}' | |||||
| def keyword(self): | |||||
| return 'Compiling (%s)' % self.env.SPHINX_OUTPUT_FORMAT | |||||
| def runnable_status(self): | |||||
| for x in self.run_after: | |||||
| if not x.hasrun: | |||||
| return Task.ASK_LATER | |||||
| self.signature() | |||||
| ret = Task.Task.runnable_status(self) | |||||
| if ret == Task.SKIP_ME: | |||||
| # in case the files were removed | |||||
| self.add_install() | |||||
| return ret | |||||
| def post_run(self): | |||||
| self.add_install() | |||||
| return Task.Task.post_run(self) | |||||
| def add_install(self): | |||||
| nodes = self.sphinx_output_directory.ant_glob('**/*', quiet=True) | |||||
| self.outputs += nodes | |||||
| self.generator.add_install_files(install_to=self.install_path, | |||||
| install_from=nodes, | |||||
| postpone=False, | |||||
| cwd=self.sphinx_output_directory.make_node(self.env.SPHINX_OUTPUT_FORMAT), | |||||
| relative_trick=True) | |||||
| @@ -0,0 +1,648 @@ | |||||
| #! /usr/bin/env python | |||||
| # encoding: utf-8 | |||||
| # Thomas Nagy, 2019 (ita) | |||||
| """ | |||||
| Filesystem-based cache system to share and re-use build artifacts | |||||
| Cache access operations (copy to and from) are delegated to | |||||
| independent pre-forked worker subprocesses. | |||||
| The following environment variables may be set: | |||||
| * WAFCACHE: several possibilities: | |||||
| - File cache: | |||||
| absolute path of the waf cache (~/.cache/wafcache_user, | |||||
| where `user` represents the currently logged-in user) | |||||
| - URL to a cache server, for example: | |||||
| export WAFCACHE=http://localhost:8080/files/ | |||||
| in that case, GET/POST requests are made to urls of the form | |||||
| http://localhost:8080/files/000000000/0 (cache management is delegated to the server) | |||||
| - GCS, S3 or MINIO bucket | |||||
| gs://my-bucket/ (uses gsutil command line tool or WAFCACHE_CMD) | |||||
| s3://my-bucket/ (uses aws command line tool or WAFCACHE_CMD) | |||||
| minio://my-bucket/ (uses mc command line tool or WAFCACHE_CMD) | |||||
| * WAFCACHE_CMD: bucket upload/download command, for example: | |||||
| WAFCACHE_CMD="gsutil cp %{SRC} %{TGT}" | |||||
| Note that the WAFCACHE bucket value is used for the source or destination | |||||
| depending on the operation (upload or download). For example, with: | |||||
| WAFCACHE="gs://mybucket/" | |||||
| the following commands may be run: | |||||
| gsutil cp build/myprogram gs://mybucket/aa/aaaaa/1 | |||||
| gsutil cp gs://mybucket/bb/bbbbb/2 build/somefile | |||||
| * WAFCACHE_NO_PUSH: if set, disables pushing to the cache | |||||
| * WAFCACHE_VERBOSITY: if set, displays more detailed cache operations | |||||
| * WAFCACHE_STATS: if set, displays cache usage statistics on exit | |||||
| File cache specific options: | |||||
| Files are copied using hard links by default; if the cache is located | |||||
| onto another partition, the system switches to file copies instead. | |||||
| * WAFCACHE_TRIM_MAX_FOLDER: maximum amount of tasks to cache (1M) | |||||
| * WAFCACHE_EVICT_MAX_BYTES: maximum amount of cache size in bytes (10GB) | |||||
| * WAFCACHE_EVICT_INTERVAL_MINUTES: minimum time interval to try | |||||
| and trim the cache (3 minutes) | |||||
| Upload specific options: | |||||
| * WAFCACHE_ASYNC_WORKERS: define a number of workers to upload results asynchronously | |||||
| this may improve build performance with many/long file uploads | |||||
| the default is unset (synchronous uploads) | |||||
| * WAFCACHE_ASYNC_NOWAIT: do not wait for uploads to complete (default: False) | |||||
| this requires asynchonous uploads to have an effect | |||||
| Usage:: | |||||
| def build(bld): | |||||
| bld.load('wafcache') | |||||
| ... | |||||
| To troubleshoot:: | |||||
| waf clean build --zone=wafcache | |||||
| """ | |||||
| import atexit, base64, errno, fcntl, getpass, os, re, shutil, sys, time, threading, traceback, urllib3, shlex | |||||
| try: | |||||
| import subprocess32 as subprocess | |||||
| except ImportError: | |||||
| import subprocess | |||||
| base_cache = os.path.expanduser('~/.cache/') | |||||
| if not os.path.isdir(base_cache): | |||||
| base_cache = '/tmp/' | |||||
| default_wafcache_dir = os.path.join(base_cache, 'wafcache_' + getpass.getuser()) | |||||
| CACHE_DIR = os.environ.get('WAFCACHE', default_wafcache_dir) | |||||
| WAFCACHE_CMD = os.environ.get('WAFCACHE_CMD') | |||||
| TRIM_MAX_FOLDERS = int(os.environ.get('WAFCACHE_TRIM_MAX_FOLDER', 1000000)) | |||||
| EVICT_INTERVAL_MINUTES = int(os.environ.get('WAFCACHE_EVICT_INTERVAL_MINUTES', 3)) | |||||
| EVICT_MAX_BYTES = int(os.environ.get('WAFCACHE_EVICT_MAX_BYTES', 10**10)) | |||||
| WAFCACHE_NO_PUSH = 1 if os.environ.get('WAFCACHE_NO_PUSH') else 0 | |||||
| WAFCACHE_VERBOSITY = 1 if os.environ.get('WAFCACHE_VERBOSITY') else 0 | |||||
| WAFCACHE_STATS = 1 if os.environ.get('WAFCACHE_STATS') else 0 | |||||
| WAFCACHE_ASYNC_WORKERS = os.environ.get('WAFCACHE_ASYNC_WORKERS') | |||||
| WAFCACHE_ASYNC_NOWAIT = os.environ.get('WAFCACHE_ASYNC_NOWAIT') | |||||
| OK = "ok" | |||||
| re_waf_cmd = re.compile('(?P<src>%{SRC})|(?P<tgt>%{TGT})') | |||||
| try: | |||||
| import cPickle | |||||
| except ImportError: | |||||
| import pickle as cPickle | |||||
| if __name__ != '__main__': | |||||
| from waflib import Task, Logs, Utils, Build | |||||
| def can_retrieve_cache(self): | |||||
| """ | |||||
| New method for waf Task classes | |||||
| """ | |||||
| if not self.outputs: | |||||
| return False | |||||
| self.cached = False | |||||
| sig = self.signature() | |||||
| ssig = Utils.to_hex(self.uid() + sig) | |||||
| if WAFCACHE_STATS: | |||||
| self.generator.bld.cache_reqs += 1 | |||||
| files_to = [node.abspath() for node in self.outputs] | |||||
| proc = get_process() | |||||
| err = cache_command(proc, ssig, [], files_to) | |||||
| process_pool.append(proc) | |||||
| if err.startswith(OK): | |||||
| if WAFCACHE_VERBOSITY: | |||||
| Logs.pprint('CYAN', ' Fetched %r from cache' % files_to) | |||||
| else: | |||||
| Logs.debug('wafcache: fetched %r from cache', files_to) | |||||
| if WAFCACHE_STATS: | |||||
| self.generator.bld.cache_hits += 1 | |||||
| else: | |||||
| if WAFCACHE_VERBOSITY: | |||||
| Logs.pprint('YELLOW', ' No cache entry %s' % files_to) | |||||
| else: | |||||
| Logs.debug('wafcache: No cache entry %s: %s', files_to, err) | |||||
| return False | |||||
| self.cached = True | |||||
| return True | |||||
| def put_files_cache(self): | |||||
| """ | |||||
| New method for waf Task classes | |||||
| """ | |||||
| if WAFCACHE_NO_PUSH or getattr(self, 'cached', None) or not self.outputs: | |||||
| return | |||||
| files_from = [] | |||||
| for node in self.outputs: | |||||
| path = node.abspath() | |||||
| if not os.path.isfile(path): | |||||
| return | |||||
| files_from.append(path) | |||||
| bld = self.generator.bld | |||||
| old_sig = self.signature() | |||||
| for node in self.inputs: | |||||
| try: | |||||
| del node.ctx.cache_sig[node] | |||||
| except KeyError: | |||||
| pass | |||||
| delattr(self, 'cache_sig') | |||||
| sig = self.signature() | |||||
| def _async_put_files_cache(bld, ssig, files_from): | |||||
| proc = get_process() | |||||
| if WAFCACHE_ASYNC_WORKERS: | |||||
| with bld.wafcache_lock: | |||||
| if bld.wafcache_stop: | |||||
| process_pool.append(proc) | |||||
| return | |||||
| bld.wafcache_procs.add(proc) | |||||
| err = cache_command(proc, ssig, files_from, []) | |||||
| process_pool.append(proc) | |||||
| if err.startswith(OK): | |||||
| if WAFCACHE_VERBOSITY: | |||||
| Logs.pprint('CYAN', ' Successfully uploaded %s to cache' % files_from) | |||||
| else: | |||||
| Logs.debug('wafcache: Successfully uploaded %r to cache', files_from) | |||||
| if WAFCACHE_STATS: | |||||
| bld.cache_puts += 1 | |||||
| else: | |||||
| if WAFCACHE_VERBOSITY: | |||||
| Logs.pprint('RED', ' Error caching step results %s: %s' % (files_from, err)) | |||||
| else: | |||||
| Logs.debug('wafcache: Error caching results %s: %s', files_from, err) | |||||
| if old_sig == sig: | |||||
| ssig = Utils.to_hex(self.uid() + sig) | |||||
| if WAFCACHE_ASYNC_WORKERS: | |||||
| fut = bld.wafcache_executor.submit(_async_put_files_cache, bld, ssig, files_from) | |||||
| bld.wafcache_uploads.append(fut) | |||||
| else: | |||||
| _async_put_files_cache(bld, ssig, files_from) | |||||
| else: | |||||
| Logs.debug('wafcache: skipped %r upload due to late input modifications %r', self.outputs, self.inputs) | |||||
| bld.task_sigs[self.uid()] = self.cache_sig | |||||
| def hash_env_vars(self, env, vars_lst): | |||||
| """ | |||||
| Reimplement BuildContext.hash_env_vars so that the resulting hash does not depend on local paths | |||||
| """ | |||||
| if not env.table: | |||||
| env = env.parent | |||||
| if not env: | |||||
| return Utils.SIG_NIL | |||||
| idx = str(id(env)) + str(vars_lst) | |||||
| try: | |||||
| cache = self.cache_env | |||||
| except AttributeError: | |||||
| cache = self.cache_env = {} | |||||
| else: | |||||
| try: | |||||
| return self.cache_env[idx] | |||||
| except KeyError: | |||||
| pass | |||||
| v = str([env[a] for a in vars_lst]) | |||||
| v = v.replace(self.srcnode.abspath().__repr__()[:-1], '') | |||||
| m = Utils.md5() | |||||
| m.update(v.encode()) | |||||
| ret = m.digest() | |||||
| Logs.debug('envhash: %r %r', ret, v) | |||||
| cache[idx] = ret | |||||
| return ret | |||||
| def uid(self): | |||||
| """ | |||||
| Reimplement Task.uid() so that the signature does not depend on local paths | |||||
| """ | |||||
| try: | |||||
| return self.uid_ | |||||
| except AttributeError: | |||||
| m = Utils.md5() | |||||
| src = self.generator.bld.srcnode | |||||
| up = m.update | |||||
| up(self.__class__.__name__.encode()) | |||||
| for x in self.inputs + self.outputs: | |||||
| up(x.path_from(src).encode()) | |||||
| self.uid_ = m.digest() | |||||
| return self.uid_ | |||||
| def make_cached(cls): | |||||
| """ | |||||
| Enable the waf cache for a given task class | |||||
| """ | |||||
| if getattr(cls, 'nocache', None) or getattr(cls, 'has_cache', False): | |||||
| return | |||||
| full_name = "%s.%s" % (cls.__module__, cls.__name__) | |||||
| if full_name in ('waflib.Tools.ccroot.vnum', 'waflib.Build.inst'): | |||||
| return | |||||
| m1 = getattr(cls, 'run', None) | |||||
| def run(self): | |||||
| if getattr(self, 'nocache', False): | |||||
| return m1(self) | |||||
| if self.can_retrieve_cache(): | |||||
| return 0 | |||||
| return m1(self) | |||||
| cls.run = run | |||||
| m2 = getattr(cls, 'post_run', None) | |||||
| def post_run(self): | |||||
| if getattr(self, 'nocache', False): | |||||
| return m2(self) | |||||
| ret = m2(self) | |||||
| self.put_files_cache() | |||||
| return ret | |||||
| cls.post_run = post_run | |||||
| cls.has_cache = True | |||||
| process_pool = [] | |||||
| def get_process(): | |||||
| """ | |||||
| Returns a worker process that can process waf cache commands | |||||
| The worker process is assumed to be returned to the process pool when unused | |||||
| """ | |||||
| try: | |||||
| return process_pool.pop() | |||||
| except IndexError: | |||||
| filepath = os.path.dirname(os.path.abspath(__file__)) + os.sep + 'wafcache.py' | |||||
| cmd = [sys.executable, '-c', Utils.readf(filepath)] | |||||
| return subprocess.Popen(cmd, stdout=subprocess.PIPE, stdin=subprocess.PIPE, bufsize=0) | |||||
| def atexit_pool(): | |||||
| for proc in process_pool: | |||||
| proc.kill() | |||||
| atexit.register(atexit_pool) | |||||
| def build(bld): | |||||
| """ | |||||
| Called during the build process to enable file caching | |||||
| """ | |||||
| if WAFCACHE_ASYNC_WORKERS: | |||||
| try: | |||||
| num_workers = int(WAFCACHE_ASYNC_WORKERS) | |||||
| except ValueError: | |||||
| Logs.warn('Invalid WAFCACHE_ASYNC_WORKERS specified: %r' % WAFCACHE_ASYNC_WORKERS) | |||||
| else: | |||||
| from concurrent.futures import ThreadPoolExecutor | |||||
| bld.wafcache_executor = ThreadPoolExecutor(max_workers=num_workers) | |||||
| bld.wafcache_uploads = [] | |||||
| bld.wafcache_procs = set([]) | |||||
| bld.wafcache_stop = False | |||||
| bld.wafcache_lock = threading.Lock() | |||||
| def finalize_upload_async(bld): | |||||
| if WAFCACHE_ASYNC_NOWAIT: | |||||
| with bld.wafcache_lock: | |||||
| bld.wafcache_stop = True | |||||
| for fut in reversed(bld.wafcache_uploads): | |||||
| fut.cancel() | |||||
| for proc in bld.wafcache_procs: | |||||
| proc.kill() | |||||
| bld.wafcache_procs.clear() | |||||
| else: | |||||
| Logs.pprint('CYAN', '... waiting for wafcache uploads to complete (%s uploads)' % len(bld.wafcache_uploads)) | |||||
| bld.wafcache_executor.shutdown(wait=True) | |||||
| bld.add_post_fun(finalize_upload_async) | |||||
| if WAFCACHE_STATS: | |||||
| # Init counter for statistics and hook to print results at the end | |||||
| bld.cache_reqs = bld.cache_hits = bld.cache_puts = 0 | |||||
| def printstats(bld): | |||||
| hit_ratio = 0 | |||||
| if bld.cache_reqs > 0: | |||||
| hit_ratio = (bld.cache_hits / bld.cache_reqs) * 100 | |||||
| Logs.pprint('CYAN', ' wafcache stats: %s requests, %s hits (ratio: %.2f%%), %s writes' % | |||||
| (bld.cache_reqs, bld.cache_hits, hit_ratio, bld.cache_puts) ) | |||||
| bld.add_post_fun(printstats) | |||||
| if process_pool: | |||||
| # already called once | |||||
| return | |||||
| # pre-allocation | |||||
| processes = [get_process() for x in range(bld.jobs)] | |||||
| process_pool.extend(processes) | |||||
| Task.Task.can_retrieve_cache = can_retrieve_cache | |||||
| Task.Task.put_files_cache = put_files_cache | |||||
| Task.Task.uid = uid | |||||
| Build.BuildContext.hash_env_vars = hash_env_vars | |||||
| for x in reversed(list(Task.classes.values())): | |||||
| make_cached(x) | |||||
| def cache_command(proc, sig, files_from, files_to): | |||||
| """ | |||||
| Create a command for cache worker processes, returns a pickled | |||||
| base64-encoded tuple containing the task signature, a list of files to | |||||
| cache and a list of files files to get from cache (one of the lists | |||||
| is assumed to be empty) | |||||
| """ | |||||
| obj = base64.b64encode(cPickle.dumps([sig, files_from, files_to])) | |||||
| proc.stdin.write(obj) | |||||
| proc.stdin.write('\n'.encode()) | |||||
| proc.stdin.flush() | |||||
| obj = proc.stdout.readline() | |||||
| if not obj: | |||||
| raise OSError('Preforked sub-process %r died' % proc.pid) | |||||
| return cPickle.loads(base64.b64decode(obj)) | |||||
| try: | |||||
| copyfun = os.link | |||||
| except NameError: | |||||
| copyfun = shutil.copy2 | |||||
| def atomic_copy(orig, dest): | |||||
| """ | |||||
| Copy files to the cache, the operation is atomic for a given file | |||||
| """ | |||||
| global copyfun | |||||
| tmp = dest + '.tmp' | |||||
| up = os.path.dirname(dest) | |||||
| try: | |||||
| os.makedirs(up) | |||||
| except OSError: | |||||
| pass | |||||
| try: | |||||
| copyfun(orig, tmp) | |||||
| except OSError as e: | |||||
| if e.errno == errno.EXDEV: | |||||
| copyfun = shutil.copy2 | |||||
| copyfun(orig, tmp) | |||||
| else: | |||||
| raise | |||||
| os.rename(tmp, dest) | |||||
| def lru_trim(): | |||||
| """ | |||||
| the cache folders take the form: | |||||
| `CACHE_DIR/0b/0b180f82246d726ece37c8ccd0fb1cde2650d7bfcf122ec1f169079a3bfc0ab9` | |||||
| they are listed in order of last access, and then removed | |||||
| until the amount of folders is within TRIM_MAX_FOLDERS and the total space | |||||
| taken by files is less than EVICT_MAX_BYTES | |||||
| """ | |||||
| lst = [] | |||||
| for up in os.listdir(CACHE_DIR): | |||||
| if len(up) == 2: | |||||
| sub = os.path.join(CACHE_DIR, up) | |||||
| for hval in os.listdir(sub): | |||||
| path = os.path.join(sub, hval) | |||||
| size = 0 | |||||
| for fname in os.listdir(path): | |||||
| try: | |||||
| size += os.lstat(os.path.join(path, fname)).st_size | |||||
| except OSError: | |||||
| pass | |||||
| lst.append((os.stat(path).st_mtime, size, path)) | |||||
| lst.sort(key=lambda x: x[0]) | |||||
| lst.reverse() | |||||
| tot = sum(x[1] for x in lst) | |||||
| while tot > EVICT_MAX_BYTES or len(lst) > TRIM_MAX_FOLDERS: | |||||
| _, tmp_size, path = lst.pop() | |||||
| tot -= tmp_size | |||||
| tmp = path + '.remove' | |||||
| try: | |||||
| shutil.rmtree(tmp) | |||||
| except OSError: | |||||
| pass | |||||
| try: | |||||
| os.rename(path, tmp) | |||||
| except OSError: | |||||
| sys.stderr.write('Could not rename %r to %r\n' % (path, tmp)) | |||||
| else: | |||||
| try: | |||||
| shutil.rmtree(tmp) | |||||
| except OSError: | |||||
| sys.stderr.write('Could not remove %r\n' % tmp) | |||||
| sys.stderr.write("Cache trimmed: %r bytes in %r folders left\n" % (tot, len(lst))) | |||||
| def lru_evict(): | |||||
| """ | |||||
| Reduce the cache size | |||||
| """ | |||||
| lockfile = os.path.join(CACHE_DIR, 'all.lock') | |||||
| try: | |||||
| st = os.stat(lockfile) | |||||
| except EnvironmentError as e: | |||||
| if e.errno == errno.ENOENT: | |||||
| with open(lockfile, 'w') as f: | |||||
| f.write('') | |||||
| return | |||||
| else: | |||||
| raise | |||||
| if st.st_mtime < time.time() - EVICT_INTERVAL_MINUTES * 60: | |||||
| # check every EVICT_INTERVAL_MINUTES minutes if the cache is too big | |||||
| # OCLOEXEC is unnecessary because no processes are spawned | |||||
| fd = os.open(lockfile, os.O_RDWR | os.O_CREAT, 0o755) | |||||
| try: | |||||
| try: | |||||
| fcntl.flock(fd, fcntl.LOCK_EX | fcntl.LOCK_NB) | |||||
| except EnvironmentError: | |||||
| if WAFCACHE_VERBOSITY: | |||||
| sys.stderr.write('wafcache: another cleaning process is running\n') | |||||
| else: | |||||
| # now dow the actual cleanup | |||||
| lru_trim() | |||||
| os.utime(lockfile, None) | |||||
| finally: | |||||
| os.close(fd) | |||||
| class netcache(object): | |||||
| def __init__(self): | |||||
| self.http = urllib3.PoolManager() | |||||
| def url_of(self, sig, i): | |||||
| return "%s/%s/%s" % (CACHE_DIR, sig, i) | |||||
| def upload(self, file_path, sig, i): | |||||
| url = self.url_of(sig, i) | |||||
| with open(file_path, 'rb') as f: | |||||
| file_data = f.read() | |||||
| r = self.http.request('POST', url, timeout=60, | |||||
| fields={ 'file': ('%s/%s' % (sig, i), file_data), }) | |||||
| if r.status >= 400: | |||||
| raise OSError("Invalid status %r %r" % (url, r.status)) | |||||
| def download(self, file_path, sig, i): | |||||
| url = self.url_of(sig, i) | |||||
| with self.http.request('GET', url, preload_content=False, timeout=60) as inf: | |||||
| if inf.status >= 400: | |||||
| raise OSError("Invalid status %r %r" % (url, inf.status)) | |||||
| with open(file_path, 'wb') as out: | |||||
| shutil.copyfileobj(inf, out) | |||||
| def copy_to_cache(self, sig, files_from, files_to): | |||||
| try: | |||||
| for i, x in enumerate(files_from): | |||||
| if not os.path.islink(x): | |||||
| self.upload(x, sig, i) | |||||
| except Exception: | |||||
| return traceback.format_exc() | |||||
| return OK | |||||
| def copy_from_cache(self, sig, files_from, files_to): | |||||
| try: | |||||
| for i, x in enumerate(files_to): | |||||
| self.download(x, sig, i) | |||||
| except Exception: | |||||
| return traceback.format_exc() | |||||
| return OK | |||||
| class fcache(object): | |||||
| def __init__(self): | |||||
| if not os.path.exists(CACHE_DIR): | |||||
| try: | |||||
| os.makedirs(CACHE_DIR) | |||||
| except OSError: | |||||
| pass | |||||
| if not os.path.exists(CACHE_DIR): | |||||
| raise ValueError('Could not initialize the cache directory') | |||||
| def copy_to_cache(self, sig, files_from, files_to): | |||||
| """ | |||||
| Copy files to the cache, existing files are overwritten, | |||||
| and the copy is atomic only for a given file, not for all files | |||||
| that belong to a given task object | |||||
| """ | |||||
| try: | |||||
| for i, x in enumerate(files_from): | |||||
| dest = os.path.join(CACHE_DIR, sig[:2], sig, str(i)) | |||||
| atomic_copy(x, dest) | |||||
| except Exception: | |||||
| return traceback.format_exc() | |||||
| else: | |||||
| # attempt trimming if caching was successful: | |||||
| # we may have things to trim! | |||||
| try: | |||||
| lru_evict() | |||||
| except Exception: | |||||
| return traceback.format_exc() | |||||
| return OK | |||||
| def copy_from_cache(self, sig, files_from, files_to): | |||||
| """ | |||||
| Copy files from the cache | |||||
| """ | |||||
| try: | |||||
| for i, x in enumerate(files_to): | |||||
| orig = os.path.join(CACHE_DIR, sig[:2], sig, str(i)) | |||||
| atomic_copy(orig, x) | |||||
| # success! update the cache time | |||||
| os.utime(os.path.join(CACHE_DIR, sig[:2], sig), None) | |||||
| except Exception: | |||||
| return traceback.format_exc() | |||||
| return OK | |||||
| class bucket_cache(object): | |||||
| def bucket_copy(self, source, target): | |||||
| if WAFCACHE_CMD: | |||||
| def replacer(match): | |||||
| if match.group('src'): | |||||
| return source | |||||
| elif match.group('tgt'): | |||||
| return target | |||||
| cmd = [re_waf_cmd.sub(replacer, x) for x in shlex.split(WAFCACHE_CMD)] | |||||
| elif CACHE_DIR.startswith('s3://'): | |||||
| cmd = ['aws', 's3', 'cp', source, target] | |||||
| elif CACHE_DIR.startswith('gs://'): | |||||
| cmd = ['gsutil', 'cp', source, target] | |||||
| else: | |||||
| cmd = ['mc', 'cp', source, target] | |||||
| proc = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE) | |||||
| out, err = proc.communicate() | |||||
| if proc.returncode: | |||||
| raise OSError('Error copy %r to %r using: %r (exit %r):\n out:%s\n err:%s' % ( | |||||
| source, target, cmd, proc.returncode, out.decode(errors='replace'), err.decode(errors='replace'))) | |||||
| def copy_to_cache(self, sig, files_from, files_to): | |||||
| try: | |||||
| for i, x in enumerate(files_from): | |||||
| dest = os.path.join(CACHE_DIR, sig[:2], sig, str(i)) | |||||
| self.bucket_copy(x, dest) | |||||
| except Exception: | |||||
| return traceback.format_exc() | |||||
| return OK | |||||
| def copy_from_cache(self, sig, files_from, files_to): | |||||
| try: | |||||
| for i, x in enumerate(files_to): | |||||
| orig = os.path.join(CACHE_DIR, sig[:2], sig, str(i)) | |||||
| self.bucket_copy(orig, x) | |||||
| except EnvironmentError: | |||||
| return traceback.format_exc() | |||||
| return OK | |||||
| def loop(service): | |||||
| """ | |||||
| This function is run when this file is run as a standalone python script, | |||||
| it assumes a parent process that will communicate the commands to it | |||||
| as pickled-encoded tuples (one line per command) | |||||
| The commands are to copy files to the cache or copy files from the | |||||
| cache to a target destination | |||||
| """ | |||||
| # one operation is performed at a single time by a single process | |||||
| # therefore stdin never has more than one line | |||||
| txt = sys.stdin.readline().strip() | |||||
| if not txt: | |||||
| # parent process probably ended | |||||
| sys.exit(1) | |||||
| ret = OK | |||||
| [sig, files_from, files_to] = cPickle.loads(base64.b64decode(txt)) | |||||
| if files_from: | |||||
| # TODO return early when pushing files upstream | |||||
| ret = service.copy_to_cache(sig, files_from, files_to) | |||||
| elif files_to: | |||||
| # the build process waits for workers to (possibly) obtain files from the cache | |||||
| ret = service.copy_from_cache(sig, files_from, files_to) | |||||
| else: | |||||
| ret = "Invalid command" | |||||
| obj = base64.b64encode(cPickle.dumps(ret)) | |||||
| sys.stdout.write(obj.decode()) | |||||
| sys.stdout.write('\n') | |||||
| sys.stdout.flush() | |||||
| if __name__ == '__main__': | |||||
| if CACHE_DIR.startswith('s3://') or CACHE_DIR.startswith('gs://') or CACHE_DIR.startswith('minio://'): | |||||
| if CACHE_DIR.startswith('minio://'): | |||||
| CACHE_DIR = CACHE_DIR[8:] # minio doesn't need the protocol part, uses config aliases | |||||
| service = bucket_cache() | |||||
| elif CACHE_DIR.startswith('http'): | |||||
| service = netcache() | |||||
| else: | |||||
| service = fcache() | |||||
| while 1: | |||||
| try: | |||||
| loop(service) | |||||
| except KeyboardInterrupt: | |||||
| break | |||||
| @@ -99,7 +99,7 @@ env.PROJ_CONFIGURATION = { | |||||
| ... | ... | ||||
| } | } | ||||
| 'Release': { | 'Release': { | ||||
| 'ARCHS' x86_64' | |||||
| 'ARCHS': x86_64' | |||||
| ... | ... | ||||
| } | } | ||||
| } | } | ||||
| @@ -163,12 +163,12 @@ class XCodeNode(object): | |||||
| result = result + "\t\t}" | result = result + "\t\t}" | ||||
| return result | return result | ||||
| elif isinstance(value, str): | elif isinstance(value, str): | ||||
| return "\"%s\"" % value | |||||
| return '"%s"' % value.replace('"', '\\\\\\"') | |||||
| elif isinstance(value, list): | elif isinstance(value, list): | ||||
| result = "(\n" | result = "(\n" | ||||
| for i in value: | for i in value: | ||||
| result = result + "\t\t\t%s,\n" % self.tostring(i) | |||||
| result = result + "\t\t)" | |||||
| result = result + "\t\t\t\t%s,\n" % self.tostring(i) | |||||
| result = result + "\t\t\t)" | |||||
| return result | return result | ||||
| elif isinstance(value, XCodeNode): | elif isinstance(value, XCodeNode): | ||||
| return value._id | return value._id | ||||
| @@ -565,13 +565,13 @@ def process_xcode(self): | |||||
| # Override target specific build settings | # Override target specific build settings | ||||
| bldsettings = { | bldsettings = { | ||||
| 'HEADER_SEARCH_PATHS': ['$(inherited)'] + self.env['INCPATHS'], | 'HEADER_SEARCH_PATHS': ['$(inherited)'] + self.env['INCPATHS'], | ||||
| 'LIBRARY_SEARCH_PATHS': ['$(inherited)'] + Utils.to_list(self.env.LIBPATH) + Utils.to_list(self.env.STLIBPATH) + Utils.to_list(self.env.LIBDIR) , | |||||
| 'LIBRARY_SEARCH_PATHS': ['$(inherited)'] + Utils.to_list(self.env.LIBPATH) + Utils.to_list(self.env.STLIBPATH) + Utils.to_list(self.env.LIBDIR), | |||||
| 'FRAMEWORK_SEARCH_PATHS': ['$(inherited)'] + Utils.to_list(self.env.FRAMEWORKPATH), | 'FRAMEWORK_SEARCH_PATHS': ['$(inherited)'] + Utils.to_list(self.env.FRAMEWORKPATH), | ||||
| 'OTHER_LDFLAGS': libs + ' ' + frameworks, | |||||
| 'OTHER_LIBTOOLFLAGS': bld.env['LINKFLAGS'], | |||||
| 'OTHER_LDFLAGS': libs + ' ' + frameworks + ' ' + ' '.join(bld.env['LINKFLAGS']), | |||||
| 'OTHER_CPLUSPLUSFLAGS': Utils.to_list(self.env['CXXFLAGS']), | 'OTHER_CPLUSPLUSFLAGS': Utils.to_list(self.env['CXXFLAGS']), | ||||
| 'OTHER_CFLAGS': Utils.to_list(self.env['CFLAGS']), | 'OTHER_CFLAGS': Utils.to_list(self.env['CFLAGS']), | ||||
| 'INSTALL_PATH': [] | |||||
| 'INSTALL_PATH': [], | |||||
| 'GCC_PREPROCESSOR_DEFINITIONS': self.env['DEFINES'] | |||||
| } | } | ||||
| # Install path | # Install path | ||||
| @@ -591,7 +591,7 @@ def process_xcode(self): | |||||
| # The keys represents different build configuration, e.g. Debug, Release and so on.. | # The keys represents different build configuration, e.g. Debug, Release and so on.. | ||||
| # Insert our generated build settings to all configuration names | # Insert our generated build settings to all configuration names | ||||
| keys = set(settings.keys() + bld.env.PROJ_CONFIGURATION.keys()) | |||||
| keys = set(settings.keys()) | set(bld.env.PROJ_CONFIGURATION.keys()) | |||||
| for k in keys: | for k in keys: | ||||
| if k in settings: | if k in settings: | ||||
| settings[k].update(bldsettings) | settings[k].update(bldsettings) | ||||
| @@ -56,7 +56,7 @@ def r1(code): | |||||
| @subst('Runner.py') | @subst('Runner.py') | ||||
| def r4(code): | def r4(code): | ||||
| "generator syntax" | "generator syntax" | ||||
| return code.replace('next(self.biter)', 'self.biter.next()') | |||||
| return code.replace('next(self.biter)', 'self.biter.next()').replace('self.daemon = True', 'self.setDaemon(1)') | |||||
| @subst('Context.py') | @subst('Context.py') | ||||
| def r5(code): | def r5(code): | ||||
| @@ -27,6 +27,10 @@ def run(): | |||||
| [cmd, kwargs, cargs] = cPickle.loads(base64.b64decode(txt)) | [cmd, kwargs, cargs] = cPickle.loads(base64.b64decode(txt)) | ||||
| cargs = cargs or {} | cargs = cargs or {} | ||||
| if not 'close_fds' in kwargs: | |||||
| # workers have no fds | |||||
| kwargs['close_fds'] = False | |||||
| ret = 1 | ret = 1 | ||||
| out, err, ex, trace = (None, None, None, None) | out, err, ex, trace = (None, None, None, None) | ||||
| try: | try: | ||||