Damien George | c7271a8 | 2022-06-08 14:47:21 +1000 | [diff] [blame] | 1 | """ |
| 2 | This pre-processor parses a single file containing a list of |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 3 | `MP_REGISTER_MODULE(MP_QSTR_module_name, obj_module)` or |
| 4 | `MP_REGISTER_EXTENSIBLE_MODULE(MP_QSTR_module_name, obj_module)` |
| 5 | (i.e. the output of `py/makeqstrdefs.py cat module`). |
| 6 | |
| 7 | The output is a header (typically moduledefs.h) which is included by |
| 8 | py/objmodule.c that contains entries to be included in the definition of |
| 9 | - mp_rom_map_elem_t mp_builtin_module_table[] |
| 10 | - mp_rom_map_elem_t mp_builtin_extensible_module_table[] |
| 11 | |
| 12 | Extensible modules are modules that can be overridden from the filesystem, see |
| 13 | py/builtinimnport.c:process_import_at_level. Regular modules will always use |
| 14 | the built-in version. |
Damien George | c7271a8 | 2022-06-08 14:47:21 +1000 | [diff] [blame] | 15 | """ |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 16 | |
| 17 | from __future__ import print_function |
| 18 | |
Phil Howard | 37d5114 | 2022-06-14 11:17:03 +0100 | [diff] [blame] | 19 | import sys |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 20 | import re |
Damien George | 673e154 | 2019-04-12 11:34:52 +1000 | [diff] [blame] | 21 | import io |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 22 | import argparse |
| 23 | |
| 24 | |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 25 | register_pattern = re.compile( |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 26 | r"\s*(MP_REGISTER_MODULE|MP_REGISTER_EXTENSIBLE_MODULE)\(MP_QSTR_(.*?),\s*(.*?)\);", |
| 27 | flags=re.DOTALL, |
| 28 | ) |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 29 | |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 30 | delegation_pattern = re.compile( |
| 31 | r"\s*(?:MP_REGISTER_MODULE_DELEGATION)\((.*?),\s*(.*?)\);", |
| 32 | flags=re.DOTALL, |
| 33 | ) |
| 34 | |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 35 | |
Damien George | 47f6343 | 2022-05-31 17:10:14 +1000 | [diff] [blame] | 36 | def find_module_registrations(filename): |
| 37 | """Find any MP_REGISTER_MODULE definitions in the provided file. |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 38 | |
Damien George | 47f6343 | 2022-05-31 17:10:14 +1000 | [diff] [blame] | 39 | :param str filename: path to file to check |
Damien George | efe23ac | 2022-05-31 22:56:11 +1000 | [diff] [blame] | 40 | :return: List[(module_name, obj_module)] |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 41 | """ |
| 42 | global pattern |
| 43 | |
Damien George | 47f6343 | 2022-05-31 17:10:14 +1000 | [diff] [blame] | 44 | with io.open(filename, encoding="utf-8") as c_file_obj: |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 45 | c = c_file_obj.read() |
| 46 | return set(re.findall(register_pattern, c)), set(re.findall(delegation_pattern, c)) |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 47 | |
| 48 | |
| 49 | def generate_module_table_header(modules): |
Damien George | 0665907 | 2020-08-29 15:14:29 +1000 | [diff] [blame] | 50 | """Generate header with module table entries for builtin modules. |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 51 | |
Damien George | efe23ac | 2022-05-31 22:56:11 +1000 | [diff] [blame] | 52 | :param List[(module_name, obj_module)] modules: module defs |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 53 | :return: None |
| 54 | """ |
| 55 | |
| 56 | # Print header file for all external modules. |
Jim Mussared | 4694501 | 2022-04-20 16:05:44 +1000 | [diff] [blame] | 57 | mod_defs = set() |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 58 | extensible_mod_defs = set() |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 59 | for macro_name, module_name, obj_module in modules: |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 60 | mod_def = "MODULE_DEF_{}".format(module_name.upper()) |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 61 | if macro_name == "MP_REGISTER_MODULE": |
| 62 | mod_defs.add(mod_def) |
| 63 | elif macro_name == "MP_REGISTER_EXTENSIBLE_MODULE": |
| 64 | extensible_mod_defs.add(mod_def) |
Phil Howard | 37d5114 | 2022-06-14 11:17:03 +0100 | [diff] [blame] | 65 | if "," in obj_module: |
| 66 | print( |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 67 | "ERROR: Call to {}({}, {}) should be {}({}, {})\n".format( |
| 68 | macro_name, |
| 69 | module_name, |
| 70 | obj_module, |
| 71 | macro_name, |
| 72 | module_name, |
| 73 | obj_module.split(",")[0], |
Phil Howard | 37d5114 | 2022-06-14 11:17:03 +0100 | [diff] [blame] | 74 | ), |
| 75 | file=sys.stderr, |
| 76 | ) |
| 77 | sys.exit(1) |
Damien George | 69661f3 | 2020-02-27 15:36:53 +1100 | [diff] [blame] | 78 | print( |
| 79 | ( |
Damien George | efe23ac | 2022-05-31 22:56:11 +1000 | [diff] [blame] | 80 | "extern const struct _mp_obj_module_t {obj_module};\n" |
| 81 | "#undef {mod_def}\n" |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 82 | "#define {mod_def} {{ MP_ROM_QSTR(MP_QSTR_{module_name}), MP_ROM_PTR(&{obj_module}) }},\n" |
Damien George | 69661f3 | 2020-02-27 15:36:53 +1100 | [diff] [blame] | 83 | ).format( |
| 84 | module_name=module_name, |
| 85 | obj_module=obj_module, |
Damien George | 69661f3 | 2020-02-27 15:36:53 +1100 | [diff] [blame] | 86 | mod_def=mod_def, |
| 87 | ) |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 88 | ) |
| 89 | |
| 90 | print("\n#define MICROPY_REGISTERED_MODULES \\") |
| 91 | |
Jim Mussared | 4694501 | 2022-04-20 16:05:44 +1000 | [diff] [blame] | 92 | for mod_def in sorted(mod_defs): |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 93 | print(" {mod_def} \\".format(mod_def=mod_def)) |
| 94 | |
| 95 | print("// MICROPY_REGISTERED_MODULES") |
| 96 | |
Jim Mussared | 24c02c4 | 2023-06-02 12:28:07 +1000 | [diff] [blame] | 97 | print("\n#define MICROPY_REGISTERED_EXTENSIBLE_MODULES \\") |
| 98 | |
| 99 | for mod_def in sorted(extensible_mod_defs): |
| 100 | print(" {mod_def} \\".format(mod_def=mod_def)) |
| 101 | |
| 102 | print("// MICROPY_REGISTERED_EXTENSIBLE_MODULES") |
| 103 | |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 104 | |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 105 | def generate_module_delegations(delegations): |
Jim Mussared | e6926d6 | 2023-06-06 22:49:50 +1000 | [diff] [blame] | 106 | if not delegations: |
| 107 | return |
| 108 | |
Damien George | 5ce1a03 | 2023-06-12 13:09:48 +1000 | [diff] [blame] | 109 | print() |
| 110 | for obj_module, fun_name in delegations: |
| 111 | print("extern void {}(mp_obj_t self_in, qstr attr, mp_obj_t *dest);".format(fun_name)) |
| 112 | print("#define MICROPY_MODULE_DELEGATIONS \\") |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 113 | for obj_module, fun_name in delegations: |
| 114 | print( |
Damien George | 44295c9 | 2023-06-12 13:09:29 +1000 | [diff] [blame] | 115 | " {{ MP_ROM_PTR(&{obj_module}), {fun_name} }}, \\".format( |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 116 | obj_module=obj_module, fun_name=fun_name |
| 117 | ) |
| 118 | ) |
| 119 | print("// MICROPY_MODULE_DELEGATIONS") |
| 120 | |
| 121 | |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 122 | def main(): |
| 123 | parser = argparse.ArgumentParser() |
Damien George | 47f6343 | 2022-05-31 17:10:14 +1000 | [diff] [blame] | 124 | parser.add_argument("file", nargs=1, help="file with MP_REGISTER_MODULE definitions") |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 125 | args = parser.parse_args() |
| 126 | |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 127 | print("// Automatically generated by makemoduledefs.py.\n") |
| 128 | |
| 129 | modules, delegations = find_module_registrations(args.file[0]) |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 130 | generate_module_table_header(sorted(modules)) |
Jim Mussared | 13c817e | 2023-06-05 15:52:57 +1000 | [diff] [blame] | 131 | generate_module_delegations(sorted(delegations)) |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 132 | |
| 133 | |
Damien George | 69661f3 | 2020-02-27 15:36:53 +1100 | [diff] [blame] | 134 | if __name__ == "__main__": |
Andrew Leech | cf22f47 | 2019-02-18 14:58:44 +1100 | [diff] [blame] | 135 | main() |