| 
									
										
										
										
											2024-04-30 18:26:34 -07:00
										 |  |  | #ifdef _Py_TIER2
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2024-02-26 08:42:53 -08:00
										 |  |  |  * This file contains the support code for CPython's uops optimizer. | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |  * It also performs some simple optimizations. | 
					
						
							|  |  |  |  * It performs a traditional data-flow analysis[1] over the trace of uops. | 
					
						
							|  |  |  |  * Using the information gained, it chooses to emit, or skip certain instructions | 
					
						
							|  |  |  |  * if possible. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * [1] For information on data-flow analysis, please see | 
					
						
							|  |  |  |  * https://clang.llvm.org/docs/DataFlowAnalysisIntro.html
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * */ | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | #include "Python.h"
 | 
					
						
							|  |  |  | #include "opcode.h"
 | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | #include "pycore_dict.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | #include "pycore_interp.h"
 | 
					
						
							|  |  |  | #include "pycore_opcode_metadata.h"
 | 
					
						
							|  |  |  | #include "pycore_opcode_utils.h"
 | 
					
						
							|  |  |  | #include "pycore_pystate.h"       // _PyInterpreterState_GET()
 | 
					
						
							| 
									
										
										
										
											2024-01-02 14:09:57 -08:00
										 |  |  | #include "pycore_uop_metadata.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | #include "pycore_long.h"
 | 
					
						
							| 
									
										
										
										
											2025-03-20 15:39:38 +00:00
										 |  |  | #include "pycore_interpframe.h"  // _PyFrame_GetCode
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | #include "pycore_optimizer.h"
 | 
					
						
							|  |  |  | #include "pycore_object.h"
 | 
					
						
							|  |  |  | #include "pycore_function.h"
 | 
					
						
							|  |  |  | #include "pycore_uop_ids.h"
 | 
					
						
							|  |  |  | #include "pycore_range.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdarg.h>
 | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | #include <stdbool.h>
 | 
					
						
							|  |  |  | #include <stdint.h>
 | 
					
						
							|  |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef Py_DEBUG
 | 
					
						
							| 
									
										
										
										
											2024-02-20 12:24:35 -08:00
										 |  |  |     extern const char *_PyUOpName(int index); | 
					
						
							| 
									
										
										
										
											2024-03-18 11:08:43 -07:00
										 |  |  |     extern void _PyUOpPrint(const _PyUOpInstruction *uop); | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     static const char *const DEBUG_ENV = "PYTHON_OPT_DEBUG"; | 
					
						
							|  |  |  |     static inline int get_lltrace(void) { | 
					
						
							|  |  |  |         char *uop_debug = Py_GETENV(DEBUG_ENV); | 
					
						
							|  |  |  |         int lltrace = 0; | 
					
						
							|  |  |  |         if (uop_debug != NULL && *uop_debug >= '0') { | 
					
						
							|  |  |  |             lltrace = *uop_debug - '0';  // TODO: Parse an int and all that
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return lltrace; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     #define DPRINTF(level, ...) \
 | 
					
						
							|  |  |  |     if (get_lltrace() >= (level)) { printf(__VA_ARGS__); } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     #define DPRINTF(level, ...)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | get_mutations(PyObject* dict) { | 
					
						
							|  |  |  |     assert(PyDict_CheckExact(dict)); | 
					
						
							|  |  |  |     PyDictObject *d = (PyDictObject *)dict; | 
					
						
							| 
									
										
										
										
											2024-10-01 12:39:56 -04:00
										 |  |  |     return (d->_ma_watcher_tag >> DICT_MAX_WATCHERS) & ((1 << DICT_WATCHED_MUTATION_BITS)-1); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-24 12:08:31 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | increment_mutations(PyObject* dict) { | 
					
						
							|  |  |  |     assert(PyDict_CheckExact(dict)); | 
					
						
							|  |  |  |     PyDictObject *d = (PyDictObject *)dict; | 
					
						
							| 
									
										
										
										
											2024-10-01 12:39:56 -04:00
										 |  |  |     d->_ma_watcher_tag += (1 << DICT_MAX_WATCHERS); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-12 16:07:38 +00:00
										 |  |  | /* The first two dict watcher IDs are reserved for CPython,
 | 
					
						
							|  |  |  |  * so we don't need to check that they haven't been used */ | 
					
						
							|  |  |  | #define BUILTINS_WATCHER_ID 0
 | 
					
						
							|  |  |  | #define GLOBALS_WATCHER_ID  1
 | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  | #define TYPE_WATCHER_ID  0
 | 
					
						
							| 
									
										
										
										
											2024-02-12 16:07:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | globals_watcher_callback(PyDict_WatchEvent event, PyObject* dict, | 
					
						
							|  |  |  |                          PyObject* key, PyObject* new_value) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-12 16:07:38 +00:00
										 |  |  |     RARE_EVENT_STAT_INC(watched_globals_modification); | 
					
						
							|  |  |  |     assert(get_mutations(dict) < _Py_MAX_ALLOWED_GLOBALS_MODIFICATIONS); | 
					
						
							| 
									
										
										
										
											2024-02-26 12:51:47 -05:00
										 |  |  |     _Py_Executors_InvalidateDependency(_PyInterpreterState_GET(), dict, 1); | 
					
						
							| 
									
										
										
										
											2024-02-12 16:07:38 +00:00
										 |  |  |     increment_mutations(dict); | 
					
						
							|  |  |  |     PyDict_Unwatch(GLOBALS_WATCHER_ID, dict); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  | static int | 
					
						
							|  |  |  | type_watcher_callback(PyTypeObject* type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _Py_Executors_InvalidateDependency(_PyInterpreterState_GET(), type, 1); | 
					
						
							|  |  |  |     PyType_Unwatch(TYPE_WATCHER_ID, (PyObject *)type); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  | convert_global_to_const(_PyUOpInstruction *inst, PyObject *obj, bool pop) | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |     assert(inst->opcode == _LOAD_GLOBAL_MODULE || inst->opcode == _LOAD_GLOBAL_BUILTINS || inst->opcode == _LOAD_ATTR_MODULE); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     assert(PyDict_CheckExact(obj)); | 
					
						
							|  |  |  |     PyDictObject *dict = (PyDictObject *)obj; | 
					
						
							|  |  |  |     assert(dict->ma_keys->dk_kind == DICT_KEYS_UNICODE); | 
					
						
							|  |  |  |     PyDictUnicodeEntry *entries = DK_UNICODE_ENTRIES(dict->ma_keys); | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |     int64_t index = inst->operand1; | 
					
						
							|  |  |  |     assert(index <= UINT16_MAX); | 
					
						
							|  |  |  |     if ((int)index >= dict->ma_keys->dk_nentries) { | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |     PyObject *res = entries[index].me_value; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     if (res == NULL) { | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (_Py_IsImmortal(res)) { | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |         inst->opcode = pop ? _POP_TOP_LOAD_CONST_INLINE_BORROW : _LOAD_CONST_INLINE_BORROW; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |         inst->opcode = pop ? _POP_TOP_LOAD_CONST_INLINE : _LOAD_CONST_INLINE; | 
					
						
							| 
									
										
										
										
											2025-01-27 16:24:48 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (inst->oparg & 1) { | 
					
						
							|  |  |  |         assert(inst[1].opcode == _PUSH_NULL_CONDITIONAL); | 
					
						
							|  |  |  |         assert(inst[1].oparg & 1); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |     inst->operand0 = (uint64_t)res; | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  |     return res; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | incorrect_keys(_PyUOpInstruction *inst, PyObject *obj) | 
					
						
							| 
									
										
										
										
											2024-01-24 12:08:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     if (!PyDict_CheckExact(obj)) { | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyDictObject *dict = (PyDictObject *)obj; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |     if (dict->ma_keys->dk_version != inst->operand0) { | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Returns 1 if successfully optimized
 | 
					
						
							|  |  |  |  *         0 if the trace is not suitable for optimization (yet) | 
					
						
							|  |  |  |  *        -1 if there was an error. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | remove_globals(_PyInterpreterFrame *frame, _PyUOpInstruction *buffer, | 
					
						
							|  |  |  |                int buffer_size, _PyBloomFilter *dependencies) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyInterpreterState *interp = _PyInterpreterState_GET(); | 
					
						
							|  |  |  |     PyObject *builtins = frame->f_builtins; | 
					
						
							|  |  |  |     if (builtins != interp->builtins) { | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |         OPT_STAT_INC(remove_globals_builtins_changed); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyObject *globals = frame->f_globals; | 
					
						
							| 
									
										
										
										
											2024-09-24 13:08:18 -07:00
										 |  |  |     PyFunctionObject *function = _PyFrame_GetFunction(frame); | 
					
						
							| 
									
										
										
										
											2024-03-08 09:47:41 +00:00
										 |  |  |     assert(PyFunction_Check(function)); | 
					
						
							|  |  |  |     assert(function->func_builtins == builtins); | 
					
						
							|  |  |  |     assert(function->func_globals == globals); | 
					
						
							|  |  |  |     uint32_t function_version = _PyFunction_GetVersionForCurrentState(function); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     /* In order to treat globals as constants, we need to
 | 
					
						
							|  |  |  |      * know that the globals dict is the one we expected, and | 
					
						
							|  |  |  |      * that it hasn't changed | 
					
						
							|  |  |  |      * In order to treat builtins as constants,  we need to | 
					
						
							|  |  |  |      * know that the builtins dict is the one we expected, and | 
					
						
							|  |  |  |      * that it hasn't changed and that the global dictionary's | 
					
						
							|  |  |  |      * keys have not changed */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* These values represent stacks of booleans (one bool per bit).
 | 
					
						
							|  |  |  |      * Pushing a frame shifts left, popping a frame shifts right. */ | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |     uint32_t function_checked = 0; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     uint32_t builtins_watched = 0; | 
					
						
							|  |  |  |     uint32_t globals_watched = 0; | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |     uint32_t prechecked_function_version = 0; | 
					
						
							| 
									
										
										
										
											2024-02-12 16:07:38 +00:00
										 |  |  |     if (interp->dict_state.watchers[GLOBALS_WATCHER_ID] == NULL) { | 
					
						
							|  |  |  |         interp->dict_state.watchers[GLOBALS_WATCHER_ID] = globals_watcher_callback; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  |     if (interp->type_watchers[TYPE_WATCHER_ID] == NULL) { | 
					
						
							|  |  |  |         interp->type_watchers[TYPE_WATCHER_ID] = type_watcher_callback; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     for (int pc = 0; pc < buffer_size; pc++) { | 
					
						
							|  |  |  |         _PyUOpInstruction *inst = &buffer[pc]; | 
					
						
							|  |  |  |         int opcode = inst->opcode; | 
					
						
							|  |  |  |         switch(opcode) { | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |             case _GUARD_GLOBALS_VERSION: | 
					
						
							|  |  |  |                 if (incorrect_keys(inst, globals)) { | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |                     OPT_STAT_INC(remove_globals_incorrect_keys); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                     return 0; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |                 if (get_mutations(globals) >= _Py_MAX_ALLOWED_GLOBALS_MODIFICATIONS) { | 
					
						
							| 
									
										
										
										
											2024-10-09 08:18:25 -07:00
										 |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |                 if ((globals_watched & 1) == 0) { | 
					
						
							|  |  |  |                     PyDict_Watch(GLOBALS_WATCHER_ID, globals); | 
					
						
							|  |  |  |                     _Py_BloomFilter_Add(dependencies, globals); | 
					
						
							|  |  |  |                     globals_watched |= 1; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                 if (function_checked & 1) { | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                     buffer[pc].opcode = NOP; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 else { | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                     buffer[pc].opcode = _CHECK_FUNCTION; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |                     buffer[pc].operand0 = function_version; | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                     function_checked |= 1; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |             case _LOAD_GLOBAL_BUILTINS: | 
					
						
							|  |  |  |                 if (incorrect_keys(inst, builtins)) { | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |                     OPT_STAT_INC(remove_globals_incorrect_keys); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                     return 0; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |                 if (interp->rare_events.builtin_dict >= _Py_MAX_ALLOWED_BUILTINS_MODIFICATIONS) { | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |                 if ((builtins_watched & 1) == 0) { | 
					
						
							|  |  |  |                     PyDict_Watch(BUILTINS_WATCHER_ID, builtins); | 
					
						
							|  |  |  |                     builtins_watched |= 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (function_checked & globals_watched & 1) { | 
					
						
							|  |  |  |                     convert_global_to_const(inst, builtins, false); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case _LOAD_GLOBAL_MODULE: | 
					
						
							|  |  |  |                 if (incorrect_keys(inst, globals)) { | 
					
						
							|  |  |  |                     OPT_STAT_INC(remove_globals_incorrect_keys); | 
					
						
							|  |  |  |                     return 0; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (get_mutations(globals) >= _Py_MAX_ALLOWED_GLOBALS_MODIFICATIONS) { | 
					
						
							| 
									
										
										
										
											2024-10-09 08:18:25 -07:00
										 |  |  |                     continue; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 if ((globals_watched & 1) == 0) { | 
					
						
							|  |  |  |                     PyDict_Watch(GLOBALS_WATCHER_ID, globals); | 
					
						
							|  |  |  |                     _Py_BloomFilter_Add(dependencies, globals); | 
					
						
							|  |  |  |                     globals_watched |= 1; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |                 if ((function_checked & 1) == 0 && buffer[pc-1].opcode == _NOP) { | 
					
						
							|  |  |  |                     buffer[pc-1].opcode = _CHECK_FUNCTION; | 
					
						
							|  |  |  |                     buffer[pc-1].operand0 = function_version; | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                     function_checked |= 1; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-02-28 18:00:38 +00:00
										 |  |  |                 if (function_checked & 1) { | 
					
						
							|  |  |  |                     convert_global_to_const(inst, globals, false); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case _PUSH_FRAME: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 builtins_watched <<= 1; | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                 globals_watched <<= 1; | 
					
						
							|  |  |  |                 function_checked <<= 1; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |                 uint64_t operand = buffer[pc].operand0; | 
					
						
							| 
									
										
											  
											
												gh-117045: Add code object to function version cache (#117028)
Changes to the function version cache:
- In addition to the function object, also store the code object,
  and allow the latter to be retrieved even if the function has been evicted.
- Stop assigning new function versions after a critical attribute (e.g. `__code__`)
  has been modified; the version is permanently reset to zero in this case.
- Changes to `__annotations__` are no longer considered critical. (This fixes gh-109998.)
Changes to the Tier 2 optimization machinery:
- If we cannot map a function version to a function, but it is still mapped to a code object,
  we continue projecting the trace.
  The operand of the `_PUSH_FRAME` and `_POP_FRAME` opcodes can be either NULL,
  a function object, or a code object with the lowest bit set.
This allows us to trace through code that calls an ephemeral function,
i.e., a function that may not be alive when we are constructing the executor,
e.g. a generator expression or certain nested functions.
We will lose globals removal inside such functions,
but we can still do other peephole operations
(and even possibly [call inlining](https://github.com/python/cpython/pull/116290),
if we decide to do it), which only need the code object.
As before, if we cannot retrieve the code object from the cache, we stop projecting.
											
										 
											2024-03-21 12:37:41 -07:00
										 |  |  |                 if (operand == 0 || (operand & 1)) { | 
					
						
							|  |  |  |                     // It's either a code object or NULL, so bail
 | 
					
						
							|  |  |  |                     return 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 PyFunctionObject *func = (PyFunctionObject *)operand; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 if (func == NULL) { | 
					
						
							|  |  |  |                     return 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 assert(PyFunction_Check(func)); | 
					
						
							| 
									
										
										
										
											2024-03-08 09:47:41 +00:00
										 |  |  |                 function_version = func->func_version; | 
					
						
							|  |  |  |                 if (prechecked_function_version == function_version) { | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                     function_checked |= 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 prechecked_function_version = 0; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 globals = func->func_globals; | 
					
						
							|  |  |  |                 builtins = func->func_builtins; | 
					
						
							|  |  |  |                 if (builtins != interp->builtins) { | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |                     OPT_STAT_INC(remove_globals_builtins_changed); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                     return 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-06-17 14:40:11 +01:00
										 |  |  |             case _RETURN_VALUE: | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |             { | 
					
						
							|  |  |  |                 builtins_watched >>= 1; | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                 globals_watched >>= 1; | 
					
						
							|  |  |  |                 function_checked >>= 1; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |                 uint64_t operand = buffer[pc].operand0; | 
					
						
							| 
									
										
											  
											
												gh-117045: Add code object to function version cache (#117028)
Changes to the function version cache:
- In addition to the function object, also store the code object,
  and allow the latter to be retrieved even if the function has been evicted.
- Stop assigning new function versions after a critical attribute (e.g. `__code__`)
  has been modified; the version is permanently reset to zero in this case.
- Changes to `__annotations__` are no longer considered critical. (This fixes gh-109998.)
Changes to the Tier 2 optimization machinery:
- If we cannot map a function version to a function, but it is still mapped to a code object,
  we continue projecting the trace.
  The operand of the `_PUSH_FRAME` and `_POP_FRAME` opcodes can be either NULL,
  a function object, or a code object with the lowest bit set.
This allows us to trace through code that calls an ephemeral function,
i.e., a function that may not be alive when we are constructing the executor,
e.g. a generator expression or certain nested functions.
We will lose globals removal inside such functions,
but we can still do other peephole operations
(and even possibly [call inlining](https://github.com/python/cpython/pull/116290),
if we decide to do it), which only need the code object.
As before, if we cannot retrieve the code object from the cache, we stop projecting.
											
										 
											2024-03-21 12:37:41 -07:00
										 |  |  |                 if (operand == 0 || (operand & 1)) { | 
					
						
							|  |  |  |                     // It's either a code object or NULL, so bail
 | 
					
						
							|  |  |  |                     return 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 PyFunctionObject *func = (PyFunctionObject *)operand; | 
					
						
							|  |  |  |                 if (func == NULL) { | 
					
						
							|  |  |  |                     return 1; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 assert(PyFunction_Check(func)); | 
					
						
							| 
									
										
										
										
											2024-03-08 09:47:41 +00:00
										 |  |  |                 function_version = func->func_version; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |                 globals = func->func_globals; | 
					
						
							|  |  |  |                 builtins = func->func_builtins; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |             case _CHECK_FUNCTION_EXACT_ARGS: | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |                 prechecked_function_version = (uint32_t)buffer[pc].operand0; | 
					
						
							| 
									
										
										
										
											2024-03-06 13:12:23 +00:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |             default: | 
					
						
							| 
									
										
										
										
											2024-08-02 00:19:05 +01:00
										 |  |  |                 if (is_terminator(inst)) { | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |                     return 1; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define STACK_LEVEL()     ((int)(stack_pointer - ctx->frame->stack))
 | 
					
						
							| 
									
										
										
										
											2024-06-25 16:42:29 +01:00
										 |  |  | #define STACK_SIZE()      ((int)(ctx->frame->stack_len))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define WITHIN_STACK_BOUNDS() \
 | 
					
						
							|  |  |  |     (STACK_LEVEL() >= 0 && STACK_LEVEL() <= STACK_SIZE()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define GETLOCAL(idx)          ((ctx->frame->locals[idx]))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define REPLACE_OP(INST, OP, ARG, OPERAND)    \
 | 
					
						
							|  |  |  |     INST->opcode = OP;            \ | 
					
						
							|  |  |  |     INST->oparg = ARG;            \ | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |     INST->operand0 = OPERAND; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | /* Shortened forms for convenience, used in optimizer_bytecodes.c */ | 
					
						
							|  |  |  | #define sym_is_not_null _Py_uop_sym_is_not_null
 | 
					
						
							|  |  |  | #define sym_is_const _Py_uop_sym_is_const
 | 
					
						
							|  |  |  | #define sym_get_const _Py_uop_sym_get_const
 | 
					
						
							|  |  |  | #define sym_new_unknown _Py_uop_sym_new_unknown
 | 
					
						
							|  |  |  | #define sym_new_not_null _Py_uop_sym_new_not_null
 | 
					
						
							|  |  |  | #define sym_new_type _Py_uop_sym_new_type
 | 
					
						
							|  |  |  | #define sym_is_null _Py_uop_sym_is_null
 | 
					
						
							|  |  |  | #define sym_new_const _Py_uop_sym_new_const
 | 
					
						
							|  |  |  | #define sym_new_null _Py_uop_sym_new_null
 | 
					
						
							| 
									
										
										
										
											2024-03-05 15:06:00 +00:00
										 |  |  | #define sym_has_type _Py_uop_sym_has_type
 | 
					
						
							| 
									
										
										
										
											2024-04-22 13:34:06 +01:00
										 |  |  | #define sym_get_type _Py_uop_sym_get_type
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | #define sym_matches_type _Py_uop_sym_matches_type
 | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  | #define sym_matches_type_version _Py_uop_sym_matches_type_version
 | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  | #define sym_set_null(SYM) _Py_uop_sym_set_null(ctx, SYM)
 | 
					
						
							|  |  |  | #define sym_set_non_null(SYM) _Py_uop_sym_set_non_null(ctx, SYM)
 | 
					
						
							|  |  |  | #define sym_set_type(SYM, TYPE) _Py_uop_sym_set_type(ctx, SYM, TYPE)
 | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  | #define sym_set_type_version(SYM, VERSION) _Py_uop_sym_set_type_version(ctx, SYM, VERSION)
 | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  | #define sym_set_const(SYM, CNST) _Py_uop_sym_set_const(ctx, SYM, CNST)
 | 
					
						
							| 
									
										
										
										
											2024-02-28 14:38:01 -08:00
										 |  |  | #define sym_is_bottom _Py_uop_sym_is_bottom
 | 
					
						
							| 
									
										
										
										
											2024-03-05 11:23:46 +00:00
										 |  |  | #define sym_truthiness _Py_uop_sym_truthiness
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | #define frame_new _Py_uop_frame_new
 | 
					
						
							|  |  |  | #define frame_pop _Py_uop_frame_pop
 | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  | #define sym_new_tuple _Py_uop_sym_new_tuple
 | 
					
						
							|  |  |  | #define sym_tuple_getitem _Py_uop_sym_tuple_getitem
 | 
					
						
							|  |  |  | #define sym_tuple_length _Py_uop_sym_tuple_length
 | 
					
						
							|  |  |  | #define sym_is_immortal _Py_uop_sym_is_immortal
 | 
					
						
							| 
									
										
										
										
											2025-03-02 13:21:34 -08:00
										 |  |  | #define sym_new_truthiness _Py_uop_sym_new_truthiness
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-05 11:23:46 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | optimize_to_bool( | 
					
						
							|  |  |  |     _PyUOpInstruction *this_instr, | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  |     JitOptContext *ctx, | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  |     JitOptRef value, | 
					
						
							|  |  |  |     JitOptRef *result_ptr) | 
					
						
							| 
									
										
										
										
											2024-03-05 11:23:46 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (sym_matches_type(value, &PyBool_Type)) { | 
					
						
							|  |  |  |         REPLACE_OP(this_instr, _NOP, 0, 0); | 
					
						
							|  |  |  |         *result_ptr = value; | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-02 13:21:34 -08:00
										 |  |  |     int truthiness = sym_truthiness(ctx, value); | 
					
						
							| 
									
										
										
										
											2024-03-05 11:23:46 +00:00
										 |  |  |     if (truthiness >= 0) { | 
					
						
							|  |  |  |         PyObject *load = truthiness ? Py_True : Py_False; | 
					
						
							|  |  |  |         REPLACE_OP(this_instr, _POP_TOP_LOAD_CONST_INLINE_BORROW, 0, (uintptr_t)load); | 
					
						
							|  |  |  |         *result_ptr = sym_new_const(ctx, load); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-02-27 13:25:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-05 15:06:00 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | eliminate_pop_guard(_PyUOpInstruction *this_instr, bool exit) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     REPLACE_OP(this_instr, _POP_TOP, 0, 0); | 
					
						
							|  |  |  |     if (exit) { | 
					
						
							|  |  |  |         REPLACE_OP((this_instr+1), _EXIT_TRACE, 0, 0); | 
					
						
							|  |  |  |         this_instr[1].target = this_instr->target; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  | static JitOptRef | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  | lookup_attr(JitOptContext *ctx, _PyUOpInstruction *this_instr, | 
					
						
							|  |  |  |             PyTypeObject *type, PyObject *name, uint16_t immortal, | 
					
						
							|  |  |  |             uint16_t mortal) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // The cached value may be dead, so we need to do the lookup again... :(
 | 
					
						
							|  |  |  |     if (type && PyType_Check(type)) { | 
					
						
							|  |  |  |         PyObject *lookup = _PyType_Lookup(type, name); | 
					
						
							|  |  |  |         if (lookup) { | 
					
						
							|  |  |  |             int opcode = _Py_IsImmortal(lookup) ? immortal : mortal; | 
					
						
							|  |  |  |             REPLACE_OP(this_instr, opcode, 0, (uintptr_t)lookup); | 
					
						
							|  |  |  |             return sym_new_const(ctx, lookup); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return sym_new_not_null(ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 14:40:11 +01:00
										 |  |  | /* _PUSH_FRAME/_RETURN_VALUE's operand can be 0, a PyFunctionObject *, or a
 | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |  * PyCodeObject *. Retrieve the code object if possible. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static PyCodeObject * | 
					
						
							|  |  |  | get_code(_PyUOpInstruction *op) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-17 14:40:11 +01:00
										 |  |  |     assert(op->opcode == _PUSH_FRAME || op->opcode == _RETURN_VALUE || op->opcode == _RETURN_GENERATOR); | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |     PyCodeObject *co = NULL; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |     uint64_t operand = op->operand0; | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |     if (operand == 0) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (operand & 1) { | 
					
						
							|  |  |  |         co = (PyCodeObject *)(operand & ~1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyFunctionObject *func = (PyFunctionObject *)operand; | 
					
						
							|  |  |  |         assert(PyFunction_Check(func)); | 
					
						
							|  |  |  |         co = (PyCodeObject *)func->func_code; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     assert(PyCode_Check(co)); | 
					
						
							|  |  |  |     return co; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-03 01:10:51 +08:00
										 |  |  | static PyCodeObject * | 
					
						
							|  |  |  | get_code_with_logging(_PyUOpInstruction *op) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyCodeObject *co = NULL; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |     uint64_t push_operand = op->operand0; | 
					
						
							| 
									
										
										
										
											2024-10-03 01:10:51 +08:00
										 |  |  |     if (push_operand & 1) { | 
					
						
							|  |  |  |         co = (PyCodeObject *)(push_operand & ~1); | 
					
						
							|  |  |  |         DPRINTF(3, "code=%p ", co); | 
					
						
							|  |  |  |         assert(PyCode_Check(co)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         PyFunctionObject *func = (PyFunctionObject *)push_operand; | 
					
						
							|  |  |  |         DPRINTF(3, "func=%p ", func); | 
					
						
							|  |  |  |         if (func == NULL) { | 
					
						
							|  |  |  |             DPRINTF(3, "\n"); | 
					
						
							|  |  |  |             DPRINTF(1, "Missing function\n"); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         co = (PyCodeObject *)func->func_code; | 
					
						
							|  |  |  |         DPRINTF(3, "code=%p ", co); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return co; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  | // TODO (gh-134584) generate most of this table automatically
 | 
					
						
							|  |  |  | const uint16_t op_without_decref_inputs[MAX_UOP_ID + 1] = { | 
					
						
							|  |  |  |     [_BINARY_OP_MULTIPLY_FLOAT] = _BINARY_OP_MULTIPLY_FLOAT__NO_DECREF_INPUTS, | 
					
						
							|  |  |  |     [_BINARY_OP_ADD_FLOAT] = _BINARY_OP_ADD_FLOAT__NO_DECREF_INPUTS, | 
					
						
							|  |  |  |     [_BINARY_OP_SUBTRACT_FLOAT] = _BINARY_OP_SUBTRACT_FLOAT__NO_DECREF_INPUTS, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | /* 1 for success, 0 for not ready, cannot error at the moment. */ | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2024-02-26 08:42:53 -08:00
										 |  |  | optimize_uops( | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     PyCodeObject *co, | 
					
						
							|  |  |  |     _PyUOpInstruction *trace, | 
					
						
							|  |  |  |     int trace_len, | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  |     int curr_stacklen, | 
					
						
							|  |  |  |     _PyBloomFilter *dependencies | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-20 15:49:15 +00:00
										 |  |  |     JitOptContext context; | 
					
						
							|  |  |  |     JitOptContext *ctx = &context; | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |     uint32_t opcode = UINT16_MAX; | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |     int curr_space = 0; | 
					
						
							|  |  |  |     int max_space = 0; | 
					
						
							|  |  |  |     _PyUOpInstruction *first_valid_check_stack = NULL; | 
					
						
							|  |  |  |     _PyUOpInstruction *corresponding_check_stack = NULL; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  |     _Py_uop_abstractcontext_init(ctx); | 
					
						
							| 
									
										
										
										
											2024-06-08 05:41:45 -04:00
										 |  |  |     _Py_UOpsAbstractFrame *frame = _Py_uop_frame_new(ctx, co, curr_stacklen, NULL, 0); | 
					
						
							| 
									
										
										
										
											2024-02-27 10:51:26 +00:00
										 |  |  |     if (frame == NULL) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx->curr_frame_depth++; | 
					
						
							|  |  |  |     ctx->frame = frame; | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  |     ctx->done = false; | 
					
						
							|  |  |  |     ctx->out_of_space = false; | 
					
						
							|  |  |  |     ctx->contradiction = false; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     _PyUOpInstruction *this_instr = NULL; | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  |     for (int i = 0; !ctx->done; i++) { | 
					
						
							|  |  |  |         assert(i < trace_len); | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |         this_instr = &trace[i]; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         int oparg = this_instr->oparg; | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |         opcode = this_instr->opcode; | 
					
						
							| 
									
										
										
										
											2025-06-17 23:25:53 +08:00
										 |  |  |         JitOptRef *stack_pointer = ctx->frame->stack_pointer; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-18 11:08:43 -07:00
										 |  |  | #ifdef Py_DEBUG
 | 
					
						
							|  |  |  |         if (get_lltrace() >= 3) { | 
					
						
							|  |  |  |             printf("%4d abs: ", (int)(this_instr - trace)); | 
					
						
							|  |  |  |             _PyUOpPrint(this_instr); | 
					
						
							|  |  |  |             printf(" "); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |         switch (opcode) { | 
					
						
							| 
									
										
										
										
											2024-03-18 11:08:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-26 08:42:53 -08:00
										 |  |  | #include "optimizer_cases.c.h"
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             default: | 
					
						
							| 
									
										
										
										
											2024-03-18 11:08:43 -07:00
										 |  |  |                 DPRINTF(1, "\nUnknown opcode in abstract interpreter\n"); | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |                 Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |         assert(ctx->frame != NULL); | 
					
						
							|  |  |  |         DPRINTF(3, " stack_level %d\n", STACK_LEVEL()); | 
					
						
							|  |  |  |         ctx->frame->stack_pointer = stack_pointer; | 
					
						
							|  |  |  |         assert(STACK_LEVEL() >= 0); | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  |     if (ctx->out_of_space) { | 
					
						
							|  |  |  |         DPRINTF(3, "\n"); | 
					
						
							|  |  |  |         DPRINTF(1, "Out of space in abstract interpreter\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ctx->contradiction) { | 
					
						
							|  |  |  |         // Attempted to push a "bottom" (contradiction) symbol onto the stack.
 | 
					
						
							|  |  |  |         // This means that the abstract interpreter has hit unreachable code.
 | 
					
						
							|  |  |  |         // We *could* generate an _EXIT_TRACE or _FATAL_ERROR here, but hitting
 | 
					
						
							|  |  |  |         // bottom indicates type instability, so we are probably better off
 | 
					
						
							|  |  |  |         // retrying later.
 | 
					
						
							|  |  |  |         DPRINTF(3, "\n"); | 
					
						
							|  |  |  |         DPRINTF(1, "Hit bottom in abstract interpreter\n"); | 
					
						
							|  |  |  |         _Py_uop_abstractcontext_fini(ctx); | 
					
						
							|  |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-29 10:55:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |     /* Either reached the end or cannot optimize further, but there
 | 
					
						
							|  |  |  |      * would be no benefit in retrying later */ | 
					
						
							| 
									
										
										
										
											2024-03-20 18:24:02 +00:00
										 |  |  |     _Py_uop_abstractcontext_fini(ctx); | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |     if (first_valid_check_stack != NULL) { | 
					
						
							|  |  |  |         assert(first_valid_check_stack->opcode == _CHECK_STACK_SPACE); | 
					
						
							|  |  |  |         assert(max_space > 0); | 
					
						
							|  |  |  |         assert(max_space <= INT_MAX); | 
					
						
							|  |  |  |         assert(max_space <= INT32_MAX); | 
					
						
							|  |  |  |         first_valid_check_stack->opcode = _CHECK_STACK_SPACE_OPERAND; | 
					
						
							| 
									
										
										
										
											2024-11-09 11:35:33 +08:00
										 |  |  |         first_valid_check_stack->operand0 = max_space; | 
					
						
							| 
									
										
										
										
											2024-04-18 11:09:30 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     return trace_len; | 
					
						
							| 
									
										
										
										
											2024-05-10 17:43:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  |     DPRINTF(3, "\n"); | 
					
						
							|  |  |  |     DPRINTF(1, "Encountered error in abstract interpreter\n"); | 
					
						
							|  |  |  |     if (opcode <= MAX_UOP_ID) { | 
					
						
							|  |  |  |         OPT_ERROR_IN_OPCODE(opcode); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     _Py_uop_abstractcontext_fini(ctx); | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  | const uint16_t op_without_push[MAX_UOP_ID + 1] = { | 
					
						
							|  |  |  |     [_COPY] = _NOP, | 
					
						
							|  |  |  |     [_LOAD_CONST_INLINE] = _NOP, | 
					
						
							|  |  |  |     [_LOAD_CONST_INLINE_BORROW] = _NOP, | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  |     [_LOAD_CONST_UNDER_INLINE] = _POP_TOP_LOAD_CONST_INLINE, | 
					
						
							|  |  |  |     [_LOAD_CONST_UNDER_INLINE_BORROW] = _POP_TOP_LOAD_CONST_INLINE_BORROW, | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |     [_LOAD_FAST] = _NOP, | 
					
						
							|  |  |  |     [_LOAD_FAST_BORROW] = _NOP, | 
					
						
							|  |  |  |     [_LOAD_SMALL_INT] = _NOP, | 
					
						
							|  |  |  |     [_POP_TOP_LOAD_CONST_INLINE] = _POP_TOP, | 
					
						
							|  |  |  |     [_POP_TOP_LOAD_CONST_INLINE_BORROW] = _POP_TOP, | 
					
						
							|  |  |  |     [_POP_TWO_LOAD_CONST_INLINE_BORROW] = _POP_TWO, | 
					
						
							| 
									
										
										
										
											2025-05-22 12:52:47 -04:00
										 |  |  |     [_POP_CALL_TWO_LOAD_CONST_INLINE_BORROW] = _POP_CALL_TWO, | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  | const bool op_skip[MAX_UOP_ID + 1] = { | 
					
						
							|  |  |  |     [_NOP] = true, | 
					
						
							|  |  |  |     [_CHECK_VALIDITY] = true, | 
					
						
							| 
									
										
										
										
											2025-05-23 08:48:45 -04:00
										 |  |  |     [_CHECK_PERIODIC] = true, | 
					
						
							| 
									
										
										
										
											2025-05-22 12:52:47 -04:00
										 |  |  |     [_SET_IP] = true, | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  | const uint16_t op_without_pop[MAX_UOP_ID + 1] = { | 
					
						
							|  |  |  |     [_POP_TOP] = _NOP, | 
					
						
							|  |  |  |     [_POP_TOP_LOAD_CONST_INLINE] = _LOAD_CONST_INLINE, | 
					
						
							|  |  |  |     [_POP_TOP_LOAD_CONST_INLINE_BORROW] = _LOAD_CONST_INLINE_BORROW, | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  |     [_POP_TWO] = _POP_TOP, | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |     [_POP_TWO_LOAD_CONST_INLINE_BORROW] = _POP_TOP_LOAD_CONST_INLINE_BORROW, | 
					
						
							| 
									
										
										
										
											2025-05-22 12:52:47 -04:00
										 |  |  |     [_POP_CALL_TWO_LOAD_CONST_INLINE_BORROW] = _POP_CALL_ONE_LOAD_CONST_INLINE_BORROW, | 
					
						
							|  |  |  |     [_POP_CALL_ONE_LOAD_CONST_INLINE_BORROW] = _POP_CALL_LOAD_CONST_INLINE_BORROW, | 
					
						
							|  |  |  |     [_POP_CALL_TWO] = _POP_CALL_ONE, | 
					
						
							|  |  |  |     [_POP_CALL_ONE] = _POP_CALL, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const uint16_t op_without_pop_null[MAX_UOP_ID + 1] = { | 
					
						
							|  |  |  |     [_POP_CALL] = _POP_TOP, | 
					
						
							|  |  |  |     [_POP_CALL_LOAD_CONST_INLINE_BORROW] = _POP_TOP_LOAD_CONST_INLINE_BORROW, | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | remove_unneeded_uops(_PyUOpInstruction *buffer, int buffer_size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |     /* Remove _SET_IP and _CHECK_VALIDITY where possible.
 | 
					
						
							|  |  |  |      * _SET_IP is needed if the following instruction escapes or | 
					
						
							|  |  |  |      * could error. _CHECK_VALIDITY is needed if the previous | 
					
						
							|  |  |  |      * instruction could have escaped. */ | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     int last_set_ip = -1; | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  |     bool may_have_escaped = true; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     for (int pc = 0; pc < buffer_size; pc++) { | 
					
						
							|  |  |  |         int opcode = buffer[pc].opcode; | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |         switch (opcode) { | 
					
						
							| 
									
										
										
										
											2024-04-19 09:26:42 +01:00
										 |  |  |             case _START_EXECUTOR: | 
					
						
							|  |  |  |                 may_have_escaped = false; | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |             case _SET_IP: | 
					
						
							| 
									
										
										
										
											2024-03-11 13:37:48 +00:00
										 |  |  |                 buffer[pc].opcode = _NOP; | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |                 last_set_ip = pc; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case _CHECK_VALIDITY: | 
					
						
							|  |  |  |                 if (may_have_escaped) { | 
					
						
							|  |  |  |                     may_have_escaped = false; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |                 else { | 
					
						
							| 
									
										
										
										
											2024-03-11 13:37:48 +00:00
										 |  |  |                     buffer[pc].opcode = _NOP; | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |             default: | 
					
						
							| 
									
										
										
										
											2024-02-22 14:48:25 +00:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |                 // Cancel out pushes and pops, repeatedly. So:
 | 
					
						
							|  |  |  |                 //     _LOAD_FAST + _POP_TWO_LOAD_CONST_INLINE_BORROW + _POP_TOP
 | 
					
						
							|  |  |  |                 // ...becomes:
 | 
					
						
							|  |  |  |                 //     _NOP + _POP_TOP + _NOP
 | 
					
						
							| 
									
										
										
										
											2025-05-22 12:52:47 -04:00
										 |  |  |                 while (op_without_pop[opcode] || op_without_pop_null[opcode]) { | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |                     _PyUOpInstruction *last = &buffer[pc - 1]; | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  |                     while (op_skip[last->opcode]) { | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |                         last--; | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2025-05-23 08:48:45 -04:00
										 |  |  |                     if (op_without_push[last->opcode] && op_without_pop[opcode]) { | 
					
						
							| 
									
										
										
										
											2025-05-22 12:52:47 -04:00
										 |  |  |                         last->opcode = op_without_push[last->opcode]; | 
					
						
							|  |  |  |                         opcode = buffer[pc].opcode = op_without_pop[opcode]; | 
					
						
							|  |  |  |                         if (op_without_pop[last->opcode]) { | 
					
						
							|  |  |  |                             opcode = last->opcode; | 
					
						
							|  |  |  |                             pc = last - buffer; | 
					
						
							|  |  |  |                         } | 
					
						
							| 
									
										
										
										
											2025-05-20 18:09:51 -04:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2025-05-22 12:52:47 -04:00
										 |  |  |                     else if (last->opcode == _PUSH_NULL) { | 
					
						
							|  |  |  |                         // Handle _POP_CALL and _POP_CALL_LOAD_CONST_INLINE_BORROW separately.
 | 
					
						
							|  |  |  |                         // This looks for a preceding _PUSH_NULL instruction and
 | 
					
						
							|  |  |  |                         // simplifies to _POP_TOP(_LOAD_CONST_INLINE_BORROW).
 | 
					
						
							|  |  |  |                         last->opcode = _NOP; | 
					
						
							|  |  |  |                         opcode = buffer[pc].opcode = op_without_pop_null[opcode]; | 
					
						
							|  |  |  |                         assert(opcode); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     else { | 
					
						
							|  |  |  |                         break; | 
					
						
							| 
									
										
										
										
											2025-05-22 11:15:03 -04:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2024-03-05 15:23:08 +00:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-04-19 09:25:07 +01:00
										 |  |  |                 /* _PUSH_FRAME doesn't escape or error, but it
 | 
					
						
							|  |  |  |                  * does need the IP for the return address */ | 
					
						
							|  |  |  |                 bool needs_ip = opcode == _PUSH_FRAME; | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |                 if (_PyUop_Flags[opcode] & HAS_ESCAPES_FLAG) { | 
					
						
							|  |  |  |                     needs_ip = true; | 
					
						
							|  |  |  |                     may_have_escaped = true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (needs_ip && last_set_ip >= 0) { | 
					
						
							| 
									
										
										
										
											2025-04-01 16:55:05 -07:00
										 |  |  |                     assert(buffer[last_set_ip].opcode == _NOP); | 
					
						
							|  |  |  |                     buffer[last_set_ip].opcode = _SET_IP; | 
					
						
							| 
									
										
										
										
											2024-02-13 16:28:19 +00:00
										 |  |  |                     last_set_ip = -1; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-04-21 09:58:55 -07:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2025-04-21 09:58:55 -07:00
										 |  |  |             case _JUMP_TO_TOP: | 
					
						
							|  |  |  |             case _EXIT_TRACE: | 
					
						
							|  |  |  |                 return pc + 1; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //  0 - failure, no error raised, just fall back to Tier 1
 | 
					
						
							|  |  |  | // -1 - failure, and raise error
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  | //  > 0 - length of optimized trace
 | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | int | 
					
						
							|  |  |  | _Py_uop_analyze_and_optimize( | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     _PyInterpreterFrame *frame, | 
					
						
							| 
									
										
										
										
											2023-11-06 11:28:52 +00:00
										 |  |  |     _PyUOpInstruction *buffer, | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     int length, | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     int curr_stacklen, | 
					
						
							|  |  |  |     _PyBloomFilter *dependencies | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     OPT_STAT_INC(optimizer_attempts); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     int err = remove_globals(frame, buffer, length, dependencies); | 
					
						
							|  |  |  |     if (err <= 0) { | 
					
						
							|  |  |  |         return err; | 
					
						
							| 
									
										
										
										
											2024-02-02 12:14:34 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     length = optimize_uops( | 
					
						
							| 
									
										
											  
											
												gh-117045: Add code object to function version cache (#117028)
Changes to the function version cache:
- In addition to the function object, also store the code object,
  and allow the latter to be retrieved even if the function has been evicted.
- Stop assigning new function versions after a critical attribute (e.g. `__code__`)
  has been modified; the version is permanently reset to zero in this case.
- Changes to `__annotations__` are no longer considered critical. (This fixes gh-109998.)
Changes to the Tier 2 optimization machinery:
- If we cannot map a function version to a function, but it is still mapped to a code object,
  we continue projecting the trace.
  The operand of the `_PUSH_FRAME` and `_POP_FRAME` opcodes can be either NULL,
  a function object, or a code object with the lowest bit set.
This allows us to trace through code that calls an ephemeral function,
i.e., a function that may not be alive when we are constructing the executor,
e.g. a generator expression or certain nested functions.
We will lose globals removal inside such functions,
but we can still do other peephole operations
(and even possibly [call inlining](https://github.com/python/cpython/pull/116290),
if we decide to do it), which only need the code object.
As before, if we cannot retrieve the code object from the cache, we stop projecting.
											
										 
											2024-03-21 12:37:41 -07:00
										 |  |  |         _PyFrame_GetCode(frame), buffer, | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |         length, curr_stacklen, dependencies); | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     if (length <= 0) { | 
					
						
							|  |  |  |         return length; | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     length = remove_unneeded_uops(buffer, length); | 
					
						
							|  |  |  |     assert(length > 0); | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     OPT_STAT_INC(optimizer_successes); | 
					
						
							| 
									
										
										
										
											2024-03-26 09:35:11 +00:00
										 |  |  |     return length; | 
					
						
							| 
									
										
										
										
											2023-08-16 02:04:17 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-04-30 18:26:34 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* _Py_TIER2 */
 |