| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | #ifndef Py_INTERNAL_OPTIMIZER_H
 | 
					
						
							|  |  |  | #define Py_INTERNAL_OPTIMIZER_H
 | 
					
						
							|  |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  | extern "C" { | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef Py_BUILD_CORE
 | 
					
						
							|  |  |  | #  error "this header requires Py_BUILD_CORE define"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-19 15:23:32 +01:00
										 |  |  | #include "pycore_typedefs.h"      // _PyInterpreterFrame
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | #include "pycore_uop_ids.h"
 | 
					
						
							| 
									
										
										
										
											2025-06-27 19:37:44 +08:00
										 |  |  | #include "pycore_stackref.h"      // _PyStackRef
 | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | #include <stdbool.h>
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct _PyExecutorLinkListNode { | 
					
						
							|  |  |  |     struct _PyExecutorObject *next; | 
					
						
							|  |  |  |     struct _PyExecutorObject *previous; | 
					
						
							|  |  |  | } _PyExecutorLinkListNode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Bloom filter with m = 256
 | 
					
						
							|  |  |  |  * https://en.wikipedia.org/wiki/Bloom_filter */
 | 
					
						
							|  |  |  | #define _Py_BLOOM_FILTER_WORDS 8
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |     uint32_t bits[_Py_BLOOM_FILTER_WORDS]; | 
					
						
							|  |  |  | } _PyBloomFilter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |     uint8_t opcode; | 
					
						
							|  |  |  |     uint8_t oparg; | 
					
						
							| 
									
										
										
										
											2024-09-26 17:35:42 -07:00
										 |  |  |     uint8_t valid:1; | 
					
						
							|  |  |  |     uint8_t linked:1; | 
					
						
							|  |  |  |     uint8_t chain_depth:6;  // Must be big enough for MAX_CHAIN_DEPTH - 1.
 | 
					
						
							|  |  |  |     bool warm; | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  |     int index;           // Index of ENTER_EXECUTOR (if code isn't NULL, below).
 | 
					
						
							|  |  |  |     _PyBloomFilter bloom; | 
					
						
							|  |  |  |     _PyExecutorLinkListNode links; | 
					
						
							|  |  |  |     PyCodeObject *code;  // Weak (NULL if no corresponding ENTER_EXECUTOR).
 | 
					
						
							|  |  |  | } _PyVMData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Depending on the format,
 | 
					
						
							|  |  |  |  * the 32 bits between the oparg and operand are: | 
					
						
							|  |  |  |  * UOP_FORMAT_TARGET: | 
					
						
							|  |  |  |  *    uint32_t target; | 
					
						
							|  |  |  |  * UOP_FORMAT_JUMP | 
					
						
							|  |  |  |  *    uint16_t jump_target; | 
					
						
							|  |  |  |  *    uint16_t error_target; | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2024-07-01 13:17:40 -07:00
										 |  |  |     uint16_t opcode:15; | 
					
						
							|  |  |  |     uint16_t format:1; | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  |     uint16_t oparg; | 
					
						
							|  |  |  |     union { | 
					
						
							|  |  |  |         uint32_t target; | 
					
						
							|  |  |  |         struct { | 
					
						
							| 
									
										
										
										
											2024-07-01 13:17:40 -07:00
										 |  |  |             uint16_t jump_target; | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  |             uint16_t error_target; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |     uint64_t operand0;  // A cache entry
 | 
					
						
							|  |  |  |     uint64_t operand1; | 
					
						
							| 
									
										
										
										
											2024-12-13 11:00:00 +00:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  |     uint64_t execution_count; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | } _PyUOpInstruction; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-01 16:26:07 +01:00
										 |  |  | typedef struct _PyExitData { | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  |     uint32_t target; | 
					
						
							| 
									
										
										
										
											2025-08-01 16:26:07 +01:00
										 |  |  |     uint16_t index; | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  |     _Py_BackoffCounter temperature; | 
					
						
							| 
									
										
										
										
											2025-05-04 10:05:35 +01:00
										 |  |  |     struct _PyExecutorObject *executor; | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | } _PyExitData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _PyExecutorObject { | 
					
						
							|  |  |  |     PyObject_VAR_HEAD | 
					
						
							|  |  |  |     const _PyUOpInstruction *trace; | 
					
						
							|  |  |  |     _PyVMData vm_data; /* Used by the VM, but opaque to the optimizer */ | 
					
						
							|  |  |  |     uint32_t exit_count; | 
					
						
							|  |  |  |     uint32_t code_size; | 
					
						
							|  |  |  |     size_t jit_size; | 
					
						
							|  |  |  |     void *jit_code; | 
					
						
							|  |  |  |     _PyExitData exits[1]; | 
					
						
							|  |  |  | } _PyExecutorObject; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-04 10:05:35 +01:00
										 |  |  | /* If pending deletion list gets large enough, then scan,
 | 
					
						
							|  |  |  |  * and free any executors that aren't executing | 
					
						
							|  |  |  |  * i.e. any that aren't a thread's current_executor. */ | 
					
						
							|  |  |  | #define EXECUTOR_DELETE_LIST_MAX 100
 | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Export for '_opcode' shared extension (JIT compiler).
 | 
					
						
							|  |  |  | PyAPI_FUNC(_PyExecutorObject*) _Py_GetExecutor(PyCodeObject *code, int offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void _Py_ExecutorInit(_PyExecutorObject *, const _PyBloomFilter *); | 
					
						
							|  |  |  | void _Py_ExecutorDetach(_PyExecutorObject *); | 
					
						
							|  |  |  | void _Py_BloomFilter_Init(_PyBloomFilter *); | 
					
						
							|  |  |  | void _Py_BloomFilter_Add(_PyBloomFilter *bloom, void *obj); | 
					
						
							|  |  |  | PyAPI_FUNC(void) _Py_Executor_DependsOn(_PyExecutorObject *executor, void *obj); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define _Py_MAX_ALLOWED_BUILTINS_MODIFICATIONS 3
 | 
					
						
							|  |  |  | #define _Py_MAX_ALLOWED_GLOBALS_MODIFICATIONS 6
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef _Py_TIER2
 | 
					
						
							|  |  |  | PyAPI_FUNC(void) _Py_Executors_InvalidateDependency(PyInterpreterState *interp, void *obj, int is_invalidation); | 
					
						
							|  |  |  | PyAPI_FUNC(void) _Py_Executors_InvalidateAll(PyInterpreterState *interp, int is_invalidation); | 
					
						
							| 
									
										
										
										
											2024-09-26 17:35:42 -07:00
										 |  |  | PyAPI_FUNC(void) _Py_Executors_InvalidateCold(PyInterpreterState *interp); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | #else
 | 
					
						
							|  |  |  | #  define _Py_Executors_InvalidateDependency(A, B, C) ((void)0)
 | 
					
						
							|  |  |  | #  define _Py_Executors_InvalidateAll(A, B) ((void)0)
 | 
					
						
							| 
									
										
										
										
											2024-09-26 17:35:42 -07:00
										 |  |  | #  define _Py_Executors_InvalidateCold(A) ((void)0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-26 17:35:42 -07:00
										 |  |  | // Used as the threshold to trigger executor invalidation when
 | 
					
						
							|  |  |  | // trace_run_counter is greater than this value.
 | 
					
						
							|  |  |  | #define JIT_CLEANUP_THRESHOLD 100000
 | 
					
						
							| 
									
										
										
										
											2024-06-26 13:54:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | // This is the length of the trace we project initially.
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  | #define UOP_MAX_TRACE_LENGTH 800
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define TRACE_STACK_SIZE 5
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-19 15:23:32 +01:00
										 |  |  | int _Py_uop_analyze_and_optimize(_PyInterpreterFrame *frame, | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     _PyUOpInstruction *trace, int trace_len, int curr_stackentries, | 
					
						
							|  |  |  |     _PyBloomFilter *dependencies); | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-29 13:53:25 -07:00
										 |  |  | extern PyTypeObject _PyUOpExecutor_Type; | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 18:20:12 +02:00
										 |  |  | #define UOP_FORMAT_TARGET 0
 | 
					
						
							| 
									
										
										
										
											2024-07-01 13:17:40 -07:00
										 |  |  | #define UOP_FORMAT_JUMP 1
 | 
					
						
							| 
									
										
										
										
											2024-05-10 18:20:12 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline uint32_t uop_get_target(const _PyUOpInstruction *inst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(inst->format == UOP_FORMAT_TARGET); | 
					
						
							|  |  |  |     return inst->target; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint16_t uop_get_jump_target(const _PyUOpInstruction *inst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(inst->format == UOP_FORMAT_JUMP); | 
					
						
							|  |  |  |     return inst->jump_target; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline uint16_t uop_get_error_target(const _PyUOpInstruction *inst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(inst->format != UOP_FORMAT_TARGET); | 
					
						
							|  |  |  |     return inst->error_target; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | // Holds locals, stack, locals, stack ... co_consts (in that order)
 | 
					
						
							|  |  |  | #define MAX_ABSTRACT_INTERP_SIZE 4096
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | #define TY_ARENA_SIZE (UOP_MAX_TRACE_LENGTH * 5)
 | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Need extras for root frame and for overflow frame (see TRACE_STACK_PUSH())
 | 
					
						
							|  |  |  | #define MAX_ABSTRACT_FRAME_DEPTH (TRACE_STACK_SIZE + 2)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 12:39:31 -07:00
										 |  |  | // The maximum number of side exits that we can take before requiring forward
 | 
					
						
							|  |  |  | // progress (and inserting a new ENTER_EXECUTOR instruction). In practice, this
 | 
					
						
							|  |  |  | // is the "maximum amount of polymorphism" that an isolated trace tree can
 | 
					
						
							|  |  |  | // handle before rejoining the rest of the program.
 | 
					
						
							|  |  |  | #define MAX_CHAIN_DEPTH 4
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | /* Symbols */ | 
					
						
							|  |  |  | /* See explanation in optimizer_symbols.c */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef enum _JitSymType { | 
					
						
							|  |  |  |     JIT_SYM_UNKNOWN_TAG = 1, | 
					
						
							|  |  |  |     JIT_SYM_NULL_TAG = 2, | 
					
						
							|  |  |  |     JIT_SYM_NON_NULL_TAG = 3, | 
					
						
							|  |  |  |     JIT_SYM_BOTTOM_TAG = 4, | 
					
						
							|  |  |  |     JIT_SYM_TYPE_VERSION_TAG = 5, | 
					
						
							|  |  |  |     JIT_SYM_KNOWN_CLASS_TAG = 6, | 
					
						
							|  |  |  |     JIT_SYM_KNOWN_VALUE_TAG = 7, | 
					
						
							|  |  |  |     JIT_SYM_TUPLE_TAG = 8, | 
					
						
							| 
									
										
										
										
											2025-03-02 13:21:34 -08:00
										 |  |  |     JIT_SYM_TRUTHINESS_TAG = 9, | 
					
						
							| 
									
										
										
										
											2025-06-19 11:10:29 +01:00
										 |  |  |     JIT_SYM_COMPACT_INT = 10, | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | } JitSymType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _jit_opt_known_class { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							|  |  |  |     uint32_t version; | 
					
						
							|  |  |  |     PyTypeObject *type; | 
					
						
							|  |  |  | } JitOptKnownClass; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _jit_opt_known_version { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							|  |  |  |     uint32_t version; | 
					
						
							|  |  |  | } JitOptKnownVersion; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _jit_opt_known_value { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							|  |  |  |     PyObject *value; | 
					
						
							|  |  |  | } JitOptKnownValue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MAX_SYMBOLIC_TUPLE_SIZE 7
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _jit_opt_tuple { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							|  |  |  |     uint8_t length; | 
					
						
							|  |  |  |     uint16_t items[MAX_SYMBOLIC_TUPLE_SIZE]; | 
					
						
							|  |  |  | } JitOptTuple; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-02 13:21:34 -08:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							| 
									
										
										
										
											2025-03-21 00:59:41 +01:00
										 |  |  |     bool invert; | 
					
						
							| 
									
										
										
										
											2025-03-02 13:21:34 -08:00
										 |  |  |     uint16_t value; | 
					
						
							|  |  |  | } JitOptTruthiness; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-19 11:10:29 +01:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							|  |  |  | } JitOptCompactInt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | typedef union _jit_opt_symbol { | 
					
						
							|  |  |  |     uint8_t tag; | 
					
						
							|  |  |  |     JitOptKnownClass cls; | 
					
						
							|  |  |  |     JitOptKnownValue value; | 
					
						
							|  |  |  |     JitOptKnownVersion version; | 
					
						
							|  |  |  |     JitOptTuple tuple; | 
					
						
							| 
									
										
										
										
											2025-03-02 13:21:34 -08:00
										 |  |  |     JitOptTruthiness truthiness; | 
					
						
							| 
									
										
										
										
											2025-06-19 11:10:29 +01:00
										 |  |  |     JitOptCompactInt compact; | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | } JitOptSymbol; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  | // This mimics the _PyStackRef API
 | 
					
						
							|  |  |  | typedef union { | 
					
						
							|  |  |  |     uintptr_t bits; | 
					
						
							|  |  |  | } JitOptRef; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define REF_IS_BORROWED 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define JIT_BITS_TO_PTR_MASKED(REF) ((JitOptSymbol *)(((REF).bits) & (~REF_IS_BORROWED)))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline JitOptSymbol * | 
					
						
							|  |  |  | PyJitRef_Unwrap(JitOptRef ref) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return JIT_BITS_TO_PTR_MASKED(ref); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool _Py_uop_symbol_is_immortal(JitOptSymbol *sym); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline JitOptRef | 
					
						
							|  |  |  | PyJitRef_Wrap(JitOptSymbol *sym) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (JitOptRef){.bits=(uintptr_t)sym}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline JitOptRef | 
					
						
							|  |  |  | PyJitRef_Borrow(JitOptRef ref) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (JitOptRef){ .bits = ref.bits | REF_IS_BORROWED }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const JitOptRef PyJitRef_NULL = {.bits = REF_IS_BORROWED}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline bool | 
					
						
							|  |  |  | PyJitRef_IsNull(JitOptRef ref) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return ref.bits == PyJitRef_NULL.bits; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int | 
					
						
							|  |  |  | PyJitRef_IsBorrowed(JitOptRef ref) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (ref.bits & REF_IS_BORROWED) == REF_IS_BORROWED; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct _Py_UOpsAbstractFrame { | 
					
						
							|  |  |  |     // Max stacklen
 | 
					
						
							|  |  |  |     int stack_len; | 
					
						
							|  |  |  |     int locals_len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  |     JitOptRef *stack_pointer; | 
					
						
							|  |  |  |     JitOptRef *stack; | 
					
						
							|  |  |  |     JitOptRef *locals; | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct _Py_UOpsAbstractFrame _Py_UOpsAbstractFrame; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct ty_arena { | 
					
						
							|  |  |  |     int ty_curr_number; | 
					
						
							|  |  |  |     int ty_max_number; | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  |     JitOptSymbol arena[TY_ARENA_SIZE]; | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | } ty_arena; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | typedef struct _JitOptContext { | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  |     char done; | 
					
						
							|  |  |  |     char out_of_space; | 
					
						
							|  |  |  |     bool contradiction; | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  |     // The current "executing" frame.
 | 
					
						
							|  |  |  |     _Py_UOpsAbstractFrame *frame; | 
					
						
							|  |  |  |     _Py_UOpsAbstractFrame frames[MAX_ABSTRACT_FRAME_DEPTH]; | 
					
						
							|  |  |  |     int curr_frame_depth; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Arena for the symbolic types.
 | 
					
						
							|  |  |  |     ty_arena t_arena; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  |     JitOptRef *n_consumed; | 
					
						
							|  |  |  |     JitOptRef *limit; | 
					
						
							|  |  |  |     JitOptRef locals_and_stack[MAX_ABSTRACT_INTERP_SIZE]; | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | } JitOptContext; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  | extern bool _Py_uop_sym_is_null(JitOptRef sym); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_is_not_null(JitOptRef sym); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_is_const(JitOptContext *ctx, JitOptRef sym); | 
					
						
							|  |  |  | extern PyObject *_Py_uop_sym_get_const(JitOptContext *ctx, JitOptRef sym); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_new_unknown(JitOptContext *ctx); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_new_not_null(JitOptContext *ctx); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_new_type( | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  |     JitOptContext *ctx, PyTypeObject *typ); | 
					
						
							| 
									
										
										
										
											2025-06-19 11:10:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  | extern JitOptRef _Py_uop_sym_new_const(JitOptContext *ctx, PyObject *const_val); | 
					
						
							| 
									
										
										
										
											2025-06-27 19:37:44 +08:00
										 |  |  | extern JitOptRef _Py_uop_sym_new_const_steal(JitOptContext *ctx, PyObject *const_val); | 
					
						
							|  |  |  | bool _Py_uop_sym_is_safe_const(JitOptContext *ctx, JitOptRef sym); | 
					
						
							|  |  |  | _PyStackRef _Py_uop_sym_get_const_as_stackref(JitOptContext *ctx, JitOptRef sym); | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  | extern JitOptRef _Py_uop_sym_new_null(JitOptContext *ctx); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_has_type(JitOptRef sym); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_matches_type(JitOptRef sym, PyTypeObject *typ); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_matches_type_version(JitOptRef sym, unsigned int version); | 
					
						
							|  |  |  | extern void _Py_uop_sym_set_null(JitOptContext *ctx, JitOptRef sym); | 
					
						
							|  |  |  | extern void _Py_uop_sym_set_non_null(JitOptContext *ctx, JitOptRef sym); | 
					
						
							|  |  |  | extern void _Py_uop_sym_set_type(JitOptContext *ctx, JitOptRef sym, PyTypeObject *typ); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_set_type_version(JitOptContext *ctx, JitOptRef sym, unsigned int version); | 
					
						
							|  |  |  | extern void _Py_uop_sym_set_const(JitOptContext *ctx, JitOptRef sym, PyObject *const_val); | 
					
						
							|  |  |  | extern bool _Py_uop_sym_is_bottom(JitOptRef sym); | 
					
						
							|  |  |  | extern int _Py_uop_sym_truthiness(JitOptContext *ctx, JitOptRef sym); | 
					
						
							|  |  |  | extern PyTypeObject *_Py_uop_sym_get_type(JitOptRef sym); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_new_tuple(JitOptContext *ctx, int size, JitOptRef *args); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_tuple_getitem(JitOptContext *ctx, JitOptRef sym, int item); | 
					
						
							|  |  |  | extern int _Py_uop_sym_tuple_length(JitOptRef sym); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_new_truthiness(JitOptContext *ctx, JitOptRef value, bool truthy); | 
					
						
							| 
									
										
										
										
											2025-06-19 11:10:29 +01:00
										 |  |  | extern bool _Py_uop_sym_is_compact_int(JitOptRef sym); | 
					
						
							|  |  |  | extern JitOptRef _Py_uop_sym_new_compact_int(JitOptContext *ctx); | 
					
						
							|  |  |  | extern void _Py_uop_sym_set_compact_int(JitOptContext *ctx,  JitOptRef sym); | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern void _Py_uop_abstractcontext_init(JitOptContext *ctx); | 
					
						
							|  |  |  | extern void _Py_uop_abstractcontext_fini(JitOptContext *ctx); | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern _Py_UOpsAbstractFrame *_Py_uop_frame_new( | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  |     JitOptContext *ctx, | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  |     PyCodeObject *co, | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  |     int curr_stackentries, | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  |     JitOptRef *args, | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  |     int arg_len); | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | extern int _Py_uop_frame_pop(JitOptContext *ctx); | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | PyAPI_FUNC(PyObject *) _Py_uop_symbols_test(PyObject *self, PyObject *ignored); | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-19 15:23:32 +01:00
										 |  |  | PyAPI_FUNC(int) _PyOptimizer_Optimize(_PyInterpreterFrame *frame, _Py_CODEUNIT *start, _PyExecutorObject **exec_ptr, int chain_depth); | 
					
						
							| 
									
										
										
										
											2024-02-29 08:11:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-01 16:26:07 +01:00
										 |  |  | static inline _PyExecutorObject *_PyExecutor_FromExit(_PyExitData *exit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyExitData *exit0 = exit - exit->index; | 
					
						
							|  |  |  |     return (_PyExecutorObject *)(((char *)exit0) - offsetof(_PyExecutorObject, exits)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern _PyExecutorObject *_PyExecutor_GetColdExecutor(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyAPI_FUNC(void) _PyExecutor_ClearExit(_PyExitData *exit); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-02 00:19:05 +01:00
										 |  |  | static inline int is_terminator(const _PyUOpInstruction *uop) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int opcode = uop->opcode; | 
					
						
							|  |  |  |     return ( | 
					
						
							|  |  |  |         opcode == _EXIT_TRACE || | 
					
						
							| 
									
										
										
										
											2025-02-07 11:41:17 -08:00
										 |  |  |         opcode == _JUMP_TO_TOP | 
					
						
							| 
									
										
										
										
											2024-08-02 00:19:05 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-01 16:26:07 +01:00
										 |  |  | extern void _PyExecutor_Free(_PyExecutorObject *self); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-13 11:00:00 +00:00
										 |  |  | PyAPI_FUNC(int) _PyDumpExecutors(FILE *out); | 
					
						
							| 
									
										
										
										
											2025-05-04 10:05:35 +01:00
										 |  |  | #ifdef _Py_TIER2
 | 
					
						
							|  |  |  | extern void _Py_ClearExecutorDeletionList(PyInterpreterState *interp); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-12-13 11:00:00 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif /* !Py_INTERNAL_OPTIMIZER_H */
 |