| enum { |
| ID_INFO_KIND_GLOBAL_IMPLICIT, |
| ID_INFO_KIND_GLOBAL_EXPLICIT, |
| ID_INFO_KIND_LOCAL, // in a function f, written and only referenced by f |
| ID_INFO_KIND_CELL, // in a function f, read/written by children of f |
| ID_INFO_KIND_FREE, // in a function f, belongs to the parent of f |
| }; |
| |
| typedef struct _id_info_t { |
| // TODO compress this info to make structure smaller in memory |
| bool param; |
| int kind; |
| qstr qstr; |
| |
| // when it's an ID_INFO_KIND_LOCAL this is the unique number of the local |
| // whet it's an ID_INFO_KIND_CELL/FREE this is the unique number of the closed over variable |
| int local_num; |
| } id_info_t; |
| |
| // scope is a "block" in Python parlance |
| typedef enum { SCOPE_MODULE, SCOPE_FUNCTION, SCOPE_LAMBDA, SCOPE_LIST_COMP, SCOPE_DICT_COMP, SCOPE_SET_COMP, SCOPE_GEN_EXPR, SCOPE_CLASS } scope_kind_t; |
| typedef struct _scope_t { |
| scope_kind_t kind; |
| struct _scope_t *parent; |
| struct _scope_t *next; |
| mp_parse_node_t pn; |
| qstr source_file; |
| qstr simple_name; |
| int id_info_alloc; |
| int id_info_len; |
| id_info_t *id_info; |
| uint scope_flags; // see runtime0.h |
| int num_params; |
| /* not needed |
| int num_default_params; |
| int num_dict_params; |
| */ |
| int num_locals; |
| int stack_size; |
| uint unique_code_id; |
| uint emit_options; |
| } scope_t; |
| |
| scope_t *scope_new(scope_kind_t kind, mp_parse_node_t pn, qstr source_file, uint unique_code_id, uint emit_options); |
| void scope_free(scope_t *scope); |
| id_info_t *scope_find_or_add_id(scope_t *scope, qstr qstr, bool *added); |
| id_info_t *scope_find(scope_t *scope, qstr qstr); |
| id_info_t *scope_find_global(scope_t *scope, qstr qstr); |
| id_info_t *scope_find_local_in_parent(scope_t *scope, qstr qstr); |
| void scope_close_over_in_parents(scope_t *scope, qstr qstr); |
| void scope_declare_global(scope_t *scope, qstr qstr); |
| void scope_declare_nonlocal(scope_t *scope, qstr qstr); |
| void scope_print_info(scope_t *s); |