#ifndef Py_INTERNAL_CODE_H #define Py_INTERNAL_CODE_H #ifdef __cplusplus extern "C" { #endif typedef struct { PyObject *ptr; /* Cached pointer (borrowed reference) */ uint64_t globals_ver; /* ma_version of global dict */ uint64_t builtins_ver; /* ma_version of builtin dict */ } _PyOpcache_LoadGlobal; typedef struct { PyTypeObject *type; Py_ssize_t hint; unsigned int tp_version_tag; } _PyOpCodeOpt_LoadAttr; struct _PyOpcache { union { _PyOpcache_LoadGlobal lg; _PyOpCodeOpt_LoadAttr la; } u; char optimized; }; /* "Locals plus" for a code object is the set of locals + cell vars + * free vars. This relates to variable names as well as offsets into * the "fast locals" storage array of execution frames. The compiler * builds the list of names, their offsets, and the corresponding * kind of local. * * Those kinds represent the source of the initial value and the * variable's scope (as related to closures). A "local" is an * argument or other variable defined in the current scope. A "free" * variable is one that is defined in an outer scope and comes from * the function's closure. A "cell" variable is a local that escapes * into an inner function as part of a closure, and thus must be * wrapped in a cell. Any "local" can also be a "cell", but the * "free" kind is mutually exclusive with both. */ // We would use an enum if C let us specify the storage type. typedef unsigned char _PyLocalsPlusKind; /* Note that these all fit within _PyLocalsPlusKind, as do combinations. */ // Later, we will use the smaller numbers to differentiate the different // kinds of locals (e.g. pos-only arg, varkwargs, local-only). #define CO_FAST_LOCAL 0x20 #define CO_FAST_CELL 0x40 #define CO_FAST_FREE 0x80 typedef _PyLocalsPlusKind *_PyLocalsPlusKinds; static inline int _PyCode_InitLocalsPlusKinds(int num, _PyLocalsPlusKinds *pkinds) { if (num == 0) { *pkinds = NULL; return 0; } _PyLocalsPlusKinds kinds = PyMem_NEW(_PyLocalsPlusKind, num); if (kinds == NULL) { PyErr_NoMemory(); return -1; } *pkinds = kinds; return 0; } static inline void _PyCode_ClearLocalsPlusKinds(_PyLocalsPlusKinds kinds) { if (kinds != NULL) { PyMem_Free(kinds); } } struct _PyCodeConstructor { /* metadata */ PyObject *filename; PyObject *name; int flags; /* the code */ PyObject *code; int firstlineno; PyObject *linetable; /* used by the code */ PyObject *consts; PyObject *names; /* mapping frame offsets to information */ PyObject *localsplusnames; _PyLocalsPlusKinds localspluskinds; /* args (within varnames) */ int argcount; int posonlyargcount; // XXX Replace argcount with posorkwargcount (argcount - posonlyargcount). int kwonlyargcount; /* needed to create the frame */ int stacksize; /* used by the eval loop */ PyObject *exceptiontable; }; // Using an "arguments struct" like this is helpful for maintainability // in a case such as this with many parameters. It does bear a risk: // if the struct changes and callers are not updated properly then the // compiler will not catch problems (like a missing argument). This can // cause hard-to-debug problems. The risk is mitigated by the use of // check_code() in codeobject.c. However, we may decide to switch // back to a regular function signature. Regardless, this approach // wouldn't be appropriate if this weren't a strictly internal API. // (See the comments in https://github.com/python/cpython/pull/26258.) PyAPI_FUNC(int) _PyCode_Validate(struct _PyCodeConstructor *); PyAPI_FUNC(PyCodeObject *) _PyCode_New(struct _PyCodeConstructor *); /* Private API */ int _PyCode_InitOpcache(PyCodeObject *co); /* Getters for internal PyCodeObject data. */ PyAPI_FUNC(PyObject *) _PyCode_GetVarnames(PyCodeObject *); PyAPI_FUNC(PyObject *) _PyCode_GetCellvars(PyCodeObject *); PyAPI_FUNC(PyObject *) _PyCode_GetFreevars(PyCodeObject *); #ifdef __cplusplus } #endif #endif /* !Py_INTERNAL_CODE_H */