2023-06-11 12:31:23 +02:00
|
|
|
# MIT License
|
|
|
|
#
|
|
|
|
# Copyright The SCons Foundation
|
|
|
|
#
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
# a copy of this software and associated documentation files (the
|
|
|
|
# "Software"), to deal in the Software without restriction, including
|
|
|
|
# without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
# distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
# permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
# the following conditions:
|
|
|
|
#
|
|
|
|
# The above copyright notice and this permission notice shall be included
|
|
|
|
# in all copies or substantial portions of the Software.
|
|
|
|
#
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
|
|
|
|
# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
|
|
|
# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
|
|
|
"""This module defines the Python API provided to SConscript files."""
|
|
|
|
|
|
|
|
import SCons
|
|
|
|
import SCons.Action
|
|
|
|
import SCons.Builder
|
|
|
|
import SCons.Defaults
|
|
|
|
import SCons.Environment
|
|
|
|
import SCons.Errors
|
|
|
|
import SCons.Node
|
|
|
|
import SCons.Node.Alias
|
|
|
|
import SCons.Node.FS
|
|
|
|
import SCons.Platform
|
|
|
|
import SCons.SConf
|
|
|
|
import SCons.Tool
|
|
|
|
from SCons.Util import is_List, is_String, is_Dict, flatten
|
|
|
|
from SCons.Node import SConscriptNodes
|
|
|
|
from . import Main
|
|
|
|
|
|
|
|
import os
|
|
|
|
import os.path
|
|
|
|
import re
|
|
|
|
import sys
|
|
|
|
import traceback
|
|
|
|
import time
|
2024-08-21 14:52:56 +02:00
|
|
|
from typing import Tuple
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
class SConscriptReturn(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
launch_dir = os.path.abspath(os.curdir)
|
|
|
|
|
|
|
|
GlobalDict = None
|
|
|
|
|
|
|
|
# global exports set by Export():
|
|
|
|
global_exports = {}
|
|
|
|
|
|
|
|
# chdir flag
|
|
|
|
sconscript_chdir: bool = True
|
|
|
|
|
|
|
|
def get_calling_namespaces():
|
|
|
|
"""Return the locals and globals for the function that called
|
|
|
|
into this module in the current call stack."""
|
|
|
|
try: 1//0
|
|
|
|
except ZeroDivisionError:
|
|
|
|
# Don't start iterating with the current stack-frame to
|
|
|
|
# prevent creating reference cycles (f_back is safe).
|
|
|
|
frame = sys.exc_info()[2].tb_frame.f_back
|
|
|
|
|
|
|
|
# Find the first frame that *isn't* from this file. This means
|
|
|
|
# that we expect all of the SCons frames that implement an Export()
|
|
|
|
# or SConscript() call to be in this file, so that we can identify
|
|
|
|
# the first non-Script.SConscript frame as the user's local calling
|
|
|
|
# environment, and the locals and globals dictionaries from that
|
|
|
|
# frame as the calling namespaces. See the comment below preceding
|
|
|
|
# the DefaultEnvironmentCall block for even more explanation.
|
|
|
|
while frame.f_globals.get("__name__") == __name__:
|
|
|
|
frame = frame.f_back
|
|
|
|
|
|
|
|
return frame.f_locals, frame.f_globals
|
|
|
|
|
|
|
|
|
|
|
|
def compute_exports(exports):
|
|
|
|
"""Compute a dictionary of exports given one of the parameters
|
|
|
|
to the Export() function or the exports argument to SConscript()."""
|
|
|
|
|
|
|
|
loc, glob = get_calling_namespaces()
|
|
|
|
|
|
|
|
retval = {}
|
|
|
|
try:
|
|
|
|
for export in exports:
|
|
|
|
if is_Dict(export):
|
|
|
|
retval.update(export)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
retval[export] = loc[export]
|
|
|
|
except KeyError:
|
|
|
|
retval[export] = glob[export]
|
|
|
|
except KeyError as x:
|
|
|
|
raise SCons.Errors.UserError("Export of non-existent variable '%s'"%x)
|
|
|
|
|
|
|
|
return retval
|
|
|
|
|
|
|
|
class Frame:
|
|
|
|
"""A frame on the SConstruct/SConscript call stack"""
|
2024-04-15 15:35:35 +02:00
|
|
|
def __init__(self, fs, exports, sconscript) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
self.globals = BuildDefaultGlobals()
|
|
|
|
self.retval = None
|
|
|
|
self.prev_dir = fs.getcwd()
|
|
|
|
self.exports = compute_exports(exports) # exports from the calling SConscript
|
|
|
|
# make sure the sconscript attr is a Node.
|
|
|
|
if isinstance(sconscript, SCons.Node.Node):
|
|
|
|
self.sconscript = sconscript
|
|
|
|
elif sconscript == '-':
|
|
|
|
self.sconscript = None
|
|
|
|
else:
|
|
|
|
self.sconscript = fs.File(str(sconscript))
|
|
|
|
|
|
|
|
# the SConstruct/SConscript call stack:
|
|
|
|
call_stack = []
|
|
|
|
|
|
|
|
# For documentation on the methods in this file, see the scons man-page
|
|
|
|
|
|
|
|
def Return(*vars, **kw):
|
|
|
|
retval = []
|
|
|
|
try:
|
|
|
|
fvars = flatten(vars)
|
|
|
|
for var in fvars:
|
|
|
|
for v in var.split():
|
|
|
|
retval.append(call_stack[-1].globals[v])
|
|
|
|
except KeyError as x:
|
|
|
|
raise SCons.Errors.UserError("Return of non-existent variable '%s'"%x)
|
|
|
|
|
|
|
|
if len(retval) == 1:
|
|
|
|
call_stack[-1].retval = retval[0]
|
|
|
|
else:
|
|
|
|
call_stack[-1].retval = tuple(retval)
|
|
|
|
|
|
|
|
stop = kw.get('stop', True)
|
|
|
|
|
|
|
|
if stop:
|
|
|
|
raise SConscriptReturn
|
|
|
|
|
|
|
|
|
|
|
|
stack_bottom = '% Stack boTTom %' # hard to define a variable w/this name :)
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def handle_missing_SConscript(f: str, must_exist: bool = True) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
"""Take appropriate action on missing file in SConscript() call.
|
|
|
|
|
|
|
|
Print a warning or raise an exception on missing file, unless
|
2024-04-15 15:35:35 +02:00
|
|
|
missing is explicitly allowed by the *must_exist* parameter or by
|
|
|
|
a global flag.
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
Args:
|
2024-04-15 15:35:35 +02:00
|
|
|
f: path to missing configuration file
|
|
|
|
must_exist: if true (the default), fail. If false
|
|
|
|
do nothing, allowing a build to declare it's okay to be missing.
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
Raises:
|
2024-04-15 15:35:35 +02:00
|
|
|
UserError: if *must_exist* is true or if global
|
2023-06-11 12:31:23 +02:00
|
|
|
:data:`SCons.Script._no_missing_sconscript` is true.
|
2024-04-15 15:35:35 +02:00
|
|
|
|
|
|
|
.. versionchanged: 4.6.0
|
|
|
|
Changed default from False.
|
2023-06-11 12:31:23 +02:00
|
|
|
"""
|
2024-04-15 15:35:35 +02:00
|
|
|
if not must_exist: # explicitly set False: ok
|
|
|
|
return
|
|
|
|
if not SCons.Script._no_missing_sconscript: # system default changed: ok
|
|
|
|
return
|
|
|
|
msg = f"missing SConscript file {f.get_internal_path()!r}"
|
|
|
|
raise SCons.Errors.UserError(msg)
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
def _SConscript(fs, *files, **kw):
|
|
|
|
top = fs.Top
|
|
|
|
sd = fs.SConstruct_dir.rdir()
|
|
|
|
exports = kw.get('exports', [])
|
|
|
|
|
|
|
|
# evaluate each SConscript file
|
|
|
|
results = []
|
|
|
|
for fn in files:
|
|
|
|
call_stack.append(Frame(fs, exports, fn))
|
|
|
|
old_sys_path = sys.path
|
|
|
|
try:
|
|
|
|
SCons.Script.sconscript_reading = SCons.Script.sconscript_reading + 1
|
|
|
|
if fn == "-":
|
|
|
|
exec(sys.stdin.read(), call_stack[-1].globals)
|
|
|
|
else:
|
|
|
|
if isinstance(fn, SCons.Node.Node):
|
|
|
|
f = fn
|
|
|
|
else:
|
|
|
|
f = fs.File(str(fn))
|
|
|
|
_file_ = None
|
|
|
|
SConscriptNodes.add(f)
|
|
|
|
|
|
|
|
# Change directory to the top of the source
|
|
|
|
# tree to make sure the os's cwd and the cwd of
|
|
|
|
# fs match so we can open the SConscript.
|
|
|
|
fs.chdir(top, change_os_dir=True)
|
|
|
|
if f.rexists():
|
|
|
|
actual = f.rfile()
|
|
|
|
_file_ = open(actual.get_abspath(), "rb")
|
|
|
|
elif f.srcnode().rexists():
|
|
|
|
actual = f.srcnode().rfile()
|
|
|
|
_file_ = open(actual.get_abspath(), "rb")
|
|
|
|
elif f.has_src_builder():
|
|
|
|
# The SConscript file apparently exists in a source
|
|
|
|
# code management system. Build it, but then clear
|
|
|
|
# the builder so that it doesn't get built *again*
|
|
|
|
# during the actual build phase.
|
|
|
|
f.build()
|
|
|
|
f.built()
|
|
|
|
f.builder_set(None)
|
|
|
|
if f.exists():
|
|
|
|
_file_ = open(f.get_abspath(), "rb")
|
|
|
|
if _file_:
|
|
|
|
# Chdir to the SConscript directory. Use a path
|
|
|
|
# name relative to the SConstruct file so that if
|
|
|
|
# we're using the -f option, we're essentially
|
|
|
|
# creating a parallel SConscript directory structure
|
|
|
|
# in our local directory tree.
|
|
|
|
#
|
|
|
|
# XXX This is broken for multiple-repository cases
|
|
|
|
# where the SConstruct and SConscript files might be
|
|
|
|
# in different Repositories. For now, cross that
|
|
|
|
# bridge when someone comes to it.
|
|
|
|
try:
|
|
|
|
src_dir = kw['src_dir']
|
|
|
|
except KeyError:
|
|
|
|
ldir = fs.Dir(f.dir.get_path(sd))
|
|
|
|
else:
|
|
|
|
ldir = fs.Dir(src_dir)
|
|
|
|
if not ldir.is_under(f.dir):
|
|
|
|
# They specified a source directory, but
|
|
|
|
# it's above the SConscript directory.
|
|
|
|
# Do the sensible thing and just use the
|
|
|
|
# SConcript directory.
|
|
|
|
ldir = fs.Dir(f.dir.get_path(sd))
|
|
|
|
try:
|
|
|
|
fs.chdir(ldir, change_os_dir=sconscript_chdir)
|
|
|
|
except OSError:
|
|
|
|
# There was no local directory, so we should be
|
|
|
|
# able to chdir to the Repository directory.
|
|
|
|
# Note that we do this directly, not through
|
|
|
|
# fs.chdir(), because we still need to
|
|
|
|
# interpret the stuff within the SConscript file
|
|
|
|
# relative to where we are logically.
|
|
|
|
fs.chdir(ldir, change_os_dir=False)
|
|
|
|
os.chdir(actual.dir.get_abspath())
|
|
|
|
|
|
|
|
# Append the SConscript directory to the beginning
|
|
|
|
# of sys.path so Python modules in the SConscript
|
|
|
|
# directory can be easily imported.
|
|
|
|
sys.path = [ f.dir.get_abspath() ] + sys.path
|
|
|
|
|
|
|
|
# This is the magic line that actually reads up
|
|
|
|
# and executes the stuff in the SConscript file.
|
|
|
|
# The locals for this frame contain the special
|
|
|
|
# bottom-of-the-stack marker so that any
|
|
|
|
# exceptions that occur when processing this
|
|
|
|
# SConscript can base the printed frames at this
|
|
|
|
# level and not show SCons internals as well.
|
|
|
|
call_stack[-1].globals.update({stack_bottom:1})
|
|
|
|
old_file = call_stack[-1].globals.get('__file__')
|
|
|
|
try:
|
|
|
|
del call_stack[-1].globals['__file__']
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
try:
|
|
|
|
if Main.print_time:
|
|
|
|
start_time = time.perf_counter()
|
|
|
|
scriptdata = _file_.read()
|
|
|
|
scriptname = _file_.name
|
|
|
|
_file_.close()
|
2024-04-15 15:35:35 +02:00
|
|
|
if SCons.Debug.sconscript_trace:
|
|
|
|
print("scons: Entering "+str(scriptname))
|
2023-06-11 12:31:23 +02:00
|
|
|
exec(compile(scriptdata, scriptname, 'exec'), call_stack[-1].globals)
|
2024-04-15 15:35:35 +02:00
|
|
|
if SCons.Debug.sconscript_trace:
|
|
|
|
print("scons: Exiting "+str(scriptname))
|
2023-06-11 12:31:23 +02:00
|
|
|
except SConscriptReturn:
|
2024-04-15 15:35:35 +02:00
|
|
|
if SCons.Debug.sconscript_trace:
|
|
|
|
print("scons: Exiting "+str(scriptname))
|
|
|
|
else:
|
|
|
|
pass
|
2023-06-11 12:31:23 +02:00
|
|
|
finally:
|
|
|
|
if Main.print_time:
|
|
|
|
elapsed = time.perf_counter() - start_time
|
|
|
|
print('SConscript:%s took %0.3f ms' % (f.get_abspath(), elapsed * 1000.0))
|
|
|
|
|
|
|
|
if old_file is not None:
|
|
|
|
call_stack[-1].globals.update({__file__:old_file})
|
|
|
|
|
|
|
|
else:
|
2024-04-15 15:35:35 +02:00
|
|
|
handle_missing_SConscript(f, kw.get('must_exist', True))
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
finally:
|
|
|
|
SCons.Script.sconscript_reading = SCons.Script.sconscript_reading - 1
|
|
|
|
sys.path = old_sys_path
|
|
|
|
frame = call_stack.pop()
|
|
|
|
try:
|
|
|
|
fs.chdir(frame.prev_dir, change_os_dir=sconscript_chdir)
|
|
|
|
except OSError:
|
|
|
|
# There was no local directory, so chdir to the
|
|
|
|
# Repository directory. Like above, we do this
|
|
|
|
# directly.
|
|
|
|
fs.chdir(frame.prev_dir, change_os_dir=False)
|
|
|
|
rdir = frame.prev_dir.rdir()
|
|
|
|
rdir._create() # Make sure there's a directory there.
|
|
|
|
try:
|
|
|
|
os.chdir(rdir.get_abspath())
|
|
|
|
except OSError as e:
|
|
|
|
# We still couldn't chdir there, so raise the error,
|
|
|
|
# but only if actions are being executed.
|
|
|
|
#
|
|
|
|
# If the -n option was used, the directory would *not*
|
|
|
|
# have been created and we should just carry on and
|
|
|
|
# let things muddle through. This isn't guaranteed
|
|
|
|
# to work if the SConscript files are reading things
|
|
|
|
# from disk (for example), but it should work well
|
|
|
|
# enough for most configurations.
|
|
|
|
if SCons.Action.execute_actions:
|
|
|
|
raise e
|
|
|
|
|
|
|
|
results.append(frame.retval)
|
|
|
|
|
|
|
|
# if we only have one script, don't return a tuple
|
|
|
|
if len(results) == 1:
|
|
|
|
return results[0]
|
|
|
|
else:
|
|
|
|
return tuple(results)
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def SConscript_exception(file=sys.stderr) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
"""Print an exception stack trace just for the SConscript file(s).
|
|
|
|
This will show users who have Python errors where the problem is,
|
|
|
|
without cluttering the output with all of the internal calls leading
|
|
|
|
up to where we exec the SConscript."""
|
|
|
|
exc_type, exc_value, exc_tb = sys.exc_info()
|
|
|
|
tb = exc_tb
|
|
|
|
while tb and stack_bottom not in tb.tb_frame.f_locals:
|
|
|
|
tb = tb.tb_next
|
|
|
|
if not tb:
|
|
|
|
# We did not find our exec statement, so this was actually a bug
|
|
|
|
# in SCons itself. Show the whole stack.
|
|
|
|
tb = exc_tb
|
|
|
|
stack = traceback.extract_tb(tb)
|
|
|
|
try:
|
|
|
|
type = exc_type.__name__
|
|
|
|
except AttributeError:
|
|
|
|
type = str(exc_type)
|
|
|
|
if type[:11] == "exceptions.":
|
|
|
|
type = type[11:]
|
|
|
|
file.write('%s: %s:\n' % (type, exc_value))
|
|
|
|
for fname, line, func, text in stack:
|
|
|
|
file.write(' File "%s", line %d:\n' % (fname, line))
|
|
|
|
file.write(' %s\n' % text)
|
|
|
|
|
|
|
|
def annotate(node):
|
|
|
|
"""Annotate a node with the stack frame describing the
|
|
|
|
SConscript file and line number that created it."""
|
|
|
|
tb = sys.exc_info()[2]
|
|
|
|
while tb and stack_bottom not in tb.tb_frame.f_locals:
|
|
|
|
tb = tb.tb_next
|
|
|
|
if not tb:
|
|
|
|
# We did not find any exec of an SConscript file: what?!
|
|
|
|
raise SCons.Errors.InternalError("could not find SConscript stack frame")
|
|
|
|
node.creator = traceback.extract_stack(tb)[0]
|
|
|
|
|
|
|
|
# The following line would cause each Node to be annotated using the
|
|
|
|
# above function. Unfortunately, this is a *huge* performance hit, so
|
|
|
|
# leave this disabled until we find a more efficient mechanism.
|
|
|
|
#SCons.Node.Annotate = annotate
|
|
|
|
|
|
|
|
class SConsEnvironment(SCons.Environment.Base):
|
|
|
|
"""An Environment subclass that contains all of the methods that
|
|
|
|
are particular to the wrapper SCons interface and which aren't
|
|
|
|
(or shouldn't be) part of the build engine itself.
|
|
|
|
|
|
|
|
Note that not all of the methods of this class have corresponding
|
|
|
|
global functions, there are some private methods.
|
|
|
|
"""
|
|
|
|
|
|
|
|
#
|
|
|
|
# Private methods of an SConsEnvironment.
|
|
|
|
#
|
|
|
|
@staticmethod
|
2024-08-21 14:52:56 +02:00
|
|
|
def _get_major_minor_revision(version_string: str) -> Tuple[int, int, int]:
|
2023-06-11 12:31:23 +02:00
|
|
|
"""Split a version string into major, minor and (optionally)
|
|
|
|
revision parts.
|
|
|
|
|
|
|
|
This is complicated by the fact that a version string can be
|
|
|
|
something like 3.2b1."""
|
|
|
|
version = version_string.split(' ')[0].split('.')
|
|
|
|
v_major = int(version[0])
|
|
|
|
v_minor = int(re.match(r'\d+', version[1]).group())
|
|
|
|
if len(version) >= 3:
|
|
|
|
v_revision = int(re.match(r'\d+', version[2]).group())
|
|
|
|
else:
|
|
|
|
v_revision = 0
|
|
|
|
return v_major, v_minor, v_revision
|
|
|
|
|
|
|
|
def _get_SConscript_filenames(self, ls, kw):
|
|
|
|
"""
|
|
|
|
Convert the parameters passed to SConscript() calls into a list
|
|
|
|
of files and export variables. If the parameters are invalid,
|
|
|
|
throws SCons.Errors.UserError. Returns a tuple (l, e) where l
|
|
|
|
is a list of SConscript filenames and e is a list of exports.
|
|
|
|
"""
|
|
|
|
exports = []
|
|
|
|
|
|
|
|
if len(ls) == 0:
|
|
|
|
try:
|
|
|
|
dirs = kw["dirs"]
|
|
|
|
except KeyError:
|
|
|
|
raise SCons.Errors.UserError("Invalid SConscript usage - no parameters")
|
|
|
|
|
|
|
|
if not is_List(dirs):
|
|
|
|
dirs = [ dirs ]
|
|
|
|
dirs = list(map(str, dirs))
|
|
|
|
|
|
|
|
name = kw.get('name', 'SConscript')
|
|
|
|
|
|
|
|
files = [os.path.join(n, name) for n in dirs]
|
|
|
|
|
|
|
|
elif len(ls) == 1:
|
|
|
|
|
|
|
|
files = ls[0]
|
|
|
|
|
|
|
|
elif len(ls) == 2:
|
|
|
|
|
|
|
|
files = ls[0]
|
|
|
|
exports = self.Split(ls[1])
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
|
|
|
raise SCons.Errors.UserError("Invalid SConscript() usage - too many arguments")
|
|
|
|
|
|
|
|
if not is_List(files):
|
|
|
|
files = [ files ]
|
|
|
|
|
|
|
|
if kw.get('exports'):
|
|
|
|
exports.extend(self.Split(kw['exports']))
|
|
|
|
|
|
|
|
variant_dir = kw.get('variant_dir')
|
|
|
|
if variant_dir:
|
|
|
|
if len(files) != 1:
|
|
|
|
raise SCons.Errors.UserError("Invalid SConscript() usage - can only specify one SConscript with a variant_dir")
|
|
|
|
duplicate = kw.get('duplicate', 1)
|
|
|
|
src_dir = kw.get('src_dir')
|
|
|
|
if not src_dir:
|
|
|
|
src_dir, fname = os.path.split(str(files[0]))
|
|
|
|
files = [os.path.join(str(variant_dir), fname)]
|
|
|
|
else:
|
|
|
|
if not isinstance(src_dir, SCons.Node.Node):
|
|
|
|
src_dir = self.fs.Dir(src_dir)
|
|
|
|
fn = files[0]
|
|
|
|
if not isinstance(fn, SCons.Node.Node):
|
|
|
|
fn = self.fs.File(fn)
|
|
|
|
if fn.is_under(src_dir):
|
|
|
|
# Get path relative to the source directory.
|
|
|
|
fname = fn.get_path(src_dir)
|
|
|
|
files = [os.path.join(str(variant_dir), fname)]
|
|
|
|
else:
|
|
|
|
files = [fn.get_abspath()]
|
|
|
|
kw['src_dir'] = variant_dir
|
|
|
|
self.fs.VariantDir(variant_dir, src_dir, duplicate)
|
|
|
|
|
|
|
|
return (files, exports)
|
|
|
|
|
|
|
|
#
|
|
|
|
# Public methods of an SConsEnvironment. These get
|
|
|
|
# entry points in the global namespace so they can be called
|
|
|
|
# as global functions.
|
|
|
|
#
|
|
|
|
|
|
|
|
def Configure(self, *args, **kw):
|
|
|
|
if not SCons.Script.sconscript_reading:
|
|
|
|
raise SCons.Errors.UserError("Calling Configure from Builders is not supported.")
|
|
|
|
kw['_depth'] = kw.get('_depth', 0) + 1
|
|
|
|
return SCons.Environment.Base.Configure(self, *args, **kw)
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def Default(self, *targets) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
SCons.Script._Set_Default_Targets(self, targets)
|
|
|
|
|
|
|
|
@staticmethod
|
2024-08-21 14:52:56 +02:00
|
|
|
def GetSConsVersion() -> Tuple[int, int, int]:
|
|
|
|
"""Return the current SCons version.
|
|
|
|
|
|
|
|
.. versionadded:: 4.8.0
|
|
|
|
"""
|
|
|
|
return SConsEnvironment._get_major_minor_revision(SCons.__version__)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def EnsureSConsVersion(major: int, minor: int, revision: int = 0) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
"""Exit abnormally if the SCons version is not late enough."""
|
|
|
|
# split string to avoid replacement during build process
|
|
|
|
if SCons.__version__ == '__' + 'VERSION__':
|
|
|
|
SCons.Warnings.warn(SCons.Warnings.DevelopmentVersionWarning,
|
|
|
|
"EnsureSConsVersion is ignored for development version")
|
|
|
|
return
|
2024-08-21 14:52:56 +02:00
|
|
|
if SConsEnvironment.GetSConsVersion() < (major, minor, revision):
|
2023-06-11 12:31:23 +02:00
|
|
|
if revision:
|
|
|
|
scons_ver_string = '%d.%d.%d' % (major, minor, revision)
|
|
|
|
else:
|
|
|
|
scons_ver_string = '%d.%d' % (major, minor)
|
|
|
|
print("SCons %s or greater required, but you have SCons %s" % \
|
|
|
|
(scons_ver_string, SCons.__version__))
|
|
|
|
sys.exit(2)
|
|
|
|
|
|
|
|
@staticmethod
|
2024-04-15 15:35:35 +02:00
|
|
|
def EnsurePythonVersion(major, minor) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
"""Exit abnormally if the Python version is not late enough."""
|
|
|
|
if sys.version_info < (major, minor):
|
|
|
|
v = sys.version.split()[0]
|
|
|
|
print("Python %d.%d or greater required, but you have Python %s" %(major,minor,v))
|
|
|
|
sys.exit(2)
|
|
|
|
|
|
|
|
@staticmethod
|
2024-04-15 15:35:35 +02:00
|
|
|
def Exit(value: int=0) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
sys.exit(value)
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def Export(self, *vars, **kw) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
for var in vars:
|
|
|
|
global_exports.update(compute_exports(self.Split(var)))
|
|
|
|
global_exports.update(kw)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def GetLaunchDir():
|
|
|
|
global launch_dir
|
|
|
|
return launch_dir
|
|
|
|
|
|
|
|
def GetOption(self, name):
|
|
|
|
name = self.subst(name)
|
|
|
|
return SCons.Script.Main.GetOption(name)
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def Help(self, text, append: bool = False, keep_local: bool = False) -> None:
|
|
|
|
"""Update the help text.
|
|
|
|
|
|
|
|
The previous help text has *text* appended to it, except on the
|
|
|
|
first call. On first call, the values of *append* and *keep_local*
|
|
|
|
are considered to determine what is appended to.
|
2023-06-11 12:31:23 +02:00
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
Arguments:
|
|
|
|
text: string to add to the help text.
|
|
|
|
append: on first call, if true, keep the existing help text
|
|
|
|
(default False).
|
|
|
|
keep_local: on first call, if true and *append* is also true,
|
|
|
|
keep only the help text from AddOption calls.
|
|
|
|
|
|
|
|
.. versionchanged:: 4.6.0
|
|
|
|
The *keep_local* parameter was added.
|
|
|
|
"""
|
2023-06-11 12:31:23 +02:00
|
|
|
text = self.subst(text, raw=1)
|
2024-04-15 15:35:35 +02:00
|
|
|
SCons.Script.HelpFunction(text, append=append, keep_local=keep_local)
|
2023-06-11 12:31:23 +02:00
|
|
|
|
|
|
|
def Import(self, *vars):
|
|
|
|
try:
|
|
|
|
frame = call_stack[-1]
|
|
|
|
globals = frame.globals
|
|
|
|
exports = frame.exports
|
|
|
|
for var in vars:
|
|
|
|
var = self.Split(var)
|
|
|
|
for v in var:
|
|
|
|
if v == '*':
|
|
|
|
globals.update(global_exports)
|
|
|
|
globals.update(exports)
|
|
|
|
else:
|
|
|
|
if v in exports:
|
|
|
|
globals[v] = exports[v]
|
|
|
|
else:
|
|
|
|
globals[v] = global_exports[v]
|
|
|
|
except KeyError as x:
|
|
|
|
raise SCons.Errors.UserError("Import of non-existent variable '%s'"%x)
|
|
|
|
|
|
|
|
def SConscript(self, *ls, **kw):
|
|
|
|
"""Execute SCons configuration files.
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
*ls (str or list): configuration file(s) to execute.
|
|
|
|
|
|
|
|
Keyword arguments:
|
|
|
|
dirs (list): execute SConscript in each listed directory.
|
|
|
|
name (str): execute script 'name' (used only with 'dirs').
|
|
|
|
exports (list or dict): locally export variables the
|
|
|
|
called script(s) can import.
|
|
|
|
variant_dir (str): mirror sources needed for the build in
|
|
|
|
a variant directory to allow building in it.
|
|
|
|
duplicate (bool): physically duplicate sources instead of just
|
|
|
|
adjusting paths of derived files (used only with 'variant_dir')
|
|
|
|
(default is True).
|
|
|
|
must_exist (bool): fail if a requested script is missing
|
|
|
|
(default is False, default is deprecated).
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
list of variables returned by the called script
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
UserError: a script is not found and such exceptions are enabled.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def subst_element(x, subst=self.subst):
|
|
|
|
if SCons.Util.is_List(x):
|
|
|
|
x = list(map(subst, x))
|
|
|
|
else:
|
|
|
|
x = subst(x)
|
|
|
|
return x
|
|
|
|
ls = list(map(subst_element, ls))
|
|
|
|
subst_kw = {}
|
|
|
|
for key, val in kw.items():
|
|
|
|
if is_String(val):
|
|
|
|
val = self.subst(val)
|
|
|
|
elif SCons.Util.is_List(val):
|
|
|
|
val = [self.subst(v) if is_String(v) else v for v in val]
|
|
|
|
subst_kw[key] = val
|
|
|
|
|
|
|
|
files, exports = self._get_SConscript_filenames(ls, subst_kw)
|
|
|
|
subst_kw['exports'] = exports
|
|
|
|
return _SConscript(self.fs, *files, **subst_kw)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def SConscriptChdir(flag: bool) -> None:
|
|
|
|
global sconscript_chdir
|
|
|
|
sconscript_chdir = flag
|
|
|
|
|
2024-04-15 15:35:35 +02:00
|
|
|
def SetOption(self, name, value) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
name = self.subst(name)
|
|
|
|
SCons.Script.Main.SetOption(name, value)
|
|
|
|
|
|
|
|
#
|
|
|
|
#
|
|
|
|
#
|
|
|
|
SCons.Environment.Environment = SConsEnvironment
|
|
|
|
|
|
|
|
def Configure(*args, **kw):
|
|
|
|
if not SCons.Script.sconscript_reading:
|
|
|
|
raise SCons.Errors.UserError("Calling Configure from Builders is not supported.")
|
|
|
|
kw['_depth'] = 1
|
|
|
|
return SCons.SConf.SConf(*args, **kw)
|
|
|
|
|
|
|
|
# It's very important that the DefaultEnvironmentCall() class stay in this
|
|
|
|
# file, with the get_calling_namespaces() function, the compute_exports()
|
|
|
|
# function, the Frame class and the SConsEnvironment.Export() method.
|
|
|
|
# These things make up the calling stack leading up to the actual global
|
|
|
|
# Export() or SConscript() call that the user issued. We want to allow
|
|
|
|
# users to export local variables that they define, like so:
|
|
|
|
#
|
|
|
|
# def func():
|
|
|
|
# x = 1
|
|
|
|
# Export('x')
|
|
|
|
#
|
|
|
|
# To support this, the get_calling_namespaces() function assumes that
|
|
|
|
# the *first* stack frame that's not from this file is the local frame
|
|
|
|
# for the Export() or SConscript() call.
|
|
|
|
|
|
|
|
_DefaultEnvironmentProxy = None
|
|
|
|
|
|
|
|
def get_DefaultEnvironmentProxy():
|
|
|
|
global _DefaultEnvironmentProxy
|
|
|
|
if not _DefaultEnvironmentProxy:
|
|
|
|
default_env = SCons.Defaults.DefaultEnvironment()
|
|
|
|
_DefaultEnvironmentProxy = SCons.Environment.NoSubstitutionProxy(default_env)
|
|
|
|
return _DefaultEnvironmentProxy
|
|
|
|
|
|
|
|
class DefaultEnvironmentCall:
|
|
|
|
"""A class that implements "global function" calls of
|
|
|
|
Environment methods by fetching the specified method from the
|
|
|
|
DefaultEnvironment's class. Note that this uses an intermediate
|
|
|
|
proxy class instead of calling the DefaultEnvironment method
|
|
|
|
directly so that the proxy can override the subst() method and
|
|
|
|
thereby prevent expansion of construction variables (since from
|
|
|
|
the user's point of view this was called as a global function,
|
|
|
|
with no associated construction environment)."""
|
2024-04-15 15:35:35 +02:00
|
|
|
def __init__(self, method_name, subst: int=0) -> None:
|
2023-06-11 12:31:23 +02:00
|
|
|
self.method_name = method_name
|
|
|
|
if subst:
|
|
|
|
self.factory = SCons.Defaults.DefaultEnvironment
|
|
|
|
else:
|
|
|
|
self.factory = get_DefaultEnvironmentProxy
|
|
|
|
def __call__(self, *args, **kw):
|
|
|
|
env = self.factory()
|
|
|
|
method = getattr(env, self.method_name)
|
|
|
|
return method(*args, **kw)
|
|
|
|
|
|
|
|
|
|
|
|
def BuildDefaultGlobals():
|
|
|
|
"""
|
|
|
|
Create a dictionary containing all the default globals for
|
|
|
|
SConstruct and SConscript files.
|
|
|
|
"""
|
|
|
|
|
|
|
|
global GlobalDict
|
|
|
|
if GlobalDict is None:
|
|
|
|
GlobalDict = {}
|
|
|
|
|
|
|
|
import SCons.Script
|
|
|
|
d = SCons.Script.__dict__
|
2024-04-15 15:35:35 +02:00
|
|
|
def not_a_module(m, d=d, mtype=type(SCons.Script)) -> bool:
|
2023-06-11 12:31:23 +02:00
|
|
|
return not isinstance(d[m], mtype)
|
|
|
|
for m in filter(not_a_module, dir(SCons.Script)):
|
|
|
|
GlobalDict[m] = d[m]
|
|
|
|
|
|
|
|
return GlobalDict.copy()
|
|
|
|
|
|
|
|
# Local Variables:
|
|
|
|
# tab-width:4
|
|
|
|
# indent-tabs-mode:nil
|
|
|
|
# End:
|
|
|
|
# vim: set expandtab tabstop=4 shiftwidth=4:
|