|  | """Freeze modules and regen related files (e.g. Python/frozen.c). | 
|  |  | 
|  | See the notes at the top of Python/frozen.c for more info. | 
|  | """ | 
|  |  | 
|  | from collections import namedtuple | 
|  | import hashlib | 
|  | import os | 
|  | import ntpath | 
|  | import posixpath | 
|  | import argparse | 
|  | from update_file import updating_file_with_tmpfile | 
|  |  | 
|  |  | 
|  | ROOT_DIR = os.path.dirname(os.path.dirname(os.path.dirname(__file__))) | 
|  | ROOT_DIR = os.path.abspath(ROOT_DIR) | 
|  | FROZEN_ONLY = os.path.join(ROOT_DIR, 'Tools', 'freeze', 'flag.py') | 
|  |  | 
|  | STDLIB_DIR = os.path.join(ROOT_DIR, 'Lib') | 
|  | # If FROZEN_MODULES_DIR or DEEPFROZEN_MODULES_DIR is changed then the | 
|  | # .gitattributes and .gitignore files needs to be updated. | 
|  | FROZEN_MODULES_DIR = os.path.join(ROOT_DIR, 'Python', 'frozen_modules') | 
|  | DEEPFROZEN_MODULES_DIR = os.path.join(ROOT_DIR, 'Python', 'deepfreeze') | 
|  |  | 
|  | FROZEN_FILE = os.path.join(ROOT_DIR, 'Python', 'frozen.c') | 
|  | MAKEFILE = os.path.join(ROOT_DIR, 'Makefile.pre.in') | 
|  | PCBUILD_PROJECT = os.path.join(ROOT_DIR, 'PCbuild', '_freeze_module.vcxproj') | 
|  | PCBUILD_FILTERS = os.path.join(ROOT_DIR, 'PCbuild', '_freeze_module.vcxproj.filters') | 
|  | PCBUILD_PYTHONCORE = os.path.join(ROOT_DIR, 'PCbuild', 'pythoncore.vcxproj') | 
|  |  | 
|  |  | 
|  | OS_PATH = 'ntpath' if os.name == 'nt' else 'posixpath' | 
|  |  | 
|  | # These are modules that get frozen. | 
|  | TESTS_SECTION = 'Test module' | 
|  | FROZEN = [ | 
|  | # See parse_frozen_spec() for the format. | 
|  | # In cases where the frozenid is duplicated, the first one is re-used. | 
|  | ('import system', [ | 
|  | # These frozen modules are necessary for bootstrapping | 
|  | # the import system. | 
|  | 'importlib._bootstrap : _frozen_importlib', | 
|  | 'importlib._bootstrap_external : _frozen_importlib_external', | 
|  | # This module is important because some Python builds rely | 
|  | # on a builtin zip file instead of a filesystem. | 
|  | 'zipimport', | 
|  | ]), | 
|  | ('stdlib - startup, without site (python -S)', [ | 
|  | 'abc', | 
|  | 'codecs', | 
|  | # For now we do not freeze the encodings, due # to the noise all | 
|  | # those extra modules add to the text printed during the build. | 
|  | # (See https://github.com/python/cpython/pull/28398#pullrequestreview-756856469.) | 
|  | #'<encodings.*>', | 
|  | 'io', | 
|  | ]), | 
|  | ('stdlib - startup, with site', [ | 
|  | '_collections_abc', | 
|  | '_sitebuiltins', | 
|  | 'genericpath', | 
|  | 'ntpath', | 
|  | 'posixpath', | 
|  | # We must explicitly mark os.path as a frozen module | 
|  | # even though it will never be imported. | 
|  | f'{OS_PATH} : os.path', | 
|  | 'os', | 
|  | 'site', | 
|  | 'stat', | 
|  | ]), | 
|  | ('runpy - run module with -m', [ | 
|  | "importlib.util", | 
|  | "importlib.machinery", | 
|  | "runpy", | 
|  | ]), | 
|  | (TESTS_SECTION, [ | 
|  | '__hello__', | 
|  | '__hello__ : __hello_alias__', | 
|  | '__hello__ : <__phello_alias__>', | 
|  | '__hello__ : __phello_alias__.spam', | 
|  | '<__phello__.**.*>', | 
|  | f'frozen_only : __hello_only__ = {FROZEN_ONLY}', | 
|  | ]), | 
|  | ] | 
|  | BOOTSTRAP = { | 
|  | 'importlib._bootstrap', | 
|  | 'importlib._bootstrap_external', | 
|  | 'zipimport', | 
|  | } | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # platform-specific helpers | 
|  |  | 
|  | if os.path is posixpath: | 
|  | relpath_for_posix_display = os.path.relpath | 
|  |  | 
|  | def relpath_for_windows_display(path, base): | 
|  | return ntpath.relpath( | 
|  | ntpath.join(*path.split(os.path.sep)), | 
|  | ntpath.join(*base.split(os.path.sep)), | 
|  | ) | 
|  |  | 
|  | else: | 
|  | relpath_for_windows_display = ntpath.relpath | 
|  |  | 
|  | def relpath_for_posix_display(path, base): | 
|  | return posixpath.relpath( | 
|  | posixpath.join(*path.split(os.path.sep)), | 
|  | posixpath.join(*base.split(os.path.sep)), | 
|  | ) | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # specs | 
|  |  | 
|  | def parse_frozen_specs(): | 
|  | seen = {} | 
|  | for section, specs in FROZEN: | 
|  | parsed = _parse_specs(specs, section, seen) | 
|  | for item in parsed: | 
|  | frozenid, pyfile, modname, ispkg, section = item | 
|  | try: | 
|  | source = seen[frozenid] | 
|  | except KeyError: | 
|  | source = FrozenSource.from_id(frozenid, pyfile) | 
|  | seen[frozenid] = source | 
|  | else: | 
|  | assert not pyfile or pyfile == source.pyfile, item | 
|  | yield FrozenModule(modname, ispkg, section, source) | 
|  |  | 
|  |  | 
|  | def _parse_specs(specs, section, seen): | 
|  | for spec in specs: | 
|  | info, subs = _parse_spec(spec, seen, section) | 
|  | yield info | 
|  | for info in subs or (): | 
|  | yield info | 
|  |  | 
|  |  | 
|  | def _parse_spec(spec, knownids=None, section=None): | 
|  | """Yield an info tuple for each module corresponding to the given spec. | 
|  |  | 
|  | The info consists of: (frozenid, pyfile, modname, ispkg, section). | 
|  |  | 
|  | Supported formats: | 
|  |  | 
|  | frozenid | 
|  | frozenid : modname | 
|  | frozenid : modname = pyfile | 
|  |  | 
|  | "frozenid" and "modname" must be valid module names (dot-separated | 
|  | identifiers).  If "modname" is not provided then "frozenid" is used. | 
|  | If "pyfile" is not provided then the filename of the module | 
|  | corresponding to "frozenid" is used. | 
|  |  | 
|  | Angle brackets around a frozenid (e.g. '<encodings>") indicate | 
|  | it is a package.  This also means it must be an actual module | 
|  | (i.e. "pyfile" cannot have been provided).  Such values can have | 
|  | patterns to expand submodules: | 
|  |  | 
|  | <encodings.*>    - also freeze all direct submodules | 
|  | <encodings.**.*> - also freeze the full submodule tree | 
|  |  | 
|  | As with "frozenid", angle brackets around "modname" indicate | 
|  | it is a package.  However, in this case "pyfile" should not | 
|  | have been provided and patterns in "modname" are not supported. | 
|  | Also, if "modname" has brackets then "frozenid" should not, | 
|  | and "pyfile" should have been provided.. | 
|  | """ | 
|  | frozenid, _, remainder = spec.partition(':') | 
|  | modname, _, pyfile = remainder.partition('=') | 
|  | frozenid = frozenid.strip() | 
|  | modname = modname.strip() | 
|  | pyfile = pyfile.strip() | 
|  |  | 
|  | submodules = None | 
|  | if modname.startswith('<') and modname.endswith('>'): | 
|  | assert check_modname(frozenid), spec | 
|  | modname = modname[1:-1] | 
|  | assert check_modname(modname), spec | 
|  | if frozenid in knownids: | 
|  | pass | 
|  | elif pyfile: | 
|  | assert not os.path.isdir(pyfile), spec | 
|  | else: | 
|  | pyfile = _resolve_module(frozenid, ispkg=False) | 
|  | ispkg = True | 
|  | elif pyfile: | 
|  | assert check_modname(frozenid), spec | 
|  | assert not knownids or frozenid not in knownids, spec | 
|  | assert check_modname(modname), spec | 
|  | assert not os.path.isdir(pyfile), spec | 
|  | ispkg = False | 
|  | elif knownids and frozenid in knownids: | 
|  | assert check_modname(frozenid), spec | 
|  | assert check_modname(modname), spec | 
|  | ispkg = False | 
|  | else: | 
|  | assert not modname or check_modname(modname), spec | 
|  | resolved = iter(resolve_modules(frozenid)) | 
|  | frozenid, pyfile, ispkg = next(resolved) | 
|  | if not modname: | 
|  | modname = frozenid | 
|  | if ispkg: | 
|  | pkgid = frozenid | 
|  | pkgname = modname | 
|  | pkgfiles = {pyfile: pkgid} | 
|  | def iter_subs(): | 
|  | for frozenid, pyfile, ispkg in resolved: | 
|  | if pkgname: | 
|  | modname = frozenid.replace(pkgid, pkgname, 1) | 
|  | else: | 
|  | modname = frozenid | 
|  | if pyfile: | 
|  | if pyfile in pkgfiles: | 
|  | frozenid = pkgfiles[pyfile] | 
|  | pyfile = None | 
|  | elif ispkg: | 
|  | pkgfiles[pyfile] = frozenid | 
|  | yield frozenid, pyfile, modname, ispkg, section | 
|  | submodules = iter_subs() | 
|  |  | 
|  | info = (frozenid, pyfile or None, modname, ispkg, section) | 
|  | return info, submodules | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # frozen source files | 
|  |  | 
|  | class FrozenSource(namedtuple('FrozenSource', 'id pyfile frozenfile deepfreezefile')): | 
|  |  | 
|  | @classmethod | 
|  | def from_id(cls, frozenid, pyfile=None): | 
|  | if not pyfile: | 
|  | pyfile = os.path.join(STDLIB_DIR, *frozenid.split('.')) + '.py' | 
|  | #assert os.path.exists(pyfile), (frozenid, pyfile) | 
|  | frozenfile = resolve_frozen_file(frozenid, FROZEN_MODULES_DIR) | 
|  | deepfreezefile = resolve_frozen_file(frozenid, DEEPFROZEN_MODULES_DIR) | 
|  | return cls(frozenid, pyfile, frozenfile, deepfreezefile) | 
|  |  | 
|  | @property | 
|  | def frozenid(self): | 
|  | return self.id | 
|  |  | 
|  | @property | 
|  | def modname(self): | 
|  | if self.pyfile.startswith(STDLIB_DIR): | 
|  | return self.id | 
|  | return None | 
|  |  | 
|  | @property | 
|  | def symbol(self): | 
|  | # This matches what we do in Programs/_freeze_module.c: | 
|  | name = self.frozenid.replace('.', '_') | 
|  | return '_Py_M__' + name | 
|  |  | 
|  | @property | 
|  | def ispkg(self): | 
|  | if not self.pyfile: | 
|  | return False | 
|  | elif self.frozenid.endswith('.__init__'): | 
|  | return False | 
|  | else: | 
|  | return os.path.basename(self.pyfile) == '__init__.py' | 
|  |  | 
|  | @property | 
|  | def isbootstrap(self): | 
|  | return self.id in BOOTSTRAP | 
|  |  | 
|  |  | 
|  | def resolve_frozen_file(frozenid, destdir): | 
|  | """Return the filename corresponding to the given frozen ID. | 
|  |  | 
|  | For stdlib modules the ID will always be the full name | 
|  | of the source module. | 
|  | """ | 
|  | if not isinstance(frozenid, str): | 
|  | try: | 
|  | frozenid = frozenid.frozenid | 
|  | except AttributeError: | 
|  | raise ValueError(f'unsupported frozenid {frozenid!r}') | 
|  | # We use a consistent naming convention for all frozen modules. | 
|  | frozenfile = f'{frozenid}.h' | 
|  | if not destdir: | 
|  | return frozenfile | 
|  | return os.path.join(destdir, frozenfile) | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # frozen modules | 
|  |  | 
|  | class FrozenModule(namedtuple('FrozenModule', 'name ispkg section source')): | 
|  |  | 
|  | def __getattr__(self, name): | 
|  | return getattr(self.source, name) | 
|  |  | 
|  | @property | 
|  | def modname(self): | 
|  | return self.name | 
|  |  | 
|  | @property | 
|  | def orig(self): | 
|  | return self.source.modname | 
|  |  | 
|  | @property | 
|  | def isalias(self): | 
|  | orig = self.source.modname | 
|  | if not orig: | 
|  | return True | 
|  | return self.name != orig | 
|  |  | 
|  | def summarize(self): | 
|  | source = self.source.modname | 
|  | if source: | 
|  | source = f'<{source}>' | 
|  | else: | 
|  | source = relpath_for_posix_display(self.pyfile, ROOT_DIR) | 
|  | return { | 
|  | 'module': self.name, | 
|  | 'ispkg': self.ispkg, | 
|  | 'source': source, | 
|  | 'frozen': os.path.basename(self.frozenfile), | 
|  | 'checksum': _get_checksum(self.frozenfile), | 
|  | } | 
|  |  | 
|  |  | 
|  | def _iter_sources(modules): | 
|  | seen = set() | 
|  | for mod in modules: | 
|  | if mod.source not in seen: | 
|  | yield mod.source | 
|  | seen.add(mod.source) | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # generic helpers | 
|  |  | 
|  | def _get_checksum(filename): | 
|  | with open(filename, "rb") as infile: | 
|  | contents = infile.read() | 
|  | m = hashlib.sha256() | 
|  | m.update(contents) | 
|  | return m.hexdigest() | 
|  |  | 
|  |  | 
|  | def resolve_modules(modname, pyfile=None): | 
|  | if modname.startswith('<') and modname.endswith('>'): | 
|  | if pyfile: | 
|  | assert os.path.isdir(pyfile) or os.path.basename(pyfile) == '__init__.py', pyfile | 
|  | ispkg = True | 
|  | modname = modname[1:-1] | 
|  | rawname = modname | 
|  | # For now, we only expect match patterns at the end of the name. | 
|  | _modname, sep, match = modname.rpartition('.') | 
|  | if sep: | 
|  | if _modname.endswith('.**'): | 
|  | modname = _modname[:-3] | 
|  | match = f'**.{match}' | 
|  | elif match and not match.isidentifier(): | 
|  | modname = _modname | 
|  | # Otherwise it's a plain name so we leave it alone. | 
|  | else: | 
|  | match = None | 
|  | else: | 
|  | ispkg = False | 
|  | rawname = modname | 
|  | match = None | 
|  |  | 
|  | if not check_modname(modname): | 
|  | raise ValueError(f'not a valid module name ({rawname})') | 
|  |  | 
|  | if not pyfile: | 
|  | pyfile = _resolve_module(modname, ispkg=ispkg) | 
|  | elif os.path.isdir(pyfile): | 
|  | pyfile = _resolve_module(modname, pyfile, ispkg) | 
|  | yield modname, pyfile, ispkg | 
|  |  | 
|  | if match: | 
|  | pkgdir = os.path.dirname(pyfile) | 
|  | yield from iter_submodules(modname, pkgdir, match) | 
|  |  | 
|  |  | 
|  | def check_modname(modname): | 
|  | return all(n.isidentifier() for n in modname.split('.')) | 
|  |  | 
|  |  | 
|  | def iter_submodules(pkgname, pkgdir=None, match='*'): | 
|  | if not pkgdir: | 
|  | pkgdir = os.path.join(STDLIB_DIR, *pkgname.split('.')) | 
|  | if not match: | 
|  | match = '**.*' | 
|  | match_modname = _resolve_modname_matcher(match, pkgdir) | 
|  |  | 
|  | def _iter_submodules(pkgname, pkgdir): | 
|  | for entry in sorted(os.scandir(pkgdir), key=lambda e: e.name): | 
|  | matched, recursive = match_modname(entry.name) | 
|  | if not matched: | 
|  | continue | 
|  | modname = f'{pkgname}.{entry.name}' | 
|  | if modname.endswith('.py'): | 
|  | yield modname[:-3], entry.path, False | 
|  | elif entry.is_dir(): | 
|  | pyfile = os.path.join(entry.path, '__init__.py') | 
|  | # We ignore namespace packages. | 
|  | if os.path.exists(pyfile): | 
|  | yield modname, pyfile, True | 
|  | if recursive: | 
|  | yield from _iter_submodules(modname, entry.path) | 
|  |  | 
|  | return _iter_submodules(pkgname, pkgdir) | 
|  |  | 
|  |  | 
|  | def _resolve_modname_matcher(match, rootdir=None): | 
|  | if isinstance(match, str): | 
|  | if match.startswith('**.'): | 
|  | recursive = True | 
|  | pat = match[3:] | 
|  | assert match | 
|  | else: | 
|  | recursive = False | 
|  | pat = match | 
|  |  | 
|  | if pat == '*': | 
|  | def match_modname(modname): | 
|  | return True, recursive | 
|  | else: | 
|  | raise NotImplementedError(match) | 
|  | elif callable(match): | 
|  | match_modname = match(rootdir) | 
|  | else: | 
|  | raise ValueError(f'unsupported matcher {match!r}') | 
|  | return match_modname | 
|  |  | 
|  |  | 
|  | def _resolve_module(modname, pathentry=STDLIB_DIR, ispkg=False): | 
|  | assert pathentry, pathentry | 
|  | pathentry = os.path.normpath(pathentry) | 
|  | assert os.path.isabs(pathentry) | 
|  | if ispkg: | 
|  | return os.path.join(pathentry, *modname.split('.'), '__init__.py') | 
|  | return os.path.join(pathentry, *modname.split('.')) + '.py' | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # regenerating dependent files | 
|  |  | 
|  | def find_marker(lines, marker, file): | 
|  | for pos, line in enumerate(lines): | 
|  | if marker in line: | 
|  | return pos | 
|  | raise Exception(f"Can't find {marker!r} in file {file}") | 
|  |  | 
|  |  | 
|  | def replace_block(lines, start_marker, end_marker, replacements, file): | 
|  | start_pos = find_marker(lines, start_marker, file) | 
|  | end_pos = find_marker(lines, end_marker, file) | 
|  | if end_pos <= start_pos: | 
|  | raise Exception(f"End marker {end_marker!r} " | 
|  | f"occurs before start marker {start_marker!r} " | 
|  | f"in file {file}") | 
|  | replacements = [line.rstrip() + '\n' for line in replacements] | 
|  | return lines[:start_pos + 1] + replacements + lines[end_pos:] | 
|  |  | 
|  |  | 
|  | def regen_frozen(modules, frozen_modules: bool): | 
|  | headerlines = [] | 
|  | parentdir = os.path.dirname(FROZEN_FILE) | 
|  | if frozen_modules: | 
|  | for src in _iter_sources(modules): | 
|  | # Adding a comment to separate sections here doesn't add much, | 
|  | # so we don't. | 
|  | header = relpath_for_posix_display(src.frozenfile, parentdir) | 
|  | headerlines.append(f'#include "{header}"') | 
|  |  | 
|  | externlines = [] | 
|  | bootstraplines = [] | 
|  | stdliblines = [] | 
|  | testlines = [] | 
|  | aliaslines = [] | 
|  | indent = '    ' | 
|  | lastsection = None | 
|  | for mod in modules: | 
|  | if mod.isbootstrap: | 
|  | lines = bootstraplines | 
|  | elif mod.section == TESTS_SECTION: | 
|  | lines = testlines | 
|  | else: | 
|  | lines = stdliblines | 
|  | if mod.section != lastsection: | 
|  | if lastsection is not None: | 
|  | lines.append('') | 
|  | lines.append(f'/* {mod.section} */') | 
|  | lastsection = mod.section | 
|  |  | 
|  | # Also add a extern declaration for the corresponding | 
|  | # deepfreeze-generated function. | 
|  | orig_name = mod.source.id | 
|  | code_name = orig_name.replace(".", "_") | 
|  | get_code_name = "_Py_get_%s_toplevel" % code_name | 
|  | externlines.append("extern PyObject *%s(void);" % get_code_name) | 
|  |  | 
|  | symbol = mod.symbol | 
|  | pkg = 'true' if mod.ispkg else 'false' | 
|  | if not frozen_modules: | 
|  | line = ('{"%s", NULL, 0, %s, GET_CODE(%s)},' | 
|  | ) % (mod.name, pkg, code_name) | 
|  | else: | 
|  | line = ('{"%s", %s, (int)sizeof(%s), %s, GET_CODE(%s)},' | 
|  | ) % (mod.name, symbol, symbol, pkg, code_name) | 
|  | lines.append(line) | 
|  |  | 
|  | if mod.isalias: | 
|  | if not mod.orig: | 
|  | entry = '{"%s", NULL},' % (mod.name,) | 
|  | elif mod.source.ispkg: | 
|  | entry = '{"%s", "<%s"},' % (mod.name, mod.orig) | 
|  | else: | 
|  | entry = '{"%s", "%s"},' % (mod.name, mod.orig) | 
|  | aliaslines.append(indent + entry) | 
|  |  | 
|  | for lines in (bootstraplines, stdliblines, testlines): | 
|  | # TODO: Is this necessary any more? | 
|  | if not lines[0]: | 
|  | del lines[0] | 
|  | for i, line in enumerate(lines): | 
|  | if line: | 
|  | lines[i] = indent + line | 
|  |  | 
|  | print(f'# Updating {os.path.relpath(FROZEN_FILE)}') | 
|  | with updating_file_with_tmpfile(FROZEN_FILE) as (infile, outfile): | 
|  | lines = infile.readlines() | 
|  | # TODO: Use more obvious markers, e.g. | 
|  | # $START GENERATED FOOBAR$ / $END GENERATED FOOBAR$ | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "/* Includes for frozen modules: */", | 
|  | "/* End includes */", | 
|  | headerlines, | 
|  | FROZEN_FILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "/* Start extern declarations */", | 
|  | "/* End extern declarations */", | 
|  | externlines, | 
|  | FROZEN_FILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "static const struct _frozen bootstrap_modules[] =", | 
|  | "/* bootstrap sentinel */", | 
|  | bootstraplines, | 
|  | FROZEN_FILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "static const struct _frozen stdlib_modules[] =", | 
|  | "/* stdlib sentinel */", | 
|  | stdliblines, | 
|  | FROZEN_FILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "static const struct _frozen test_modules[] =", | 
|  | "/* test sentinel */", | 
|  | testlines, | 
|  | FROZEN_FILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "const struct _module_alias aliases[] =", | 
|  | "/* aliases sentinel */", | 
|  | aliaslines, | 
|  | FROZEN_FILE, | 
|  | ) | 
|  | outfile.writelines(lines) | 
|  |  | 
|  |  | 
|  | def regen_makefile(modules): | 
|  | pyfiles = [] | 
|  | frozenfiles = [] | 
|  | rules = [''] | 
|  | deepfreezerules = ["Python/deepfreeze/deepfreeze.c: $(DEEPFREEZE_DEPS)", | 
|  | "\t$(PYTHON_FOR_FREEZE) $(srcdir)/Tools/scripts/deepfreeze.py \\"] | 
|  | for src in _iter_sources(modules): | 
|  | frozen_header = relpath_for_posix_display(src.frozenfile, ROOT_DIR) | 
|  | frozenfiles.append(f'\t\t{frozen_header} \\') | 
|  |  | 
|  | pyfile = relpath_for_posix_display(src.pyfile, ROOT_DIR) | 
|  | pyfiles.append(f'\t\t{pyfile} \\') | 
|  |  | 
|  | if src.isbootstrap: | 
|  | freezecmd = '$(FREEZE_MODULE_BOOTSTRAP)' | 
|  | freezedep = '$(FREEZE_MODULE_BOOTSTRAP_DEPS)' | 
|  | else: | 
|  | freezecmd = '$(FREEZE_MODULE)' | 
|  | freezedep = '$(FREEZE_MODULE_DEPS)' | 
|  |  | 
|  | freeze = (f'{freezecmd} {src.frozenid} ' | 
|  | f'$(srcdir)/{pyfile} {frozen_header}') | 
|  | rules.extend([ | 
|  | f'{frozen_header}: {pyfile} {freezedep}', | 
|  | f'\t{freeze}', | 
|  | '', | 
|  | ]) | 
|  | deepfreezerules.append(f"\t{frozen_header}:{src.frozenid} \\") | 
|  | deepfreezerules.append('\t-o Python/deepfreeze/deepfreeze.c') | 
|  | pyfiles[-1] = pyfiles[-1].rstrip(" \\") | 
|  | frozenfiles[-1] = frozenfiles[-1].rstrip(" \\") | 
|  |  | 
|  | print(f'# Updating {os.path.relpath(MAKEFILE)}') | 
|  | with updating_file_with_tmpfile(MAKEFILE) as (infile, outfile): | 
|  | lines = infile.readlines() | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "FROZEN_FILES_IN =", | 
|  | "# End FROZEN_FILES_IN", | 
|  | pyfiles, | 
|  | MAKEFILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "FROZEN_FILES_OUT =", | 
|  | "# End FROZEN_FILES_OUT", | 
|  | frozenfiles, | 
|  | MAKEFILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "# BEGIN: freezing modules", | 
|  | "# END: freezing modules", | 
|  | rules, | 
|  | MAKEFILE, | 
|  | ) | 
|  | lines = replace_block( | 
|  | lines, | 
|  | "# BEGIN: deepfreeze modules", | 
|  | "# END: deepfreeze modules", | 
|  | deepfreezerules, | 
|  | MAKEFILE, | 
|  | ) | 
|  | outfile.writelines(lines) | 
|  |  | 
|  |  | 
|  | def regen_pcbuild(modules): | 
|  | projlines = [] | 
|  | filterlines = [] | 
|  | corelines = [] | 
|  | deepfreezerules = ['\t<Exec Command=\'$(PythonForBuild) "$(PySourcePath)Tools\\scripts\\deepfreeze.py" ^'] | 
|  | for src in _iter_sources(modules): | 
|  | pyfile = relpath_for_windows_display(src.pyfile, ROOT_DIR) | 
|  | header = relpath_for_windows_display(src.frozenfile, ROOT_DIR) | 
|  | intfile = ntpath.splitext(ntpath.basename(header))[0] + '.g.h' | 
|  | projlines.append(f'    <None Include="..\\{pyfile}">') | 
|  | projlines.append(f'      <ModName>{src.frozenid}</ModName>') | 
|  | projlines.append(f'      <IntFile>$(IntDir){intfile}</IntFile>') | 
|  | projlines.append(f'      <OutFile>$(PySourcePath){header}</OutFile>') | 
|  | projlines.append(f'    </None>') | 
|  |  | 
|  | filterlines.append(f'    <None Include="..\\{pyfile}">') | 
|  | filterlines.append('      <Filter>Python Files</Filter>') | 
|  | filterlines.append('    </None>') | 
|  | deepfreezerules.append(f'\t\t "$(PySourcePath){header}:{src.frozenid}" ^') | 
|  | deepfreezerules.append('\t\t "-o" "$(PySourcePath)Python\\deepfreeze\\deepfreeze.c"\'/>' ) | 
|  |  | 
|  | corelines.append(f'    <ClCompile Include="..\\Python\\deepfreeze\\deepfreeze.c" />') | 
|  |  | 
|  | print(f'# Updating {os.path.relpath(PCBUILD_PROJECT)}') | 
|  | with updating_file_with_tmpfile(PCBUILD_PROJECT) as (infile, outfile): | 
|  | lines = infile.readlines() | 
|  | lines = replace_block( | 
|  | lines, | 
|  | '<!-- BEGIN frozen modules -->', | 
|  | '<!-- END frozen modules -->', | 
|  | projlines, | 
|  | PCBUILD_PROJECT, | 
|  | ) | 
|  | outfile.writelines(lines) | 
|  | with updating_file_with_tmpfile(PCBUILD_PROJECT) as (infile, outfile): | 
|  | lines = infile.readlines() | 
|  | lines = replace_block( | 
|  | lines, | 
|  | '<!-- BEGIN deepfreeze rule -->', | 
|  | '<!-- END deepfreeze rule -->', | 
|  | deepfreezerules, | 
|  | PCBUILD_PROJECT, | 
|  | ) | 
|  | outfile.writelines(lines) | 
|  | print(f'# Updating {os.path.relpath(PCBUILD_FILTERS)}') | 
|  | with updating_file_with_tmpfile(PCBUILD_FILTERS) as (infile, outfile): | 
|  | lines = infile.readlines() | 
|  | lines = replace_block( | 
|  | lines, | 
|  | '<!-- BEGIN frozen modules -->', | 
|  | '<!-- END frozen modules -->', | 
|  | filterlines, | 
|  | PCBUILD_FILTERS, | 
|  | ) | 
|  | outfile.writelines(lines) | 
|  | print(f'# Updating {os.path.relpath(PCBUILD_PYTHONCORE)}') | 
|  | with updating_file_with_tmpfile(PCBUILD_PYTHONCORE) as (infile, outfile): | 
|  | lines = infile.readlines() | 
|  | lines = replace_block( | 
|  | lines, | 
|  | '<!-- BEGIN deepfreeze -->', | 
|  | '<!-- END deepfreeze -->', | 
|  | corelines, | 
|  | PCBUILD_FILTERS, | 
|  | ) | 
|  | outfile.writelines(lines) | 
|  |  | 
|  |  | 
|  | ####################################### | 
|  | # the script | 
|  |  | 
|  | parser = argparse.ArgumentParser() | 
|  | parser.add_argument("--frozen-modules", action="store_true", | 
|  | help="Use both frozen and deepfrozen modules. (default: uses only deepfrozen modules)") | 
|  |  | 
|  | def main(): | 
|  | args = parser.parse_args() | 
|  | frozen_modules: bool = args.frozen_modules | 
|  | # Expand the raw specs, preserving order. | 
|  | modules = list(parse_frozen_specs()) | 
|  |  | 
|  | # Regen build-related files. | 
|  | regen_makefile(modules) | 
|  | regen_pcbuild(modules) | 
|  | regen_frozen(modules, frozen_modules) | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | main() |