mirror of
https://github.com/python/cpython.git
synced 2024-12-01 13:55:45 +08:00
53c3fb186a
compatible with regular files (in particular it now supports the context management protocol).
88 lines
2.7 KiB
Python
88 lines
2.7 KiB
Python
import marshal
|
|
import bkfile
|
|
|
|
|
|
# Write a file containing frozen code for the modules in the dictionary.
|
|
|
|
header = """
|
|
#include "Python.h"
|
|
|
|
static struct _frozen _PyImport_FrozenModules[] = {
|
|
"""
|
|
trailer = """\
|
|
{0, 0, 0} /* sentinel */
|
|
};
|
|
"""
|
|
|
|
# if __debug__ == 0 (i.e. -O option given), set Py_OptimizeFlag in frozen app.
|
|
default_entry_point = """
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
extern int Py_FrozenMain(int, char **);
|
|
""" + ((not __debug__ and """
|
|
Py_OptimizeFlag++;
|
|
""") or "") + """
|
|
PyImport_FrozenModules = _PyImport_FrozenModules;
|
|
return Py_FrozenMain(argc, argv);
|
|
}
|
|
|
|
"""
|
|
|
|
def makefreeze(base, dict, debug=0, entry_point=None, fail_import=()):
|
|
if entry_point is None: entry_point = default_entry_point
|
|
done = []
|
|
files = []
|
|
mods = sorted(dict.keys())
|
|
for mod in mods:
|
|
m = dict[mod]
|
|
mangled = "__".join(mod.split("."))
|
|
if m.__code__:
|
|
file = 'M_' + mangled + '.c'
|
|
with bkfile.open(base + file, 'w') as outfp:
|
|
files.append(file)
|
|
if debug:
|
|
print("freezing", mod, "...")
|
|
str = marshal.dumps(m.__code__)
|
|
size = len(str)
|
|
if m.__path__:
|
|
# Indicate package by negative size
|
|
size = -size
|
|
done.append((mod, mangled, size))
|
|
writecode(outfp, mangled, str)
|
|
if debug:
|
|
print("generating table of frozen modules")
|
|
with bkfile.open(base + 'frozen.c', 'w') as outfp:
|
|
for mod, mangled, size in done:
|
|
outfp.write('extern unsigned char M_%s[];\n' % mangled)
|
|
outfp.write(header)
|
|
for mod, mangled, size in done:
|
|
outfp.write('\t{"%s", M_%s, %d},\n' % (mod, mangled, size))
|
|
outfp.write('\n')
|
|
# The following modules have a NULL code pointer, indicating
|
|
# that the frozen program should not search for them on the host
|
|
# system. Importing them will *always* raise an ImportError.
|
|
# The zero value size is never used.
|
|
for mod in fail_import:
|
|
outfp.write('\t{"%s", NULL, 0},\n' % (mod,))
|
|
outfp.write(trailer)
|
|
outfp.write(entry_point)
|
|
return files
|
|
|
|
|
|
|
|
# Write a C initializer for a module containing the frozen python code.
|
|
# The array is called M_<mod>.
|
|
|
|
def writecode(outfp, mod, str):
|
|
outfp.write('unsigned char M_%s[] = {' % mod)
|
|
for i in range(0, len(str), 16):
|
|
outfp.write('\n\t')
|
|
for c in bytes(str[i:i+16]):
|
|
outfp.write('%d,' % c)
|
|
outfp.write('\n};\n')
|
|
|
|
## def writecode(outfp, mod, str):
|
|
## outfp.write('unsigned char M_%s[%d] = "%s";\n' % (mod, len(str),
|
|
## '\\"'.join(map(lambda s: repr(s)[1:-1], str.split('"')))))
|