blob: 4cd1e96990d981059accb202142056951327a70e [file] [log] [blame]
Guido van Rossum00ff4331994-10-03 16:33:08 +00001import marshal
Guido van Rossumbaf06031998-08-25 14:06:55 +00002import bkfile
Guido van Rossum00ff4331994-10-03 16:33:08 +00003
4
5# Write a file containing frozen code for the modules in the dictionary.
6
7header = """
Guido van Rossumf75f80e1996-06-17 17:48:30 +00008#include "Python.h"
9
10static struct _frozen _PyImport_FrozenModules[] = {
Guido van Rossum00ff4331994-10-03 16:33:08 +000011"""
12trailer = """\
Guido van Rossum912a14c1998-03-05 04:56:37 +000013 {0, 0, 0} /* sentinel */
Guido van Rossum00ff4331994-10-03 16:33:08 +000014};
Guido van Rossum78fc3631998-03-20 17:37:24 +000015"""
Guido van Rossumf75f80e1996-06-17 17:48:30 +000016
Guido van Rossum1abfb9c1998-04-23 14:39:05 +000017# if __debug__ == 0 (i.e. -O option given), set Py_OptimizeFlag in frozen app.
Guido van Rossum78fc3631998-03-20 17:37:24 +000018default_entry_point = """
Guido van Rossumb3afce51997-07-19 21:58:30 +000019int
Marc-André Lemburg64b4f272002-04-04 16:15:41 +000020main(int argc, char **argv)
Guido van Rossumb3afce51997-07-19 21:58:30 +000021{
Tim Peters182b5ac2004-07-18 06:16:08 +000022 extern int Py_FrozenMain(int, char **);
Guido van Rossum1abfb9c1998-04-23 14:39:05 +000023""" + ((not __debug__ and """
24 Py_OptimizeFlag++;
25""") or "") + """
Guido van Rossum912a14c1998-03-05 04:56:37 +000026 PyImport_FrozenModules = _PyImport_FrozenModules;
27 return Py_FrozenMain(argc, argv);
Guido van Rossumb3afce51997-07-19 21:58:30 +000028}
29
Guido van Rossum00ff4331994-10-03 16:33:08 +000030"""
31
Guido van Rossum03f7f082001-10-18 19:15:32 +000032def makefreeze(base, dict, debug=0, entry_point=None, fail_import=()):
Guido van Rossum78fc3631998-03-20 17:37:24 +000033 if entry_point is None: entry_point = default_entry_point
Guido van Rossum912a14c1998-03-05 04:56:37 +000034 done = []
Guido van Rossumbaf06031998-08-25 14:06:55 +000035 files = []
Guido van Rossum53970392007-06-12 00:28:30 +000036 mods = sorted(dict.keys())
Guido van Rossum912a14c1998-03-05 04:56:37 +000037 for mod in mods:
38 m = dict[mod]
Walter Dörwaldaaab30e2002-09-11 20:36:02 +000039 mangled = "__".join(mod.split("."))
Guido van Rossum912a14c1998-03-05 04:56:37 +000040 if m.__code__:
Guido van Rossumbaf06031998-08-25 14:06:55 +000041 file = 'M_' + mangled + '.c'
42 outfp = bkfile.open(base + file, 'w')
43 files.append(file)
Guido van Rossum912a14c1998-03-05 04:56:37 +000044 if debug:
Guido van Rossum96bf7e82007-02-09 23:27:01 +000045 print("freezing", mod, "...")
Guido van Rossum912a14c1998-03-05 04:56:37 +000046 str = marshal.dumps(m.__code__)
47 size = len(str)
48 if m.__path__:
49 # Indicate package by negative size
50 size = -size
51 done.append((mod, mangled, size))
52 writecode(outfp, mangled, str)
Guido van Rossumbaf06031998-08-25 14:06:55 +000053 outfp.close()
Guido van Rossum912a14c1998-03-05 04:56:37 +000054 if debug:
Guido van Rossum96bf7e82007-02-09 23:27:01 +000055 print("generating table of frozen modules")
Guido van Rossumbaf06031998-08-25 14:06:55 +000056 outfp = bkfile.open(base + 'frozen.c', 'w')
57 for mod, mangled, size in done:
58 outfp.write('extern unsigned char M_%s[];\n' % mangled)
Guido van Rossum912a14c1998-03-05 04:56:37 +000059 outfp.write(header)
60 for mod, mangled, size in done:
61 outfp.write('\t{"%s", M_%s, %d},\n' % (mod, mangled, size))
Guido van Rossum03f7f082001-10-18 19:15:32 +000062 outfp.write('\n')
63 # The following modules have a NULL code pointer, indicating
64 # that the prozen program should not search for them on the host
65 # system. Importing them will *always* raise an ImportError.
66 # The zero value size is never used.
67 for mod in fail_import:
68 outfp.write('\t{"%s", NULL, 0},\n' % (mod,))
Guido van Rossum912a14c1998-03-05 04:56:37 +000069 outfp.write(trailer)
Guido van Rossum78fc3631998-03-20 17:37:24 +000070 outfp.write(entry_point)
Guido van Rossumbaf06031998-08-25 14:06:55 +000071 outfp.close()
72 return files
Guido van Rossum78fc3631998-03-20 17:37:24 +000073
Guido van Rossum00ff4331994-10-03 16:33:08 +000074
75
Guido van Rossum00ff4331994-10-03 16:33:08 +000076# Write a C initializer for a module containing the frozen python code.
77# The array is called M_<mod>.
78
79def writecode(outfp, mod, str):
Guido van Rossumbaf06031998-08-25 14:06:55 +000080 outfp.write('unsigned char M_%s[] = {' % mod)
Guido van Rossum912a14c1998-03-05 04:56:37 +000081 for i in range(0, len(str), 16):
82 outfp.write('\n\t')
Guido van Rossum53970392007-06-12 00:28:30 +000083 for c in bytes(str[i:i+16]):
84 outfp.write('%d,' % c)
Guido van Rossum912a14c1998-03-05 04:56:37 +000085 outfp.write('\n};\n')
Guido van Rossumbaf06031998-08-25 14:06:55 +000086
87## def writecode(outfp, mod, str):
88## outfp.write('unsigned char M_%s[%d] = "%s";\n' % (mod, len(str),
Walter Dörwald70a6b492004-02-12 17:35:32 +000089## '\\"'.join(map(lambda s: repr(s)[1:-1], str.split('"')))))