mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 05:31:20 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			361 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			361 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef Py_INTERNAL_RUNTIME_H
 | |
| #define Py_INTERNAL_RUNTIME_H
 | |
| #ifdef __cplusplus
 | |
| extern "C" {
 | |
| #endif
 | |
| 
 | |
| #ifndef Py_BUILD_CORE
 | |
| #  error "this header requires Py_BUILD_CORE define"
 | |
| #endif
 | |
| 
 | |
| #include "pycore_atexit.h"          // struct _atexit_runtime_state
 | |
| #include "pycore_ceval_state.h"     // struct _ceval_runtime_state
 | |
| #include "pycore_crossinterp.h"   // struct _xidregistry
 | |
| #include "pycore_faulthandler.h"    // struct _faulthandler_runtime_state
 | |
| #include "pycore_floatobject.h"     // struct _Py_float_runtime_state
 | |
| #include "pycore_import.h"          // struct _import_runtime_state
 | |
| #include "pycore_interp.h"          // PyInterpreterState
 | |
| #include "pycore_object_state.h"    // struct _py_object_runtime_state
 | |
| #include "pycore_parser.h"          // struct _parser_runtime_state
 | |
| #include "pycore_pyhash.h"          // struct pyhash_runtime_state
 | |
| #include "pycore_pymem.h"           // struct _pymem_allocators
 | |
| #include "pycore_pythread.h"        // struct _pythread_runtime_state
 | |
| #include "pycore_signal.h"          // struct _signals_runtime_state
 | |
| #include "pycore_tracemalloc.h"     // struct _tracemalloc_runtime_state
 | |
| #include "pycore_typeobject.h"      // struct _types_runtime_state
 | |
| #include "pycore_unicodeobject.h"   // struct _Py_unicode_runtime_state
 | |
| 
 | |
| struct _getargs_runtime_state {
 | |
|     struct _PyArg_Parser *static_parsers;
 | |
| };
 | |
| 
 | |
| /* GIL state */
 | |
| 
 | |
| struct _gilstate_runtime_state {
 | |
|     /* bpo-26558: Flag to disable PyGILState_Check().
 | |
|        If set to non-zero, PyGILState_Check() always return 1. */
 | |
|     int check_enabled;
 | |
|     /* The single PyInterpreterState used by this process'
 | |
|        GILState implementation
 | |
|     */
 | |
|     /* TODO: Given interp_main, it may be possible to kill this ref */
 | |
|     PyInterpreterState *autoInterpreterState;
 | |
| };
 | |
| 
 | |
| /* Runtime audit hook state */
 | |
| 
 | |
| typedef struct _Py_AuditHookEntry {
 | |
|     struct _Py_AuditHookEntry *next;
 | |
|     Py_AuditHookFunction hookCFunction;
 | |
|     void *userData;
 | |
| } _Py_AuditHookEntry;
 | |
| 
 | |
| typedef struct _Py_DebugOffsets {
 | |
|     char cookie[8];
 | |
|     uint64_t version;
 | |
|     // Runtime state offset;
 | |
|     struct _runtime_state {
 | |
|         uint64_t size;
 | |
|         uint64_t finalizing;
 | |
|         uint64_t interpreters_head;
 | |
|     } runtime_state;
 | |
| 
 | |
|     // Interpreter state offset;
 | |
|     struct _interpreter_state {
 | |
|         uint64_t size;
 | |
|         uint64_t id;
 | |
|         uint64_t next;
 | |
|         uint64_t threads_head;
 | |
|         uint64_t gc;
 | |
|         uint64_t imports_modules;
 | |
|         uint64_t sysdict;
 | |
|         uint64_t builtins;
 | |
|         uint64_t ceval_gil;
 | |
|         uint64_t gil_runtime_state_locked;
 | |
|         uint64_t gil_runtime_state_holder;
 | |
|     } interpreter_state;
 | |
| 
 | |
|     // Thread state offset;
 | |
|     struct _thread_state{
 | |
|         uint64_t size;
 | |
|         uint64_t prev;
 | |
|         uint64_t next;
 | |
|         uint64_t interp;
 | |
|         uint64_t current_frame;
 | |
|         uint64_t thread_id;
 | |
|         uint64_t native_thread_id;
 | |
|         uint64_t datastack_chunk;
 | |
|         uint64_t status;
 | |
|     } thread_state;
 | |
| 
 | |
|     // InterpreterFrame offset;
 | |
|     struct _interpreter_frame {
 | |
|         uint64_t size;
 | |
|         uint64_t previous;
 | |
|         uint64_t executable;
 | |
|         uint64_t instr_ptr;
 | |
|         uint64_t localsplus;
 | |
|         uint64_t owner;
 | |
|     } interpreter_frame;
 | |
| 
 | |
|     // Code object offset;
 | |
|     struct _code_object {
 | |
|         uint64_t size;
 | |
|         uint64_t filename;
 | |
|         uint64_t name;
 | |
|         uint64_t qualname;
 | |
|         uint64_t linetable;
 | |
|         uint64_t firstlineno;
 | |
|         uint64_t argcount;
 | |
|         uint64_t localsplusnames;
 | |
|         uint64_t localspluskinds;
 | |
|         uint64_t co_code_adaptive;
 | |
|     } code_object;
 | |
| 
 | |
|     // PyObject offset;
 | |
|     struct _pyobject {
 | |
|         uint64_t size;
 | |
|         uint64_t ob_type;
 | |
|     } pyobject;
 | |
| 
 | |
|     // PyTypeObject object offset;
 | |
|     struct _type_object {
 | |
|         uint64_t size;
 | |
|         uint64_t tp_name;
 | |
|     } type_object;
 | |
| 
 | |
|     // PyTuple object offset;
 | |
|     struct _tuple_object {
 | |
|         uint64_t size;
 | |
|         uint64_t ob_item;
 | |
|     } tuple_object;
 | |
| 
 | |
|     // Unicode object offset;
 | |
|     struct _unicode_object {
 | |
|         uint64_t size;
 | |
|         uint64_t state;
 | |
|         uint64_t length;
 | |
|         size_t asciiobject_size;
 | |
|     } unicode_object;
 | |
| 
 | |
|     // GC runtime state offset;
 | |
|     struct _gc {
 | |
|         uint64_t size;
 | |
|         uint64_t collecting;
 | |
|     } gc;
 | |
| } _Py_DebugOffsets;
 | |
| 
 | |
| /* Reference tracer state */
 | |
| struct _reftracer_runtime_state {
 | |
|     PyRefTracer tracer_func;
 | |
|     void* tracer_data;
 | |
| };
 | |
| 
 | |
| /* Full Python runtime state */
 | |
| 
 | |
| /* _PyRuntimeState holds the global state for the CPython runtime.
 | |
|    That data is exposed in the internal API as a static variable (_PyRuntime).
 | |
|    */
 | |
| typedef struct pyruntimestate {
 | |
|     /* This field must be first to facilitate locating it by out of process
 | |
|      * debuggers. Out of process debuggers will use the offsets contained in this
 | |
|      * field to be able to locate other fields in several interpreter structures
 | |
|      * in a way that doesn't require them to know the exact layout of those
 | |
|      * structures.
 | |
|      *
 | |
|      * IMPORTANT:
 | |
|      * This struct is **NOT** backwards compatible between minor version of the
 | |
|      * interpreter and the members, order of members and size can change between
 | |
|      * minor versions. This struct is only guaranteed to be stable between patch
 | |
|      * versions for a given minor version of the interpreter.
 | |
|      */
 | |
|     _Py_DebugOffsets debug_offsets;
 | |
| 
 | |
|     /* Has been initialized to a safe state.
 | |
| 
 | |
|        In order to be effective, this must be set to 0 during or right
 | |
|        after allocation. */
 | |
|     int _initialized;
 | |
| 
 | |
|     /* Is running Py_PreInitialize()? */
 | |
|     int preinitializing;
 | |
| 
 | |
|     /* Is Python preinitialized? Set to 1 by Py_PreInitialize() */
 | |
|     int preinitialized;
 | |
| 
 | |
|     /* Is Python core initialized? Set to 1 by _Py_InitializeCore() */
 | |
|     int core_initialized;
 | |
| 
 | |
|     /* Is Python fully initialized? Set to 1 by Py_Initialize() */
 | |
|     int initialized;
 | |
| 
 | |
|     /* Set by Py_FinalizeEx(). Only reset to NULL if Py_Initialize()
 | |
|        is called again.
 | |
| 
 | |
|        Use _PyRuntimeState_GetFinalizing() and _PyRuntimeState_SetFinalizing()
 | |
|        to access it, don't access it directly. */
 | |
|     PyThreadState *_finalizing;
 | |
|     /* The ID of the OS thread in which we are finalizing. */
 | |
|     unsigned long _finalizing_id;
 | |
| 
 | |
|     struct pyinterpreters {
 | |
|         PyMutex mutex;
 | |
|         /* The linked list of interpreters, newest first. */
 | |
|         PyInterpreterState *head;
 | |
|         /* The runtime's initial interpreter, which has a special role
 | |
|            in the operation of the runtime.  It is also often the only
 | |
|            interpreter. */
 | |
|         PyInterpreterState *main;
 | |
|         /* next_id is an auto-numbered sequence of small
 | |
|            integers.  It gets initialized in _PyInterpreterState_Enable(),
 | |
|            which is called in Py_Initialize(), and used in
 | |
|            PyInterpreterState_New().  A negative interpreter ID
 | |
|            indicates an error occurred.  The main interpreter will
 | |
|            always have an ID of 0.  Overflow results in a RuntimeError.
 | |
|            If that becomes a problem later then we can adjust, e.g. by
 | |
|            using a Python int. */
 | |
|         int64_t next_id;
 | |
|     } interpreters;
 | |
| 
 | |
|     /* Platform-specific identifier and PyThreadState, respectively, for the
 | |
|        main thread in the main interpreter. */
 | |
|     unsigned long main_thread;
 | |
|     PyThreadState *main_tstate;
 | |
| 
 | |
|     /* ---------- IMPORTANT ---------------------------
 | |
|      The fields above this line are declared as early as
 | |
|      possible to facilitate out-of-process observability
 | |
|      tools. */
 | |
| 
 | |
|     /* cross-interpreter data and utils */
 | |
|     struct _xi_runtime_state xi;
 | |
| 
 | |
|     struct _pymem_allocators allocators;
 | |
|     struct _obmalloc_global_state obmalloc;
 | |
|     struct pyhash_runtime_state pyhash_state;
 | |
|     struct _pythread_runtime_state threads;
 | |
|     struct _signals_runtime_state signals;
 | |
| 
 | |
|     /* Used for the thread state bound to the current thread. */
 | |
|     Py_tss_t autoTSSkey;
 | |
| 
 | |
|     /* Used instead of PyThreadState.trash when there is not current tstate. */
 | |
|     Py_tss_t trashTSSkey;
 | |
| 
 | |
|     PyWideStringList orig_argv;
 | |
| 
 | |
|     struct _parser_runtime_state parser;
 | |
| 
 | |
|     struct _atexit_runtime_state atexit;
 | |
| 
 | |
|     struct _import_runtime_state imports;
 | |
|     struct _ceval_runtime_state ceval;
 | |
|     struct _gilstate_runtime_state gilstate;
 | |
|     struct _getargs_runtime_state getargs;
 | |
|     struct _fileutils_state fileutils;
 | |
|     struct _faulthandler_runtime_state faulthandler;
 | |
|     struct _tracemalloc_runtime_state tracemalloc;
 | |
|     struct _reftracer_runtime_state ref_tracer;
 | |
| 
 | |
|     // The rwmutex is used to prevent overlapping global and per-interpreter
 | |
|     // stop-the-world events. Global stop-the-world events lock the mutex
 | |
|     // exclusively (as a "writer"), while per-interpreter stop-the-world events
 | |
|     // lock it non-exclusively (as "readers").
 | |
|     _PyRWMutex stoptheworld_mutex;
 | |
|     struct _stoptheworld_state stoptheworld;
 | |
| 
 | |
|     PyPreConfig preconfig;
 | |
| 
 | |
|     // Audit values must be preserved when Py_Initialize()/Py_Finalize()
 | |
|     // is called multiple times.
 | |
|     Py_OpenCodeHookFunction open_code_hook;
 | |
|     void *open_code_userdata;
 | |
|     struct {
 | |
|         PyMutex mutex;
 | |
|         _Py_AuditHookEntry *head;
 | |
|     } audit_hooks;
 | |
| 
 | |
|     struct _py_object_runtime_state object_state;
 | |
|     struct _Py_float_runtime_state float_state;
 | |
|     struct _Py_unicode_runtime_state unicode_state;
 | |
|     struct _types_runtime_state types;
 | |
| 
 | |
|     /* All the objects that are shared by the runtime's interpreters. */
 | |
|     struct _Py_cached_objects cached_objects;
 | |
|     struct _Py_static_objects static_objects;
 | |
| 
 | |
|     /* The following fields are here to avoid allocation during init.
 | |
|        The data is exposed through _PyRuntimeState pointer fields.
 | |
|        These fields should not be accessed directly outside of init.
 | |
| 
 | |
|        All other _PyRuntimeState pointer fields are populated when
 | |
|        needed and default to NULL.
 | |
| 
 | |
|        For now there are some exceptions to that rule, which require
 | |
|        allocation during init.  These will be addressed on a case-by-case
 | |
|        basis.  Most notably, we don't pre-allocated the several mutex
 | |
|        (PyThread_type_lock) fields, because on Windows we only ever get
 | |
|        a pointer type.
 | |
|        */
 | |
| 
 | |
|     /* _PyRuntimeState.interpreters.main */
 | |
|     PyInterpreterState _main_interpreter;
 | |
| 
 | |
| #if defined(__EMSCRIPTEN__) && defined(PY_CALL_TRAMPOLINE)
 | |
|     // Used in "Python/emscripten_trampoline.c" to choose between type
 | |
|     // reflection trampoline and EM_JS trampoline.
 | |
|     bool wasm_type_reflection_available;
 | |
| #endif
 | |
| 
 | |
| } _PyRuntimeState;
 | |
| 
 | |
| 
 | |
| /* other API */
 | |
| 
 | |
| // Export _PyRuntime for shared extensions which use it in static inline
 | |
| // functions for best performance, like _Py_IsMainThread() or _Py_ID().
 | |
| // It's also made accessible for debuggers and profilers.
 | |
| PyAPI_DATA(_PyRuntimeState) _PyRuntime;
 | |
| 
 | |
| extern PyStatus _PyRuntimeState_Init(_PyRuntimeState *runtime);
 | |
| extern void _PyRuntimeState_Fini(_PyRuntimeState *runtime);
 | |
| 
 | |
| #ifdef HAVE_FORK
 | |
| extern PyStatus _PyRuntimeState_ReInitThreads(_PyRuntimeState *runtime);
 | |
| #endif
 | |
| 
 | |
| /* Initialize _PyRuntimeState.
 | |
|    Return NULL on success, or return an error message on failure. */
 | |
| extern PyStatus _PyRuntime_Initialize(void);
 | |
| 
 | |
| extern void _PyRuntime_Finalize(void);
 | |
| 
 | |
| 
 | |
| static inline PyThreadState*
 | |
| _PyRuntimeState_GetFinalizing(_PyRuntimeState *runtime) {
 | |
|     return (PyThreadState*)_Py_atomic_load_ptr_relaxed(&runtime->_finalizing);
 | |
| }
 | |
| 
 | |
| static inline unsigned long
 | |
| _PyRuntimeState_GetFinalizingID(_PyRuntimeState *runtime) {
 | |
|     return _Py_atomic_load_ulong_relaxed(&runtime->_finalizing_id);
 | |
| }
 | |
| 
 | |
| static inline void
 | |
| _PyRuntimeState_SetFinalizing(_PyRuntimeState *runtime, PyThreadState *tstate) {
 | |
|     _Py_atomic_store_ptr_relaxed(&runtime->_finalizing, tstate);
 | |
|     if (tstate == NULL) {
 | |
|         _Py_atomic_store_ulong_relaxed(&runtime->_finalizing_id, 0);
 | |
|     }
 | |
|     else {
 | |
|         // XXX Re-enable this assert once gh-109860 is fixed.
 | |
|         //assert(tstate->thread_id == PyThread_get_thread_ident());
 | |
|         _Py_atomic_store_ulong_relaxed(&runtime->_finalizing_id,
 | |
|                                        tstate->thread_id);
 | |
|     }
 | |
| }
 | |
| 
 | |
| #ifdef __cplusplus
 | |
| }
 | |
| #endif
 | |
| #endif /* !Py_INTERNAL_RUNTIME_H */
 | 
