| 
									
										
										
										
											2021-06-07 18:38:06 +01:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-23 18:39:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "opcode.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-07 18:38:06 +01:00
										 |  |  | #include "pycore_code.h"
 | 
					
						
							| 
									
										
										
										
											2023-07-22 22:57:59 +02:00
										 |  |  | #include "pycore_descrobject.h"   // _PyMethodWrapper_Type
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | #include "pycore_dict.h"          // DICT_KEYS_UNICODE
 | 
					
						
							| 
									
										
										
										
											2022-02-25 16:07:14 +01:00
										 |  |  | #include "pycore_function.h"      // _PyFunction_GetVersionForCurrentState()
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | #include "pycore_long.h"          // _PyLong_IsNonNegativeCompact()
 | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | #include "pycore_moduleobject.h"
 | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  | #include "pycore_object.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-23 18:39:00 +01:00
										 |  |  | #include "pycore_opcode_metadata.h" // _PyOpcode_Caches
 | 
					
						
							| 
									
										
										
										
											2023-12-20 14:27:25 +00:00
										 |  |  | #include "pycore_uop_metadata.h"    // _PyOpcode_uop_name
 | 
					
						
							| 
									
										
										
										
											2024-04-16 08:27:18 -05:00
										 |  |  | #include "pycore_uop_ids.h"       // MAX_UOP_ID
 | 
					
						
							| 
									
										
										
										
											2023-10-26 16:30:18 +01:00
										 |  |  | #include "pycore_opcode_utils.h"  // RESUME_AT_FUNC_START
 | 
					
						
							| 
									
										
										
										
											2023-07-04 14:42:12 -07:00
										 |  |  | #include "pycore_pylifecycle.h"   // _PyOS_URandomNonblock()
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | #include "pycore_runtime.h"       // _Py_ID()
 | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-18 17:31:18 +09:00
										 |  |  | #include <stdlib.h> // rand()
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-20 12:24:35 -08:00
										 |  |  | extern const char *_PyUOpName(int index); | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 13:13:05 +01:00
										 |  |  | /* For guidance on adding or extending families of instructions see
 | 
					
						
							|  |  |  |  * ./adaptive.md | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							| 
									
										
										
										
											2023-08-04 10:34:23 +01:00
										 |  |  | GCStats _py_gc_stats[NUM_GENERATIONS] = { 0 }; | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | static PyStats _Py_stats_struct = { .gc_stats = _py_gc_stats }; | 
					
						
							|  |  |  | PyStats *_Py_stats = NULL; | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-16 18:38:29 -04:00
										 |  |  | #if PYSTATS_MAX_UOP_ID < MAX_UOP_ID
 | 
					
						
							|  |  |  | #error "Not enough space allocated for pystats. Increase PYSTATS_MAX_UOP_ID to at least MAX_UOP_ID"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  | #define ADD_STAT_TO_DICT(res, field) \
 | 
					
						
							|  |  |  |     do { \ | 
					
						
							|  |  |  |         PyObject *val = PyLong_FromUnsignedLongLong(stats->field); \ | 
					
						
							|  |  |  |         if (val == NULL) { \ | 
					
						
							|  |  |  |             Py_DECREF(res); \ | 
					
						
							|  |  |  |             return NULL; \ | 
					
						
							|  |  |  |         } \ | 
					
						
							|  |  |  |         if (PyDict_SetItemString(res, #field, val) == -1) { \ | 
					
						
							|  |  |  |             Py_DECREF(res); \ | 
					
						
							|  |  |  |             Py_DECREF(val); \ | 
					
						
							|  |  |  |             return NULL; \ | 
					
						
							|  |  |  |         } \ | 
					
						
							|  |  |  |         Py_DECREF(val); \ | 
					
						
							|  |  |  |     } while(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject* | 
					
						
							|  |  |  | stats_to_dict(SpecializationStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *res = PyDict_New(); | 
					
						
							|  |  |  |     if (res == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     ADD_STAT_TO_DICT(res, success); | 
					
						
							|  |  |  |     ADD_STAT_TO_DICT(res, failure); | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |     ADD_STAT_TO_DICT(res, hit); | 
					
						
							|  |  |  |     ADD_STAT_TO_DICT(res, deferred); | 
					
						
							|  |  |  |     ADD_STAT_TO_DICT(res, miss); | 
					
						
							|  |  |  |     ADD_STAT_TO_DICT(res, deopt); | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |     PyObject *failure_kinds = PyTuple_New(SPECIALIZATION_FAILURE_KINDS); | 
					
						
							|  |  |  |     if (failure_kinds == NULL) { | 
					
						
							|  |  |  |         Py_DECREF(res); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (int i = 0; i < SPECIALIZATION_FAILURE_KINDS; i++) { | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |         PyObject *stat = PyLong_FromUnsignedLongLong(stats->failure_kinds[i]); | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |         if (stat == NULL) { | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |             Py_DECREF(res); | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |             Py_DECREF(failure_kinds); | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |         PyTuple_SET_ITEM(failure_kinds, i, stat); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     if (PyDict_SetItemString(res, "failure_kinds", failure_kinds)) { | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |         Py_DECREF(res); | 
					
						
							|  |  |  |         Py_DECREF(failure_kinds); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |     Py_DECREF(failure_kinds); | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #undef ADD_STAT_TO_DICT
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | add_stat_dict( | 
					
						
							|  |  |  |     PyObject *res, | 
					
						
							|  |  |  |     int opcode, | 
					
						
							|  |  |  |     const char *name) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     SpecializationStats *stats = &_Py_stats_struct.opcode_stats[opcode].specialization; | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |     PyObject *d = stats_to_dict(stats); | 
					
						
							|  |  |  |     if (d == NULL) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int err = PyDict_SetItemString(res, name, d); | 
					
						
							|  |  |  |     Py_DECREF(d); | 
					
						
							|  |  |  |     return err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyObject* | 
					
						
							|  |  |  | _Py_GetSpecializationStats(void) { | 
					
						
							|  |  |  |     PyObject *stats = PyDict_New(); | 
					
						
							|  |  |  |     if (stats == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int err = 0; | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     err += add_stat_dict(stats, CONTAINS_OP, "contains_op"); | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  |     err += add_stat_dict(stats, LOAD_SUPER_ATTR, "load_super_attr"); | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |     err += add_stat_dict(stats, LOAD_ATTR, "load_attr"); | 
					
						
							|  |  |  |     err += add_stat_dict(stats, LOAD_GLOBAL, "load_global"); | 
					
						
							|  |  |  |     err += add_stat_dict(stats, BINARY_SUBSCR, "binary_subscr"); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |     err += add_stat_dict(stats, STORE_SUBSCR, "store_subscr"); | 
					
						
							| 
									
										
										
										
											2021-08-11 17:34:01 +01:00
										 |  |  |     err += add_stat_dict(stats, STORE_ATTR, "store_attr"); | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |     err += add_stat_dict(stats, CALL, "call"); | 
					
						
							| 
									
										
										
										
											2024-08-22 14:17:10 +01:00
										 |  |  |     err += add_stat_dict(stats, CALL_KW, "call_kw"); | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |     err += add_stat_dict(stats, BINARY_OP, "binary_op"); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |     err += add_stat_dict(stats, COMPARE_OP, "compare_op"); | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |     err += add_stat_dict(stats, UNPACK_SEQUENCE, "unpack_sequence"); | 
					
						
							| 
									
										
										
										
											2022-09-09 14:42:29 -07:00
										 |  |  |     err += add_stat_dict(stats, FOR_ITER, "for_iter"); | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  |     err += add_stat_dict(stats, TO_BOOL, "to_bool"); | 
					
						
							|  |  |  |     err += add_stat_dict(stats, SEND, "send"); | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  |     if (err < 0) { | 
					
						
							|  |  |  |         Py_DECREF(stats); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return stats; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  | #define PRINT_STAT(i, field) \
 | 
					
						
							|  |  |  |     if (stats[i].field) { \ | 
					
						
							| 
									
										
										
										
											2023-09-12 17:12:57 -04:00
										 |  |  |         fprintf(out, "    opcode[%s]." #field " : %" PRIu64 "\n", _PyOpcode_OpName[i], stats[i].field); \ | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  | print_spec_stats(FILE *out, OpcodeStats *stats) | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-04 09:56:46 +00:00
										 |  |  |     /* Mark some opcodes as specializable for stats,
 | 
					
						
							|  |  |  |      * even though we don't specialize them yet. */ | 
					
						
							| 
									
										
										
										
											2023-09-12 17:12:57 -04:00
										 |  |  |     fprintf(out, "opcode[BINARY_SLICE].specializable : 1\n"); | 
					
						
							|  |  |  |     fprintf(out, "opcode[STORE_SLICE].specializable : 1\n"); | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     for (int i = 0; i < 256; i++) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:46:03 +01:00
										 |  |  |         if (_PyOpcode_Caches[i]) { | 
					
						
							|  |  |  |             /* Ignore jumps as they cannot be specialized */ | 
					
						
							|  |  |  |             switch (i) { | 
					
						
							|  |  |  |                 case POP_JUMP_IF_FALSE: | 
					
						
							|  |  |  |                 case POP_JUMP_IF_TRUE: | 
					
						
							|  |  |  |                 case POP_JUMP_IF_NONE: | 
					
						
							|  |  |  |                 case POP_JUMP_IF_NOT_NONE: | 
					
						
							|  |  |  |                 case JUMP_BACKWARD: | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     fprintf(out, "opcode[%s].specializable : 1\n", _PyOpcode_OpName[i]); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-02-04 09:56:46 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |         PRINT_STAT(i, specialization.success); | 
					
						
							|  |  |  |         PRINT_STAT(i, specialization.failure); | 
					
						
							|  |  |  |         PRINT_STAT(i, specialization.hit); | 
					
						
							|  |  |  |         PRINT_STAT(i, specialization.deferred); | 
					
						
							|  |  |  |         PRINT_STAT(i, specialization.miss); | 
					
						
							|  |  |  |         PRINT_STAT(i, specialization.deopt); | 
					
						
							|  |  |  |         PRINT_STAT(i, execution_count); | 
					
						
							|  |  |  |         for (int j = 0; j < SPECIALIZATION_FAILURE_KINDS; j++) { | 
					
						
							|  |  |  |             uint64_t val = stats[i].specialization.failure_kinds[j]; | 
					
						
							|  |  |  |             if (val) { | 
					
						
							| 
									
										
										
										
											2023-09-12 17:12:57 -04:00
										 |  |  |                 fprintf(out, "    opcode[%s].specialization.failure_kinds[%d] : %" | 
					
						
							|  |  |  |                     PRIu64 "\n", _PyOpcode_OpName[i], j, val); | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-04-29 05:51:55 +01:00
										 |  |  |         for (int j = 0; j < 256; j++) { | 
					
						
							| 
									
										
										
										
											2022-02-14 15:53:38 +00:00
										 |  |  |             if (stats[i].pair_count[j]) { | 
					
						
							| 
									
										
										
										
											2023-09-12 17:12:57 -04:00
										 |  |  |                 fprintf(out, "opcode[%s].pair_count[%s] : %" PRIu64 "\n", | 
					
						
							|  |  |  |                         _PyOpcode_OpName[i], _PyOpcode_OpName[j], stats[i].pair_count[j]); | 
					
						
							| 
									
										
										
										
											2022-02-14 15:53:38 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-15 13:01:42 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-07-29 17:26:53 +01:00
										 |  |  | #undef PRINT_STAT
 | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-28 15:20:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | print_call_stats(FILE *out, CallStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fprintf(out, "Calls to PyEval_EvalDefault: %" PRIu64 "\n", stats->pyeval_calls); | 
					
						
							|  |  |  |     fprintf(out, "Calls to Python functions inlined: %" PRIu64 "\n", stats->inlined_py_calls); | 
					
						
							| 
									
										
										
										
											2022-02-02 11:01:33 +00:00
										 |  |  |     fprintf(out, "Frames pushed: %" PRIu64 "\n", stats->frames_pushed); | 
					
						
							|  |  |  |     fprintf(out, "Frame objects created: %" PRIu64 "\n", stats->frame_objects_created); | 
					
						
							| 
									
										
										
										
											2022-05-27 16:31:41 +01:00
										 |  |  |     for (int i = 0; i < EVAL_CALL_KINDS; i++) { | 
					
						
							|  |  |  |         fprintf(out, "Calls via PyEval_EvalFrame[%d] : %" PRIu64 "\n", i, stats->eval_calls[i]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-28 15:20:33 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-01 15:05:18 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | print_object_stats(FILE *out, ObjectStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-05-03 16:40:24 -06:00
										 |  |  |     fprintf(out, "Object allocations from freelist: %" PRIu64 "\n", stats->from_freelist); | 
					
						
							|  |  |  |     fprintf(out, "Object frees to freelist: %" PRIu64 "\n", stats->to_freelist); | 
					
						
							| 
									
										
										
										
											2022-02-01 15:05:18 +00:00
										 |  |  |     fprintf(out, "Object allocations: %" PRIu64 "\n", stats->allocations); | 
					
						
							| 
									
										
										
										
											2022-05-03 16:40:24 -06:00
										 |  |  |     fprintf(out, "Object allocations to 512 bytes: %" PRIu64 "\n", stats->allocations512); | 
					
						
							|  |  |  |     fprintf(out, "Object allocations to 4 kbytes: %" PRIu64 "\n", stats->allocations4k); | 
					
						
							|  |  |  |     fprintf(out, "Object allocations over 4 kbytes: %" PRIu64 "\n", stats->allocations_big); | 
					
						
							| 
									
										
										
										
											2022-02-01 15:05:18 +00:00
										 |  |  |     fprintf(out, "Object frees: %" PRIu64 "\n", stats->frees); | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |     fprintf(out, "Object inline values: %" PRIu64 "\n", stats->inline_values); | 
					
						
							| 
									
										
										
										
											2022-05-18 14:38:43 +01:00
										 |  |  |     fprintf(out, "Object interpreter increfs: %" PRIu64 "\n", stats->interpreter_increfs); | 
					
						
							|  |  |  |     fprintf(out, "Object interpreter decrefs: %" PRIu64 "\n", stats->interpreter_decrefs); | 
					
						
							| 
									
										
										
										
											2022-05-16 14:35:11 +01:00
										 |  |  |     fprintf(out, "Object increfs: %" PRIu64 "\n", stats->increfs); | 
					
						
							|  |  |  |     fprintf(out, "Object decrefs: %" PRIu64 "\n", stats->decrefs); | 
					
						
							| 
									
										
										
										
											2022-02-01 15:05:18 +00:00
										 |  |  |     fprintf(out, "Object materialize dict (on request): %" PRIu64 "\n", stats->dict_materialized_on_request); | 
					
						
							|  |  |  |     fprintf(out, "Object materialize dict (new key): %" PRIu64 "\n", stats->dict_materialized_new_key); | 
					
						
							|  |  |  |     fprintf(out, "Object materialize dict (too big): %" PRIu64 "\n", stats->dict_materialized_too_big); | 
					
						
							| 
									
										
										
										
											2022-03-04 11:31:29 +00:00
										 |  |  |     fprintf(out, "Object materialize dict (str subclass): %" PRIu64 "\n", stats->dict_materialized_str_subclass); | 
					
						
							| 
									
										
										
										
											2022-12-15 09:45:03 +00:00
										 |  |  |     fprintf(out, "Object method cache hits: %" PRIu64 "\n", stats->type_cache_hits); | 
					
						
							|  |  |  |     fprintf(out, "Object method cache misses: %" PRIu64 "\n", stats->type_cache_misses); | 
					
						
							|  |  |  |     fprintf(out, "Object method cache collisions: %" PRIu64 "\n", stats->type_cache_collisions); | 
					
						
							|  |  |  |     fprintf(out, "Object method cache dunder hits: %" PRIu64 "\n", stats->type_cache_dunder_hits); | 
					
						
							|  |  |  |     fprintf(out, "Object method cache dunder misses: %" PRIu64 "\n", stats->type_cache_dunder_misses); | 
					
						
							| 
									
										
										
										
											2022-02-01 15:05:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-04 10:34:23 +01:00
										 |  |  | static void | 
					
						
							|  |  |  | print_gc_stats(FILE *out, GCStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int i = 0; i < NUM_GENERATIONS; i++) { | 
					
						
							|  |  |  |         fprintf(out, "GC[%d] collections: %" PRIu64 "\n", i, stats[i].collections); | 
					
						
							|  |  |  |         fprintf(out, "GC[%d] object visits: %" PRIu64 "\n", i, stats[i].object_visits); | 
					
						
							|  |  |  |         fprintf(out, "GC[%d] objects collected: %" PRIu64 "\n", i, stats[i].objects_collected); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-02 16:17:59 +01:00
										 |  |  | #ifdef _Py_TIER2
 | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  | static void | 
					
						
							|  |  |  | print_histogram(FILE *out, const char *name, uint64_t hist[_Py_UOP_HIST_SIZE]) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int i = 0; i < _Py_UOP_HIST_SIZE; i++) { | 
					
						
							|  |  |  |         fprintf(out, "%s[%" PRIu64"]: %" PRIu64 "\n", name, (uint64_t)1 << i, hist[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | print_optimization_stats(FILE *out, OptimizationStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fprintf(out, "Optimization attempts: %" PRIu64 "\n", stats->attempts); | 
					
						
							|  |  |  |     fprintf(out, "Optimization traces created: %" PRIu64 "\n", stats->traces_created); | 
					
						
							|  |  |  |     fprintf(out, "Optimization traces executed: %" PRIu64 "\n", stats->traces_executed); | 
					
						
							|  |  |  |     fprintf(out, "Optimization uops executed: %" PRIu64 "\n", stats->uops_executed); | 
					
						
							|  |  |  |     fprintf(out, "Optimization trace stack overflow: %" PRIu64 "\n", stats->trace_stack_overflow); | 
					
						
							|  |  |  |     fprintf(out, "Optimization trace stack underflow: %" PRIu64 "\n", stats->trace_stack_underflow); | 
					
						
							|  |  |  |     fprintf(out, "Optimization trace too long: %" PRIu64 "\n", stats->trace_too_long); | 
					
						
							| 
									
										
										
										
											2023-10-05 11:12:06 -04:00
										 |  |  |     fprintf(out, "Optimization trace too short: %" PRIu64 "\n", stats->trace_too_short); | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  |     fprintf(out, "Optimization inner loop: %" PRIu64 "\n", stats->inner_loop); | 
					
						
							|  |  |  |     fprintf(out, "Optimization recursive call: %" PRIu64 "\n", stats->recursive_call); | 
					
						
							| 
									
										
										
										
											2023-12-12 13:43:08 -08:00
										 |  |  |     fprintf(out, "Optimization low confidence: %" PRIu64 "\n", stats->low_confidence); | 
					
						
							| 
									
										
										
										
											2024-02-26 12:51:47 -05:00
										 |  |  |     fprintf(out, "Executors invalidated: %" PRIu64 "\n", stats->executors_invalidated); | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     print_histogram(out, "Trace length", stats->trace_length_hist); | 
					
						
							|  |  |  |     print_histogram(out, "Trace run length", stats->trace_run_length_hist); | 
					
						
							|  |  |  |     print_histogram(out, "Optimized trace length", stats->optimized_trace_length_hist); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 21:24:48 +08:00
										 |  |  |     fprintf(out, "Optimization optimizer attempts: %" PRIu64 "\n", stats->optimizer_attempts); | 
					
						
							|  |  |  |     fprintf(out, "Optimization optimizer successes: %" PRIu64 "\n", stats->optimizer_successes); | 
					
						
							|  |  |  |     fprintf(out, "Optimization optimizer failure no memory: %" PRIu64 "\n", | 
					
						
							|  |  |  |             stats->optimizer_failure_reason_no_memory); | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |     fprintf(out, "Optimizer remove globals builtins changed: %" PRIu64 "\n", stats->remove_globals_builtins_changed); | 
					
						
							|  |  |  |     fprintf(out, "Optimizer remove globals incorrect keys: %" PRIu64 "\n", stats->remove_globals_incorrect_keys); | 
					
						
							| 
									
										
										
										
											2024-02-20 12:24:35 -08:00
										 |  |  |     for (int i = 0; i <= MAX_UOP_ID; i++) { | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  |         if (stats->opcode[i].execution_count) { | 
					
						
							| 
									
										
										
										
											2024-02-20 12:24:35 -08:00
										 |  |  |             fprintf(out, "uops[%s].execution_count : %" PRIu64 "\n", _PyUOpName(i), stats->opcode[i].execution_count); | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-10-30 20:02:45 -04:00
										 |  |  |         if (stats->opcode[i].miss) { | 
					
						
							| 
									
										
										
										
											2024-02-20 12:24:35 -08:00
										 |  |  |             fprintf(out, "uops[%s].specialization.miss : %" PRIu64 "\n", _PyUOpName(i), stats->opcode[i].miss); | 
					
						
							| 
									
										
										
										
											2023-10-30 20:02:45 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     for (int i = 0; i < 256; i++) { | 
					
						
							|  |  |  |         if (stats->unsupported_opcode[i]) { | 
					
						
							|  |  |  |             fprintf( | 
					
						
							|  |  |  |                 out, | 
					
						
							|  |  |  |                 "unsupported_opcode[%s].count : %" PRIu64 "\n", | 
					
						
							|  |  |  |                 _PyOpcode_OpName[i], | 
					
						
							|  |  |  |                 stats->unsupported_opcode[i] | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-16 08:27:18 -05:00
										 |  |  |     for (int i = 1; i <= MAX_UOP_ID; i++){ | 
					
						
							|  |  |  |         for (int j = 1; j <= MAX_UOP_ID; j++) { | 
					
						
							|  |  |  |             if (stats->opcode[i].pair_count[j]) { | 
					
						
							|  |  |  |                 fprintf(out, "uop[%s].pair_count[%s] : %" PRIu64 "\n", | 
					
						
							|  |  |  |                         _PyOpcode_uop_name[i], _PyOpcode_uop_name[j], stats->opcode[i].pair_count[j]); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-21 13:27:46 -04:00
										 |  |  |     for (int i = 0; i < MAX_UOP_ID; i++) { | 
					
						
							|  |  |  |         if (stats->error_in_opcode[i]) { | 
					
						
							|  |  |  |             fprintf( | 
					
						
							|  |  |  |                 out, | 
					
						
							|  |  |  |                 "error_in_opcode[%s].count : %" PRIu64 "\n", | 
					
						
							|  |  |  |                 _PyUOpName(i), | 
					
						
							|  |  |  |                 stats->error_in_opcode[i] | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-05-02 16:17:59 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 06:10:51 -05:00
										 |  |  | static void | 
					
						
							|  |  |  | print_rare_event_stats(FILE *out, RareEventStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fprintf(out, "Rare event (set_class): %" PRIu64 "\n", stats->set_class); | 
					
						
							|  |  |  |     fprintf(out, "Rare event (set_bases): %" PRIu64 "\n", stats->set_bases); | 
					
						
							|  |  |  |     fprintf(out, "Rare event (set_eval_frame_func): %" PRIu64 "\n", stats->set_eval_frame_func); | 
					
						
							|  |  |  |     fprintf(out, "Rare event (builtin_dict): %" PRIu64 "\n", stats->builtin_dict); | 
					
						
							|  |  |  |     fprintf(out, "Rare event (func_modification): %" PRIu64 "\n", stats->func_modification); | 
					
						
							| 
									
										
										
										
											2024-02-12 16:07:38 +00:00
										 |  |  |     fprintf(out, "Rare event (watched_dict_modification): %" PRIu64 "\n", stats->watched_dict_modification); | 
					
						
							|  |  |  |     fprintf(out, "Rare event (watched_globals_modification): %" PRIu64 "\n", stats->watched_globals_modification); | 
					
						
							| 
									
										
										
										
											2024-01-25 06:10:51 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | print_stats(FILE *out, PyStats *stats) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     print_spec_stats(out, stats->opcode_stats); | 
					
						
							| 
									
										
										
										
											2022-01-28 15:20:33 +00:00
										 |  |  |     print_call_stats(out, &stats->call_stats); | 
					
						
							| 
									
										
										
										
											2022-02-01 15:05:18 +00:00
										 |  |  |     print_object_stats(out, &stats->object_stats); | 
					
						
							| 
									
										
										
										
											2023-08-04 10:34:23 +01:00
										 |  |  |     print_gc_stats(out, stats->gc_stats); | 
					
						
							| 
									
										
										
										
											2024-05-02 16:17:59 +01:00
										 |  |  | #ifdef _Py_TIER2
 | 
					
						
							| 
									
										
										
										
											2023-10-04 17:52:28 -04:00
										 |  |  |     print_optimization_stats(out, &stats->optimization_stats); | 
					
						
							| 
									
										
										
										
											2024-05-02 16:17:59 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-01-25 06:10:51 -05:00
										 |  |  |     print_rare_event_stats(out, &stats->rare_event_stats); | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | void | 
					
						
							|  |  |  | _Py_StatsOn(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _Py_stats = &_Py_stats_struct; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | _Py_StatsOff(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _Py_stats = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-21 15:40:54 +01:00
										 |  |  | void | 
					
						
							|  |  |  | _Py_StatsClear(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     memset(&_py_gc_stats, 0, sizeof(_py_gc_stats)); | 
					
						
							|  |  |  |     memset(&_Py_stats_struct, 0, sizeof(_Py_stats_struct)); | 
					
						
							|  |  |  |     _Py_stats_struct.gc_stats = _py_gc_stats; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | mem_is_zero(unsigned char *ptr, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (size_t i=0; i < size; i++) { | 
					
						
							|  |  |  |         if (*ptr != 0) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ptr++; | 
					
						
							| 
									
										
										
										
											2023-08-04 10:34:23 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2022-06-21 15:40:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  | _Py_PrintSpecializationStats(int to_file) | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     PyStats *stats = &_Py_stats_struct; | 
					
						
							|  |  |  | #define MEM_IS_ZERO(DATA) mem_is_zero((unsigned char*)DATA, sizeof(*(DATA)))
 | 
					
						
							|  |  |  |     int is_zero = ( | 
					
						
							|  |  |  |         MEM_IS_ZERO(stats->gc_stats)  // is a pointer
 | 
					
						
							|  |  |  |         && MEM_IS_ZERO(&stats->opcode_stats) | 
					
						
							|  |  |  |         && MEM_IS_ZERO(&stats->call_stats) | 
					
						
							|  |  |  |         && MEM_IS_ZERO(&stats->object_stats) | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | #undef MEM_IS_ZERO
 | 
					
						
							|  |  |  |     if (is_zero) { | 
					
						
							|  |  |  |         // gh-108753: -X pystats command line was used, but then _stats_off()
 | 
					
						
							|  |  |  |         // and _stats_clear() have been called: in this case, avoid printing
 | 
					
						
							|  |  |  |         // useless "all zeros" statistics.
 | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-04 11:39:52 +01:00
										 |  |  |     FILE *out = stderr; | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  |     if (to_file) { | 
					
						
							|  |  |  |         /* Write to a file instead of stderr. */ | 
					
						
							| 
									
										
										
										
											2021-08-04 11:39:52 +01:00
										 |  |  | # ifdef MS_WINDOWS
 | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  |         const char *dirname = "c:\\temp\\py_stats\\"; | 
					
						
							| 
									
										
										
										
											2021-08-04 11:39:52 +01:00
										 |  |  | # else
 | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  |         const char *dirname = "/tmp/py_stats/"; | 
					
						
							| 
									
										
										
										
											2021-08-04 11:39:52 +01:00
										 |  |  | # endif
 | 
					
						
							| 
									
										
										
										
											2021-12-16 15:56:01 +00:00
										 |  |  |         /* Use random 160 bit number as file name,
 | 
					
						
							|  |  |  |         * to avoid both accidental collisions and | 
					
						
							|  |  |  |         * symlink attacks. */ | 
					
						
							|  |  |  |         unsigned char rand[20]; | 
					
						
							|  |  |  |         char hex_name[41]; | 
					
						
							|  |  |  |         _PyOS_URandomNonblock(rand, 20); | 
					
						
							|  |  |  |         for (int i = 0; i < 20; i++) { | 
					
						
							| 
									
										
										
										
											2023-09-07 04:47:57 +02:00
										 |  |  |             hex_name[2*i] = Py_hexdigits[rand[i]&15]; | 
					
						
							|  |  |  |             hex_name[2*i+1] = Py_hexdigits[(rand[i]>>4)&15]; | 
					
						
							| 
									
										
										
										
											2021-12-16 15:56:01 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         hex_name[40] = '\0'; | 
					
						
							|  |  |  |         char buf[64]; | 
					
						
							|  |  |  |         assert(strlen(dirname) + 40 + strlen(".txt") < 64); | 
					
						
							|  |  |  |         sprintf(buf, "%s%s.txt", dirname, hex_name); | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  |         FILE *fout = fopen(buf, "w"); | 
					
						
							|  |  |  |         if (fout) { | 
					
						
							|  |  |  |             out = fout; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         fprintf(out, "Specialization stats:\n"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     print_stats(out, stats); | 
					
						
							| 
									
										
										
										
											2021-08-04 11:39:52 +01:00
										 |  |  |     if (out != stderr) { | 
					
						
							|  |  |  |         fclose(out); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-21 15:40:54 +01:00
										 |  |  | #define SPECIALIZATION_FAIL(opcode, kind) \
 | 
					
						
							|  |  |  | do { \ | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  |     if (_Py_stats) { \ | 
					
						
							|  |  |  |         _Py_stats->opcode_stats[opcode].specialization.failure_kinds[kind]++; \ | 
					
						
							| 
									
										
										
										
											2022-06-21 15:40:54 +01:00
										 |  |  |     } \ | 
					
						
							|  |  |  | } while (0) | 
					
						
							| 
									
										
										
										
											2021-06-15 13:01:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif  // Py_STATS
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 13:01:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifndef SPECIALIZATION_FAIL
 | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #  define SPECIALIZATION_FAIL(opcode, kind) ((void)0)
 | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-06-07 18:38:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  | // Initialize warmup counters and insert superinstructions. This cannot fail.
 | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  | void | 
					
						
							|  |  |  | _PyCode_Quicken(PyCodeObject *code) | 
					
						
							| 
									
										
										
										
											2021-06-07 18:38:06 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     #if ENABLE_SPECIALIZATION
 | 
					
						
							| 
									
										
										
										
											2023-01-11 20:40:43 +00:00
										 |  |  |     int opcode = 0; | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  |     _Py_CODEUNIT *instructions = _PyCode_CODE(code); | 
					
						
							| 
									
										
										
										
											2024-08-13 14:22:57 +01:00
										 |  |  |     /* The last code unit cannot have a cache, so we don't need to check it */ | 
					
						
							|  |  |  |     for (int i = 0; i < Py_SIZE(code)-1; i++) { | 
					
						
							|  |  |  |         opcode = instructions[i].op.code; | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |         int caches = _PyOpcode_Caches[opcode]; | 
					
						
							|  |  |  |         if (caches) { | 
					
						
							| 
									
										
										
										
											2023-09-11 11:20:24 -07:00
										 |  |  |             // The initial value depends on the opcode
 | 
					
						
							|  |  |  |             switch (opcode) { | 
					
						
							|  |  |  |                 case JUMP_BACKWARD: | 
					
						
							| 
									
										
										
										
											2024-04-04 08:03:27 -07:00
										 |  |  |                     instructions[i + 1].counter = initial_jump_backoff_counter(); | 
					
						
							| 
									
										
										
										
											2023-09-11 11:20:24 -07:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 case POP_JUMP_IF_FALSE: | 
					
						
							|  |  |  |                 case POP_JUMP_IF_TRUE: | 
					
						
							|  |  |  |                 case POP_JUMP_IF_NONE: | 
					
						
							|  |  |  |                 case POP_JUMP_IF_NOT_NONE: | 
					
						
							| 
									
										
										
										
											2024-04-04 08:03:27 -07:00
										 |  |  |                     instructions[i + 1].cache = 0x5555;  // Alternating 0, 1 bits
 | 
					
						
							| 
									
										
										
										
											2023-09-11 11:20:24 -07:00
										 |  |  |                     break; | 
					
						
							|  |  |  |                 default: | 
					
						
							| 
									
										
										
										
											2024-04-04 08:03:27 -07:00
										 |  |  |                     instructions[i + 1].counter = adaptive_counter_warmup(); | 
					
						
							| 
									
										
										
										
											2023-09-11 11:20:24 -07:00
										 |  |  |                     break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |             i += caches; | 
					
						
							| 
									
										
										
										
											2021-06-07 18:38:06 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     #endif /* ENABLE_SPECIALIZATION */
 | 
					
						
							| 
									
										
										
										
											2021-06-07 18:38:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  | #define SIMPLE_FUNCTION 0
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  | /* Common */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  | #define SPEC_FAIL_OTHER 0
 | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  | #define SPEC_FAIL_NO_DICT 1
 | 
					
						
							|  |  |  | #define SPEC_FAIL_OVERRIDDEN 2
 | 
					
						
							|  |  |  | #define SPEC_FAIL_OUT_OF_VERSIONS 3
 | 
					
						
							|  |  |  | #define SPEC_FAIL_OUT_OF_RANGE 4
 | 
					
						
							|  |  |  | #define SPEC_FAIL_EXPECTED_ERROR 5
 | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | #define SPEC_FAIL_WRONG_NUMBER_ARGUMENTS 6
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_CODE_COMPLEX_PARAMETERS 7
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CODE_NOT_OPTIMIZED 8
 | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  | #define SPEC_FAIL_LOAD_GLOBAL_NON_DICT 17
 | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | #define SPEC_FAIL_LOAD_GLOBAL_NON_STRING_OR_SPLIT 18
 | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  | /* Super */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-11 08:08:13 -06:00
										 |  |  | #define SPEC_FAIL_SUPER_BAD_CLASS 9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUPER_SHADOWED 10
 | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  | /* Attributes */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_ATTR_OVERRIDING_DESCRIPTOR 9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_NON_OVERRIDING_DESCRIPTOR 10
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_NOT_DESCRIPTOR 11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_METHOD 12
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_MUTABLE_CLASS 13
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_PROPERTY 14
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_NON_OBJECT_SLOT 15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_READ_ONLY 16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_AUDITED_SLOT 17
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_NOT_MANAGED_DICT 18
 | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  | #define SPEC_FAIL_ATTR_NON_STRING 19
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_ATTR_MODULE_ATTR_NOT_FOUND 20
 | 
					
						
							| 
									
										
										
										
											2022-06-14 16:44:01 +01:00
										 |  |  | #define SPEC_FAIL_ATTR_SHADOWED 21
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_BUILTIN_CLASS_METHOD 22
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_CLASS_METHOD_OBJ 23
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_OBJECT_SLOT 24
 | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-14 16:44:01 +01:00
										 |  |  | #define SPEC_FAIL_ATTR_INSTANCE_ATTRIBUTE 26
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_METACLASS_ATTRIBUTE 27
 | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  | #define SPEC_FAIL_ATTR_PROPERTY_NOT_PY_FUNCTION 28
 | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  | #define SPEC_FAIL_ATTR_NOT_IN_KEYS 29
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_NOT_IN_DICT 30
 | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  | #define SPEC_FAIL_ATTR_CLASS_ATTR_SIMPLE 31
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_CLASS_ATTR_DESCRIPTOR 32
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_BUILTIN_CLASS_METHOD_OBJ 33
 | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  | #define SPEC_FAIL_ATTR_METACLASS_OVERRIDDEN 34
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ATTR_SPLIT_DICT 35
 | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Binary subscr and store subscr */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_SUBSCR_ARRAY_INT 9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_ARRAY_SLICE 10
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_LIST_SLICE 11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_TUPLE_SLICE 12
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_STRING_SLICE 14
 | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | #define SPEC_FAIL_SUBSCR_BUFFER_INT 15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_BUFFER_SLICE 16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_SEQUENCE_INT 17
 | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  | /* Store subscr */ | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | #define SPEC_FAIL_SUBSCR_BYTEARRAY_INT 18
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_BYTEARRAY_SLICE 19
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_PY_SIMPLE 20
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_PY_OTHER 21
 | 
					
						
							|  |  |  | #define SPEC_FAIL_SUBSCR_DICT_SUBCLASS_NO_OVERRIDE 22
 | 
					
						
							| 
									
										
										
										
											2022-03-01 16:00:34 +00:00
										 |  |  | #define SPEC_FAIL_SUBSCR_NOT_HEAP_TYPE 23
 | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 08:49:58 -08:00
										 |  |  | /* Binary op */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_BINARY_OP_ADD_DIFFERENT_TYPES          9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_ADD_OTHER                   10
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_AND_DIFFERENT_TYPES         11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_AND_INT                     12
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_AND_OTHER                   13
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_FLOOR_DIVIDE                14
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_LSHIFT                      15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_MATRIX_MULTIPLY             16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_MULTIPLY_DIFFERENT_TYPES    17
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_MULTIPLY_OTHER              18
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_OR                          19
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_POWER                       20
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_REMAINDER                   21
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_RSHIFT                      22
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_SUBTRACT_DIFFERENT_TYPES    23
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_SUBTRACT_OTHER              24
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_TRUE_DIVIDE_DIFFERENT_TYPES 25
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_TRUE_DIVIDE_FLOAT           26
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_TRUE_DIVIDE_OTHER           27
 | 
					
						
							|  |  |  | #define SPEC_FAIL_BINARY_OP_XOR                         28
 | 
					
						
							| 
									
										
										
										
											2021-08-27 09:21:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  | /* Calls */ | 
					
						
							| 
									
										
										
										
											2021-11-23 09:53:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  | #define SPEC_FAIL_CALL_INSTANCE_METHOD 11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_CMETHOD 12
 | 
					
						
							| 
									
										
										
										
											2022-12-16 15:43:04 +00:00
										 |  |  | #define SPEC_FAIL_CALL_CFUNC_VARARGS 13
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_CFUNC_VARARGS_KEYWORDS 14
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_CALL_CFUNC_NOARGS 15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_CFUNC_METHOD_FASTCALL_KEYWORDS 16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_METH_DESCR_VARARGS 17
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_METH_DESCR_VARARGS_KEYWORDS 18
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_METH_DESCR_METHOD_FASTCALL_KEYWORDS 19
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_BAD_CALL_FLAGS 20
 | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  | #define SPEC_FAIL_CALL_INIT_NOT_PYTHON 21
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_CALL_PEP_523 22
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_BOUND_METHOD 23
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_CLASS_MUTABLE 26
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_METHOD_WRAPPER 28
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CALL_OPERATOR_WRAPPER 29
 | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  | #define SPEC_FAIL_CALL_INIT_NOT_SIMPLE 30
 | 
					
						
							| 
									
										
										
										
											2024-02-01 19:39:32 +00:00
										 |  |  | #define SPEC_FAIL_CALL_METACLASS 31
 | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  | #define SPEC_FAIL_CALL_INIT_NOT_INLINE_VALUES 32
 | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  | /* COMPARE_OP */ | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  | #define SPEC_FAIL_COMPARE_OP_DIFFERENT_TYPES 12
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_STRING 13
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_BIG_INT 14
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_BYTES 15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_TUPLE 16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_LIST 17
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_SET 18
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_BOOL 19
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_BASEOBJECT 20
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_FLOAT_LONG 21
 | 
					
						
							|  |  |  | #define SPEC_FAIL_COMPARE_OP_LONG_FLOAT 22
 | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  | /* FOR_ITER and SEND */ | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_GENERATOR 10
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_COROUTINE 11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_ASYNC_GENERATOR 12
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_LIST 13
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_TUPLE 14
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_SET 15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_STRING 16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_BYTES 17
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_RANGE 18
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_ITERTOOLS 19
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_DICT_KEYS 20
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_DICT_ITEMS 21
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_DICT_VALUES 22
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_ENUMERATE 23
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_MAP 24
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_ZIP 25
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_SEQ_ITER 26
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_REVERSED_LIST 27
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_CALLABLE 28
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_ASCII_STRING 29
 | 
					
						
							|  |  |  | #define SPEC_FAIL_ITER_ASYNC_GENERATOR_SEND 30
 | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  | // UNPACK_SEQUENCE
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  | #define SPEC_FAIL_UNPACK_SEQUENCE_ITERATOR 9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_UNPACK_SEQUENCE_SEQUENCE 10
 | 
					
						
							| 
									
										
										
										
											2022-02-03 18:40:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  | // TO_BOOL
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_BYTEARRAY    9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_BYTES       10
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_DICT        11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_FLOAT       12
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_MAPPING     13
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_MEMORY_VIEW 14
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_NUMBER      15
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_SEQUENCE    16
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_SET         17
 | 
					
						
							|  |  |  | #define SPEC_FAIL_TO_BOOL_TUPLE       18
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  | // CONTAINS_OP
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CONTAINS_OP_STR        9
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CONTAINS_OP_TUPLE      10
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CONTAINS_OP_LIST       11
 | 
					
						
							|  |  |  | #define SPEC_FAIL_CONTAINS_OP_USER_CLASS 12
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  | static int function_kind(PyCodeObject *code); | 
					
						
							|  |  |  | static bool function_check_args(PyObject *o, int expected_argcount, int opcode); | 
					
						
							|  |  |  | static uint32_t function_get_version(PyObject *o, int opcode); | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  | static uint32_t type_get_version(PyTypeObject *t, int opcode); | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  | specialize_module_load_attr( | 
					
						
							|  |  |  |     PyObject *owner, _Py_CODEUNIT *instr, PyObject *name | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     _PyAttrCache *cache = (_PyAttrCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     PyModuleObject *m = (PyModuleObject *)owner; | 
					
						
							| 
									
										
										
										
											2024-06-17 10:34:29 +02:00
										 |  |  |     assert((Py_TYPE(owner)->tp_flags & Py_TPFLAGS_MANAGED_DICT) == 0); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     PyDictObject *dict = (PyDictObject *)m->md_dict; | 
					
						
							|  |  |  |     if (dict == NULL) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_NO_DICT); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (dict->ma_keys->dk_kind != DICT_KEYS_UNICODE) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_NON_STRING); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-08-18 17:19:21 +08:00
										 |  |  |     Py_ssize_t index = _PyDict_LookupIndex(dict, &_Py_ID(__getattr__)); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     assert(index != DKIX_ERROR); | 
					
						
							|  |  |  |     if (index != DKIX_EMPTY) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_MODULE_ATTR_NOT_FOUND); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-08-18 17:19:21 +08:00
										 |  |  |     index = _PyDict_LookupIndex(dict, name); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     assert (index != DKIX_ERROR); | 
					
						
							|  |  |  |     if (index != (uint16_t)index) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, | 
					
						
							|  |  |  |                             index == DKIX_EMPTY ? | 
					
						
							|  |  |  |                             SPEC_FAIL_ATTR_MODULE_ATTR_NOT_FOUND : | 
					
						
							|  |  |  |                             SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-08 18:04:16 -07:00
										 |  |  |     uint32_t keys_version = _PyDictKeys_GetVersionForCurrentState( | 
					
						
							|  |  |  |             _PyInterpreterState_GET(), dict->ma_keys); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     if (keys_version == 0) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     write_u32(cache->version, keys_version); | 
					
						
							|  |  |  |     cache->index = (uint16_t)index; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = LOAD_ATTR_MODULE; | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Attribute specialization */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_LoadSuperAttr(_PyStackRef global_super_st, _PyStackRef cls_st, _Py_CODEUNIT *instr, int load_method) { | 
					
						
							|  |  |  |     PyObject *global_super = PyStackRef_AsPyObjectBorrow(global_super_st); | 
					
						
							|  |  |  |     PyObject *cls = PyStackRef_AsPyObjectBorrow(cls_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							|  |  |  |     assert(_PyOpcode_Caches[LOAD_SUPER_ATTR] == INLINE_CACHE_ENTRIES_LOAD_SUPER_ATTR); | 
					
						
							|  |  |  |     _PySuperAttrCache *cache = (_PySuperAttrCache *)(instr + 1); | 
					
						
							|  |  |  |     if (global_super != (PyObject *)&PySuper_Type) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_SUPER_ATTR, SPEC_FAIL_SUPER_SHADOWED); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-28 13:20:50 -06:00
										 |  |  |     if (!PyType_Check(cls)) { | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_SUPER_ATTR, SPEC_FAIL_SUPER_BAD_CLASS); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-11 08:08:13 -06:00
										 |  |  |     instr->op.code = load_method ? LOAD_SUPER_ATTR_METHOD : LOAD_SUPER_ATTR_ATTR; | 
					
						
							|  |  |  |     goto success; | 
					
						
							| 
									
										
										
										
											2023-04-25 11:45:51 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | fail: | 
					
						
							|  |  |  |     STAT_INC(LOAD_SUPER_ATTR, failure); | 
					
						
							|  |  |  |     assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |     instr->op.code = LOAD_SUPER_ATTR; | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							|  |  |  |     STAT_INC(LOAD_SUPER_ATTR, success); | 
					
						
							|  |  |  |     assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  | typedef enum { | 
					
						
							|  |  |  |     OVERRIDING, /* Is an overriding descriptor, and will remain so. */ | 
					
						
							|  |  |  |     METHOD, /* Attribute has Py_TPFLAGS_METHOD_DESCRIPTOR set */ | 
					
						
							|  |  |  |     PROPERTY, /* Is a property */ | 
					
						
							|  |  |  |     OBJECT_SLOT, /* Is an object slot descriptor */ | 
					
						
							|  |  |  |     OTHER_SLOT, /* Is a slot descriptor of another type */ | 
					
						
							|  |  |  |     NON_OVERRIDING, /* Is another non-overriding descriptor, and is an instance of an immutable class*/ | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |     BUILTIN_CLASSMETHOD, /* Builtin methods with METH_CLASS */ | 
					
						
							|  |  |  |     PYTHON_CLASSMETHOD, /* Python classmethod(func) object */ | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |     NON_DESCRIPTOR, /* Is not a descriptor, and is an instance of an immutable class */ | 
					
						
							|  |  |  |     MUTABLE,   /* Instance of a mutable class; might, or might not, be a descriptor */ | 
					
						
							|  |  |  |     ABSENT, /* Attribute is not present on the class */ | 
					
						
							|  |  |  |     DUNDER_CLASS, /* __class__ attribute */ | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |     GETSET_OVERRIDDEN, /* __getattribute__ or __setattr__ has been overridden */ | 
					
						
							|  |  |  |     GETATTRIBUTE_IS_PYTHON_FUNCTION  /* Descriptor requires calling a Python __getattribute__ */ | 
					
						
							| 
									
										
										
										
											2022-01-24 03:33:50 +05:30
										 |  |  | } DescriptorClassification; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  | static DescriptorClassification | 
					
						
							|  |  |  | classify_descriptor(PyObject *descriptor, bool has_getattr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (descriptor == NULL) { | 
					
						
							|  |  |  |         return ABSENT; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyTypeObject *desc_cls = Py_TYPE(descriptor); | 
					
						
							|  |  |  |     if (!(desc_cls->tp_flags & Py_TPFLAGS_IMMUTABLETYPE)) { | 
					
						
							|  |  |  |         return MUTABLE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (desc_cls->tp_descr_set) { | 
					
						
							|  |  |  |         if (desc_cls == &PyMemberDescr_Type) { | 
					
						
							|  |  |  |             PyMemberDescrObject *member = (PyMemberDescrObject *)descriptor; | 
					
						
							|  |  |  |             struct PyMemberDef *dmem = member->d_member; | 
					
						
							|  |  |  |             if (dmem->type == Py_T_OBJECT_EX || dmem->type == _Py_T_OBJECT) { | 
					
						
							|  |  |  |                 return OBJECT_SLOT; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return OTHER_SLOT; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (desc_cls == &PyProperty_Type) { | 
					
						
							|  |  |  |             /* We can't detect at runtime whether an attribute exists
 | 
					
						
							|  |  |  |                with property. So that means we may have to call | 
					
						
							|  |  |  |                __getattr__. */ | 
					
						
							|  |  |  |             return has_getattr ? GETSET_OVERRIDDEN : PROPERTY; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return OVERRIDING; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (desc_cls->tp_descr_get) { | 
					
						
							|  |  |  |         if (desc_cls->tp_flags & Py_TPFLAGS_METHOD_DESCRIPTOR) { | 
					
						
							|  |  |  |             return METHOD; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (Py_IS_TYPE(descriptor, &PyClassMethodDescr_Type)) { | 
					
						
							|  |  |  |             return BUILTIN_CLASSMETHOD; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (Py_IS_TYPE(descriptor, &PyClassMethod_Type)) { | 
					
						
							|  |  |  |             return PYTHON_CLASSMETHOD; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return NON_OVERRIDING; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return NON_DESCRIPTOR; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-24 03:33:50 +05:30
										 |  |  | static DescriptorClassification | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | analyze_descriptor(PyTypeObject *type, PyObject *name, PyObject **descr, int store) | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |     bool has_getattr = false; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     if (store) { | 
					
						
							|  |  |  |         if (type->tp_setattro != PyObject_GenericSetAttr) { | 
					
						
							|  |  |  |             *descr = NULL; | 
					
						
							|  |  |  |             return GETSET_OVERRIDDEN; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |         getattrofunc getattro_slot = type->tp_getattro; | 
					
						
							|  |  |  |         if (getattro_slot == PyObject_GenericGetAttr) { | 
					
						
							|  |  |  |             /* Normal attribute lookup; */ | 
					
						
							|  |  |  |             has_getattr = false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (getattro_slot == _Py_slot_tp_getattr_hook || | 
					
						
							|  |  |  |             getattro_slot == _Py_slot_tp_getattro) { | 
					
						
							|  |  |  |             /* One or both of __getattribute__ or __getattr__ may have been
 | 
					
						
							|  |  |  |              overridden See typeobject.c for why these functions are special. */ | 
					
						
							|  |  |  |             PyObject *getattribute = _PyType_Lookup(type, | 
					
						
							|  |  |  |                 &_Py_ID(__getattribute__)); | 
					
						
							|  |  |  |             PyInterpreterState *interp = _PyInterpreterState_GET(); | 
					
						
							|  |  |  |             bool has_custom_getattribute = getattribute != NULL && | 
					
						
							|  |  |  |                 getattribute != interp->callable_cache.object__getattribute__; | 
					
						
							|  |  |  |             has_getattr = _PyType_Lookup(type, &_Py_ID(__getattr__)) != NULL; | 
					
						
							|  |  |  |             if (has_custom_getattribute) { | 
					
						
							|  |  |  |                 if (getattro_slot == _Py_slot_tp_getattro && | 
					
						
							|  |  |  |                     !has_getattr && | 
					
						
							|  |  |  |                     Py_IS_TYPE(getattribute, &PyFunction_Type)) { | 
					
						
							|  |  |  |                     *descr = getattribute; | 
					
						
							|  |  |  |                     return GETATTRIBUTE_IS_PYTHON_FUNCTION; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 /* Potentially both __getattr__ and __getattribute__ are set.
 | 
					
						
							|  |  |  |                    Too complicated */ | 
					
						
							|  |  |  |                 *descr = NULL; | 
					
						
							|  |  |  |                 return GETSET_OVERRIDDEN; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             /* Potentially has __getattr__ but no custom __getattribute__.
 | 
					
						
							|  |  |  |                Fall through to usual descriptor analysis. | 
					
						
							|  |  |  |                Usual attribute lookup should only be allowed at runtime | 
					
						
							|  |  |  |                if we can guarantee that there is no way an exception can be | 
					
						
							|  |  |  |                raised. This means some specializations, e.g. specializing | 
					
						
							|  |  |  |                for property() isn't safe. | 
					
						
							|  |  |  |             */ | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             *descr = NULL; | 
					
						
							|  |  |  |             return GETSET_OVERRIDDEN; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     PyObject *descriptor = _PyType_Lookup(type, name); | 
					
						
							|  |  |  |     *descr = descriptor; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     if (PyUnicode_CompareWithASCIIString(name, "__class__") == 0) { | 
					
						
							|  |  |  |         if (descriptor == _PyType_Lookup(&PyBaseObject_Type, name)) { | 
					
						
							|  |  |  |             return DUNDER_CLASS; | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     return classify_descriptor(descriptor, has_getattr); | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | static int | 
					
						
							|  |  |  | specialize_dict_access( | 
					
						
							|  |  |  |     PyObject *owner, _Py_CODEUNIT *instr, PyTypeObject *type, | 
					
						
							| 
									
										
										
										
											2022-01-24 03:33:50 +05:30
										 |  |  |     DescriptorClassification kind, PyObject *name, | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |     int base_op, int values_op, int hint_op) | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |     assert(kind == NON_OVERRIDING || kind == NON_DESCRIPTOR || kind == ABSENT || | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |         kind == BUILTIN_CLASSMETHOD || kind == PYTHON_CLASSMETHOD || | 
					
						
							|  |  |  |         kind == METHOD); | 
					
						
							| 
									
										
										
										
											2021-10-07 00:55:27 +02:00
										 |  |  |     // No descriptor, or non overriding.
 | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |     if ((type->tp_flags & Py_TPFLAGS_MANAGED_DICT) == 0) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |         SPECIALIZATION_FAIL(base_op, SPEC_FAIL_ATTR_NOT_MANAGED_DICT); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     _PyAttrCache *cache = (_PyAttrCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |     if (type->tp_flags & Py_TPFLAGS_INLINE_VALUES && _PyObject_InlineValues(owner)->valid) { | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |         PyDictKeysObject *keys = ((PyHeapTypeObject *)type)->ht_cached_keys; | 
					
						
							|  |  |  |         assert(PyUnicode_CheckExact(name)); | 
					
						
							|  |  |  |         Py_ssize_t index = _PyDictKeys_StringLookup(keys, name); | 
					
						
							|  |  |  |         assert (index != DKIX_ERROR); | 
					
						
							| 
									
										
										
										
											2024-08-21 15:52:04 +01:00
										 |  |  |         if (index == DKIX_EMPTY) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(base_op, SPEC_FAIL_ATTR_NOT_IN_KEYS); | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         assert(index >= 0); | 
					
						
							|  |  |  |         char *value_addr = (char *)&_PyObject_InlineValues(owner)->values[index]; | 
					
						
							|  |  |  |         Py_ssize_t offset = value_addr - (char *)owner; | 
					
						
							|  |  |  |         if (offset != (uint16_t)offset) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(base_op, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |         write_u32(cache->version, type->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2024-08-21 15:52:04 +01:00
										 |  |  |         cache->index = (uint16_t)offset; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = values_op; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2024-04-21 22:57:05 -07:00
										 |  |  |         PyDictObject *dict = _PyObject_GetManagedDict(owner); | 
					
						
							| 
									
										
										
										
											2022-08-01 14:34:54 +01:00
										 |  |  |         if (dict == NULL || !PyDict_CheckExact(dict)) { | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |             SPECIALIZATION_FAIL(base_op, SPEC_FAIL_NO_DICT); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         // We found an instance with a __dict__.
 | 
					
						
							| 
									
										
										
										
											2023-08-09 12:14:50 -07:00
										 |  |  |         if (dict->ma_values) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             SPECIALIZATION_FAIL(base_op, SPEC_FAIL_ATTR_SPLIT_DICT); | 
					
						
							| 
									
										
										
										
											2023-08-09 12:14:50 -07:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-08-18 17:19:21 +08:00
										 |  |  |         Py_ssize_t index = | 
					
						
							|  |  |  |             _PyDict_LookupIndex(dict, name); | 
					
						
							|  |  |  |         if (index != (uint16_t)index) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |             SPECIALIZATION_FAIL(base_op, | 
					
						
							|  |  |  |                                 index == DKIX_EMPTY ? | 
					
						
							|  |  |  |                                 SPEC_FAIL_ATTR_NOT_IN_DICT : | 
					
						
							|  |  |  |                                 SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-08-18 17:19:21 +08:00
										 |  |  |         cache->index = (uint16_t)index; | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |         write_u32(cache->version, type->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = hint_op; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-07 16:02:53 +00:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  | static int specialize_attr_loadclassattr(PyObject* owner, _Py_CODEUNIT* instr, PyObject* name, | 
					
						
							|  |  |  |     PyObject* descr, DescriptorClassification kind, bool is_method); | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  | static int specialize_class_load_attr(PyObject* owner, _Py_CODEUNIT* instr, PyObject* name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  | /* Returns true if instances of obj's class are
 | 
					
						
							|  |  |  |  * likely to have `name` in their __dict__. | 
					
						
							|  |  |  |  * For objects with inline values, we check in the shared keys. | 
					
						
							|  |  |  |  * For other objects, we check their actual dictionary. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static bool | 
					
						
							|  |  |  | instance_has_key(PyObject *obj, PyObject* name) | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     PyTypeObject *cls = Py_TYPE(obj); | 
					
						
							|  |  |  |     if ((cls->tp_flags & Py_TPFLAGS_MANAGED_DICT) == 0) { | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2022-11-17 15:09:18 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     if (cls->tp_flags & Py_TPFLAGS_INLINE_VALUES) { | 
					
						
							|  |  |  |         PyDictKeysObject *keys = ((PyHeapTypeObject *)cls)->ht_cached_keys; | 
					
						
							|  |  |  |         Py_ssize_t index = _PyDictKeys_StringLookup(keys, name); | 
					
						
							|  |  |  |         return index >= 0; | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     PyDictObject *dict = _PyObject_GetManagedDict(obj); | 
					
						
							|  |  |  |     if (dict == NULL || !PyDict_CheckExact(dict)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (dict->ma_values) { | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     Py_ssize_t index = _PyDict_LookupIndex(dict, name); | 
					
						
							|  |  |  |     if (index < 0) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | specialize_instance_load_attr(PyObject* owner, _Py_CODEUNIT* instr, PyObject* name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyAttrCache *cache = (_PyAttrCache *)(instr + 1); | 
					
						
							|  |  |  |     PyTypeObject *type = Py_TYPE(owner); | 
					
						
							|  |  |  |     bool shadow = instance_has_key(owner, name); | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |     PyObject *descr = NULL; | 
					
						
							| 
									
										
										
										
											2022-01-24 03:33:50 +05:30
										 |  |  |     DescriptorClassification kind = analyze_descriptor(type, name, &descr, 0); | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |     assert(descr != NULL || kind == ABSENT || kind == GETSET_OVERRIDDEN); | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  |     if (type_get_version(type, LOAD_ATTR) == 0) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |     switch(kind) { | 
					
						
							|  |  |  |         case OVERRIDING: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_OVERRIDING_DESCRIPTOR); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case METHOD: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             if (shadow) { | 
					
						
							|  |  |  |                 goto try_instance; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             int oparg = instr->op.arg; | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             if (oparg & 1) { | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |                 if (specialize_attr_loadclassattr(owner, instr, name, descr, kind, true)) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                     return 0; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 else { | 
					
						
							|  |  |  |                     return -1; | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_METHOD); | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case PROPERTY: | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  |         { | 
					
						
							|  |  |  |             _PyLoadMethodCache *lm_cache = (_PyLoadMethodCache *)(instr + 1); | 
					
						
							|  |  |  |             assert(Py_TYPE(descr) == &PyProperty_Type); | 
					
						
							|  |  |  |             PyObject *fget = ((_PyPropertyObject *)descr)->prop_get; | 
					
						
							|  |  |  |             if (fget == NULL) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |             if (!Py_IS_TYPE(fget, &PyFunction_Type)) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_PROPERTY_NOT_PY_FUNCTION); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-09-06 04:11:38 -07:00
										 |  |  |             if (!function_check_args(fget, 1, LOAD_ATTR)) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2022-09-06 04:11:38 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-08-09 11:19:39 -07:00
										 |  |  |             if (instr->op.arg & 1) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_METHOD); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2023-08-09 11:19:39 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |             if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  |             assert(type->tp_version_tag != 0); | 
					
						
							|  |  |  |             write_u32(lm_cache->type_version, type->tp_version_tag); | 
					
						
							|  |  |  |             /* borrowed */ | 
					
						
							|  |  |  |             write_obj(lm_cache->descr, fget); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = LOAD_ATTR_PROPERTY; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-06-17 23:13:17 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case OBJECT_SLOT: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             PyMemberDescrObject *member = (PyMemberDescrObject *)descr; | 
					
						
							|  |  |  |             struct PyMemberDef *dmem = member->d_member; | 
					
						
							|  |  |  |             Py_ssize_t offset = dmem->offset; | 
					
						
							| 
									
										
										
										
											2022-11-10 03:50:34 -08:00
										 |  |  |             if (!PyObject_TypeCheck(owner, member->d_common.d_type)) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2022-11-10 03:50:34 -08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  |             if (dmem->flags & Py_AUDIT_READ) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_AUDITED_SLOT); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |             if (offset != (uint16_t)offset) { | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-11-06 13:55:04 +00:00
										 |  |  |             assert(dmem->type == Py_T_OBJECT_EX || dmem->type == _Py_T_OBJECT); | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |             assert(offset > 0); | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |             cache->index = (uint16_t)offset; | 
					
						
							|  |  |  |             write_u32(cache->version, type->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = LOAD_ATTR_SLOT; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case DUNDER_CLASS: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             Py_ssize_t offset = offsetof(PyObject, ob_type); | 
					
						
							|  |  |  |             assert(offset == (uint16_t)offset); | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |             cache->index = (uint16_t)offset; | 
					
						
							|  |  |  |             write_u32(cache->version, type->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = LOAD_ATTR_SLOT; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         case OTHER_SLOT: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_NON_OBJECT_SLOT); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case MUTABLE: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_MUTABLE_CLASS); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case GETSET_OVERRIDDEN: | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OVERRIDDEN); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |         case GETATTRIBUTE_IS_PYTHON_FUNCTION: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             assert(type->tp_getattro == _Py_slot_tp_getattro); | 
					
						
							|  |  |  |             assert(Py_IS_TYPE(descr, &PyFunction_Type)); | 
					
						
							|  |  |  |             _PyLoadMethodCache *lm_cache = (_PyLoadMethodCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2022-09-06 04:11:38 -07:00
										 |  |  |             if (!function_check_args(descr, 2, LOAD_ATTR)) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-08-09 11:19:39 -07:00
										 |  |  |             if (instr->op.arg & 1) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_METHOD); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2023-08-09 11:19:39 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-09-15 08:42:37 -07:00
										 |  |  |             uint32_t version = function_get_version(descr, LOAD_ATTR); | 
					
						
							|  |  |  |             if (version == 0) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2022-09-15 08:42:37 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |             if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-09-15 08:42:37 -07:00
										 |  |  |             write_u32(lm_cache->keys_version, version); | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |             /* borrowed */ | 
					
						
							|  |  |  |             write_obj(lm_cache->descr, descr); | 
					
						
							|  |  |  |             write_u32(lm_cache->type_version, type->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = LOAD_ATTR_GETATTRIBUTE_OVERRIDDEN; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |         case BUILTIN_CLASSMETHOD: | 
					
						
							|  |  |  |         case PYTHON_CLASSMETHOD: | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case NON_OVERRIDING: | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             if (shadow) { | 
					
						
							|  |  |  |                 goto try_instance; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case NON_DESCRIPTOR: | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             if (shadow) { | 
					
						
							|  |  |  |                 goto try_instance; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |             if ((instr->op.arg & 1) == 0) { | 
					
						
							|  |  |  |                 if (specialize_attr_loadclassattr(owner, instr, name, descr, kind, false)) { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |                     return 0; | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |         case ABSENT: | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             if (shadow) { | 
					
						
							|  |  |  |                 goto try_instance; | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_UNREACHABLE(); | 
					
						
							|  |  |  | try_instance: | 
					
						
							|  |  |  |     if (specialize_dict_access(owner, instr, type, kind, name, LOAD_ATTR, | 
					
						
							|  |  |  |                                     LOAD_ATTR_INSTANCE_VALUE, LOAD_ATTR_WITH_HINT)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | _Py_Specialize_LoadAttr(_PyStackRef owner_st, _Py_CODEUNIT *instr, PyObject *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyAttrCache *cache = (_PyAttrCache *)(instr + 1); | 
					
						
							|  |  |  |     PyObject *owner = PyStackRef_AsPyObjectBorrow(owner_st); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							|  |  |  |     assert(_PyOpcode_Caches[LOAD_ATTR] == INLINE_CACHE_ENTRIES_LOAD_ATTR); | 
					
						
							|  |  |  |     PyTypeObject *type = Py_TYPE(owner); | 
					
						
							|  |  |  |     bool fail; | 
					
						
							|  |  |  |     if (!_PyType_IsReady(type)) { | 
					
						
							|  |  |  |         // We *might* not really need this check, but we inherited it from
 | 
					
						
							|  |  |  |         // PyObject_GenericGetAttr and friends... and this way we still do the
 | 
					
						
							|  |  |  |         // right thing if someone forgets to call PyType_Ready(type):
 | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |         fail = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyModule_CheckExact(owner)) { | 
					
						
							|  |  |  |         fail = specialize_module_load_attr(owner, instr, name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyType_Check(owner)) { | 
					
						
							|  |  |  |         fail = specialize_class_load_attr(owner, instr, name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         fail = specialize_instance_load_attr(owner, instr, name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fail) { | 
					
						
							|  |  |  |         STAT_INC(LOAD_ATTR, failure); | 
					
						
							|  |  |  |         assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |         instr->op.code = LOAD_ATTR; | 
					
						
							|  |  |  |         cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         STAT_INC(LOAD_ATTR, success); | 
					
						
							|  |  |  |         assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |         cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 15:09:18 -08:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_StoreAttr(_PyStackRef owner_st, _Py_CODEUNIT *instr, PyObject *name) | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *owner = PyStackRef_AsPyObjectBorrow(owner_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  |     assert(_PyOpcode_Caches[STORE_ATTR] == INLINE_CACHE_ENTRIES_STORE_ATTR); | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     _PyAttrCache *cache = (_PyAttrCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     PyTypeObject *type = Py_TYPE(owner); | 
					
						
							| 
									
										
										
										
											2022-11-17 15:09:18 -08:00
										 |  |  |     if (!_PyType_IsReady(type)) { | 
					
						
							|  |  |  |         // We *might* not really need this check, but we inherited it from
 | 
					
						
							|  |  |  |         // PyObject_GenericSetAttr and friends... and this way we still do the
 | 
					
						
							|  |  |  |         // right thing if someone forgets to call PyType_Ready(type):
 | 
					
						
							| 
									
										
										
										
											2022-12-23 14:42:24 +00:00
										 |  |  |         SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2022-11-17 15:09:18 -08:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     if (PyModule_CheckExact(owner)) { | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |         SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_OVERRIDDEN); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     PyObject *descr; | 
					
						
							| 
									
										
										
										
											2022-01-24 03:33:50 +05:30
										 |  |  |     DescriptorClassification kind = analyze_descriptor(type, name, &descr, 1); | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  |     if (type_get_version(type, STORE_ATTR) == 0) { | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |     switch(kind) { | 
					
						
							|  |  |  |         case OVERRIDING: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_OVERRIDING_DESCRIPTOR); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case METHOD: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_METHOD); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         case PROPERTY: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_PROPERTY); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         case OBJECT_SLOT: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             PyMemberDescrObject *member = (PyMemberDescrObject *)descr; | 
					
						
							|  |  |  |             struct PyMemberDef *dmem = member->d_member; | 
					
						
							|  |  |  |             Py_ssize_t offset = dmem->offset; | 
					
						
							| 
									
										
										
										
											2022-11-10 03:50:34 -08:00
										 |  |  |             if (!PyObject_TypeCheck(owner, member->d_common.d_type)) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							|  |  |  |                 goto fail; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-07-25 15:28:30 +02:00
										 |  |  |             if (dmem->flags & Py_READONLY) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_READ_ONLY); | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |                 goto fail; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             if (offset != (uint16_t)offset) { | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |                 goto fail; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-11-06 13:55:04 +00:00
										 |  |  |             assert(dmem->type == Py_T_OBJECT_EX || dmem->type == _Py_T_OBJECT); | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |             assert(offset > 0); | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |             cache->index = (uint16_t)offset; | 
					
						
							|  |  |  |             write_u32(cache->version, type->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = STORE_ATTR_SLOT; | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |             goto success; | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case DUNDER_CLASS: | 
					
						
							|  |  |  |         case OTHER_SLOT: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_NON_OBJECT_SLOT); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case MUTABLE: | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_MUTABLE_CLASS); | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  |         case GETATTRIBUTE_IS_PYTHON_FUNCTION: | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case GETSET_OVERRIDDEN: | 
					
						
							| 
									
										
										
										
											2021-08-10 14:53:05 +01:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_OVERRIDDEN); | 
					
						
							| 
									
										
										
										
											2021-06-21 11:49:21 +01:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |         case BUILTIN_CLASSMETHOD: | 
					
						
							| 
									
										
										
										
											2022-12-23 14:42:24 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_BUILTIN_CLASS_METHOD_OBJ); | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |         case PYTHON_CLASSMETHOD: | 
					
						
							| 
									
										
										
										
											2022-12-23 14:42:24 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_CLASS_METHOD_OBJ); | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case NON_OVERRIDING: | 
					
						
							| 
									
										
										
										
											2022-12-23 14:42:24 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_CLASS_ATTR_DESCRIPTOR); | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case NON_DESCRIPTOR: | 
					
						
							| 
									
										
										
										
											2022-12-23 14:42:24 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_ATTR, SPEC_FAIL_ATTR_CLASS_ATTR_SIMPLE); | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  |         case ABSENT: | 
					
						
							| 
									
										
										
										
											2022-12-16 15:41:23 +00:00
										 |  |  |             if (specialize_dict_access(owner, instr, type, kind, name, STORE_ATTR, | 
					
						
							|  |  |  |                                     STORE_ATTR_INSTANCE_VALUE, STORE_ATTR_WITH_HINT)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | fail: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(STORE_ATTR, failure); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = STORE_ATTR; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2022-11-17 15:09:18 -08:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | success: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(STORE_ATTR, success); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-06-10 08:46:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-09 10:40:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  | load_attr_fail_kind(DescriptorClassification kind) | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (kind) { | 
					
						
							|  |  |  |         case OVERRIDING: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_OVERRIDING_DESCRIPTOR; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case METHOD: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_METHOD; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case PROPERTY: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_PROPERTY; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case OBJECT_SLOT: | 
					
						
							| 
									
										
										
										
											2022-06-14 16:44:01 +01:00
										 |  |  |             return SPEC_FAIL_ATTR_OBJECT_SLOT; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case OTHER_SLOT: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_NON_OBJECT_SLOT; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case DUNDER_CLASS: | 
					
						
							|  |  |  |             return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  |         case MUTABLE: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_MUTABLE_CLASS; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case GETSET_OVERRIDDEN: | 
					
						
							| 
									
										
										
										
											2022-09-09 14:42:29 -07:00
										 |  |  |         case GETATTRIBUTE_IS_PYTHON_FUNCTION: | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |             return SPEC_FAIL_OVERRIDDEN; | 
					
						
							|  |  |  |         case BUILTIN_CLASSMETHOD: | 
					
						
							| 
									
										
										
										
											2022-06-14 16:44:01 +01:00
										 |  |  |             return SPEC_FAIL_ATTR_BUILTIN_CLASS_METHOD; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case PYTHON_CLASSMETHOD: | 
					
						
							| 
									
										
										
										
											2022-06-14 16:44:01 +01:00
										 |  |  |             return SPEC_FAIL_ATTR_CLASS_METHOD_OBJ; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case NON_OVERRIDING: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_NON_OVERRIDING_DESCRIPTOR; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case NON_DESCRIPTOR: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             return SPEC_FAIL_ATTR_NOT_DESCRIPTOR; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |         case ABSENT: | 
					
						
							| 
									
										
										
										
											2022-06-14 16:44:01 +01:00
										 |  |  |             return SPEC_FAIL_ATTR_INSTANCE_ATTRIBUTE; | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-17 17:48:44 +08:00
										 |  |  |     Py_UNREACHABLE(); | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  | specialize_class_load_attr(PyObject *owner, _Py_CODEUNIT *instr, | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |                              PyObject *name) | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     assert(PyType_Check(owner)); | 
					
						
							|  |  |  |     PyTypeObject *cls = (PyTypeObject *)owner; | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     _PyLoadMethodCache *cache = (_PyLoadMethodCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     if (Py_TYPE(cls)->tp_getattro != _Py_type_getattro) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_METACLASS_OVERRIDDEN); | 
					
						
							| 
									
										
										
										
											2022-07-18 10:10:22 -07:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     PyObject *metadescriptor = _PyType_Lookup(Py_TYPE(cls), name); | 
					
						
							|  |  |  |     DescriptorClassification metakind = classify_descriptor(metadescriptor, false); | 
					
						
							|  |  |  |     switch (metakind) { | 
					
						
							|  |  |  |         case METHOD: | 
					
						
							|  |  |  |         case NON_DESCRIPTOR: | 
					
						
							|  |  |  |         case NON_OVERRIDING: | 
					
						
							|  |  |  |         case BUILTIN_CLASSMETHOD: | 
					
						
							|  |  |  |         case PYTHON_CLASSMETHOD: | 
					
						
							|  |  |  |         case ABSENT: | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_METACLASS_ATTRIBUTE); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |     PyObject *descr = NULL; | 
					
						
							| 
									
										
										
										
											2022-01-24 03:33:50 +05:30
										 |  |  |     DescriptorClassification kind = 0; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     kind = analyze_descriptor(cls, name, &descr, 0); | 
					
						
							|  |  |  |     if (type_get_version(cls, LOAD_ATTR) == 0) { | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |     bool metaclass_check = false; | 
					
						
							|  |  |  |     if ((Py_TYPE(cls)->tp_flags & Py_TPFLAGS_IMMUTABLETYPE) == 0) { | 
					
						
							|  |  |  |         metaclass_check = true; | 
					
						
							|  |  |  |         if (type_get_version(Py_TYPE(cls), LOAD_ATTR) == 0) { | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |     switch (kind) { | 
					
						
							|  |  |  |         case METHOD: | 
					
						
							|  |  |  |         case NON_DESCRIPTOR: | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             write_u32(cache->type_version, cls->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |             write_obj(cache->descr, descr); | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |             if (metaclass_check) { | 
					
						
							|  |  |  |                 write_u32(cache->keys_version, Py_TYPE(cls)->tp_version_tag); | 
					
						
							|  |  |  |                 instr->op.code = LOAD_ATTR_CLASS_WITH_METACLASS_CHECK; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 instr->op.code = LOAD_ATTR_CLASS; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-02-07 14:30:34 +00:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  |         case ABSENT: | 
					
						
							| 
									
										
										
										
											2022-07-18 10:10:22 -07:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							| 
									
										
										
										
											2022-02-07 14:30:34 +00:00
										 |  |  |             return -1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, load_attr_fail_kind(kind)); | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  | // Please collect stats carefully before and after modifying. A subtle change
 | 
					
						
							|  |  |  | // can cause a significant drop in cache hits. A possible test is
 | 
					
						
							|  |  |  | // python.exe -m test_typing test_re test_dis test_zlib.
 | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  | specialize_attr_loadclassattr(PyObject *owner, _Py_CODEUNIT *instr, PyObject *name, | 
					
						
							|  |  |  | PyObject *descr, DescriptorClassification kind, bool is_method) | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     _PyLoadMethodCache *cache = (_PyLoadMethodCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |     PyTypeObject *owner_cls = Py_TYPE(owner); | 
					
						
							| 
									
										
										
										
											2022-02-24 19:34:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |     assert(descr != NULL); | 
					
						
							|  |  |  |     assert((is_method && kind == METHOD) || (!is_method && kind == NON_DESCRIPTOR)); | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |     if (owner_cls->tp_flags & Py_TPFLAGS_INLINE_VALUES) { | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |         PyDictKeysObject *keys = ((PyHeapTypeObject *)owner_cls)->ht_cached_keys; | 
					
						
							| 
									
										
										
										
											2024-08-23 10:22:35 +01:00
										 |  |  |         assert(_PyDictKeys_StringLookup(keys, name) < 0); | 
					
						
							| 
									
										
										
										
											2023-03-08 18:04:16 -07:00
										 |  |  |         uint32_t keys_version = _PyDictKeys_GetVersionForCurrentState( | 
					
						
							|  |  |  |                 _PyInterpreterState_GET(), keys); | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |         if (keys_version == 0) { | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |         write_u32(cache->keys_version, keys_version); | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |         instr->op.code = is_method ? LOAD_ATTR_METHOD_WITH_VALUES : LOAD_ATTR_NONDESCRIPTOR_WITH_VALUES; | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |         Py_ssize_t dictoffset; | 
					
						
							|  |  |  |         if (owner_cls->tp_flags & Py_TPFLAGS_MANAGED_DICT) { | 
					
						
							|  |  |  |             dictoffset = MANAGED_DICT_OFFSET; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             dictoffset = owner_cls->tp_dictoffset; | 
					
						
							|  |  |  |             if (dictoffset < 0 || dictoffset > INT16_MAX + MANAGED_DICT_OFFSET) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |                 return 0; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (dictoffset == 0) { | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |             instr->op.code = is_method ? LOAD_ATTR_METHOD_NO_DICT : LOAD_ATTR_NONDESCRIPTOR_NO_DICT; | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |         else if (is_method) { | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |             PyObject *dict = *(PyObject **) ((char *)owner + dictoffset); | 
					
						
							|  |  |  |             if (dict) { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_NOT_MANAGED_DICT); | 
					
						
							|  |  |  |                 return 0; | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |             /* Cache entries must be unsigned values, so we offset the
 | 
					
						
							|  |  |  |              * dictoffset by MANAGED_DICT_OFFSET. | 
					
						
							| 
									
										
										
										
											2024-06-20 11:16:14 +08:00
										 |  |  |              * We do the reverse offset in LOAD_ATTR_METHOD_LAZY_DICT */ | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |             dictoffset -= MANAGED_DICT_OFFSET; | 
					
						
							|  |  |  |             assert(((uint16_t)dictoffset) == dictoffset); | 
					
						
							|  |  |  |             cache->dict_offset = (uint16_t)dictoffset; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = LOAD_ATTR_METHOD_LAZY_DICT; | 
					
						
							| 
									
										
										
										
											2023-01-31 13:28:32 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-07-10 11:40:35 +01:00
										 |  |  |         else { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(LOAD_ATTR, SPEC_FAIL_ATTR_CLASS_ATTR_SIMPLE); | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-10-13 14:19:34 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-17 18:47:36 +08:00
										 |  |  |     /* `descr` is borrowed. This is safe for methods (even inherited ones from
 | 
					
						
							|  |  |  |     *  super classes!) as long as tp_version_tag is validated for two main reasons: | 
					
						
							| 
									
										
										
										
											2021-08-27 09:21:01 +01:00
										 |  |  |     * | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |     *  1. The class will always hold a reference to the method so it will | 
					
						
							|  |  |  |     *  usually not be GC-ed. Should it be deleted in Python, e.g. | 
					
						
							|  |  |  |     *  `del obj.meth`, tp_version_tag will be invalidated, because of reason 2. | 
					
						
							| 
									
										
										
										
											2021-08-27 09:21:01 +01:00
										 |  |  |     * | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |     *  2. The pre-existing type method cache (MCACHE) uses the same principles | 
					
						
							| 
									
										
										
										
											2021-09-17 18:47:36 +08:00
										 |  |  |     *  of caching a borrowed descriptor. The MCACHE infrastructure does all the | 
					
						
							|  |  |  |     *  heavy lifting for us. E.g. it invalidates tp_version_tag on any MRO | 
					
						
							|  |  |  |     *  modification, on any type object change along said MRO, etc. (see | 
					
						
							|  |  |  |     *  PyType_Modified usages in typeobject.c). The MCACHE has been | 
					
						
							|  |  |  |     *  working since Python 2.6 and it's battle-tested. | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  |     */ | 
					
						
							| 
									
										
										
										
											2022-03-03 15:31:00 -08:00
										 |  |  |     write_u32(cache->type_version, owner_cls->tp_version_tag); | 
					
						
							|  |  |  |     write_obj(cache->descr, descr); | 
					
						
							| 
									
										
										
										
											2022-06-14 18:36:22 +08:00
										 |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											2021-08-17 22:55:55 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-08-27 09:21:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | _Py_Specialize_LoadGlobal( | 
					
						
							|  |  |  |     PyObject *globals, PyObject *builtins, | 
					
						
							| 
									
										
										
										
											2022-02-28 12:56:29 +00:00
										 |  |  |     _Py_CODEUNIT *instr, PyObject *name) | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  |     assert(_PyOpcode_Caches[LOAD_GLOBAL] == INLINE_CACHE_ENTRIES_LOAD_GLOBAL); | 
					
						
							| 
									
										
										
										
											2022-02-28 12:56:29 +00:00
										 |  |  |     /* Use inline cache */ | 
					
						
							|  |  |  |     _PyLoadGlobalCache *cache = (_PyLoadGlobalCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     assert(PyUnicode_CheckExact(name)); | 
					
						
							|  |  |  |     if (!PyDict_CheckExact(globals)) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_LOAD_GLOBAL_NON_DICT); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-17 12:20:51 +01:00
										 |  |  |     PyDictKeysObject * globals_keys = ((PyDictObject *)globals)->ma_keys; | 
					
						
							| 
									
										
										
										
											2022-03-03 15:17:18 +00:00
										 |  |  |     if (!DK_IS_UNICODE(globals_keys)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_LOAD_GLOBAL_NON_STRING_OR_SPLIT); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-17 12:20:51 +01:00
										 |  |  |     Py_ssize_t index = _PyDictKeys_StringLookup(globals_keys, name); | 
					
						
							|  |  |  |     if (index == DKIX_ERROR) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-08 18:04:16 -07:00
										 |  |  |     PyInterpreterState *interp = _PyInterpreterState_GET(); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     if (index != DKIX_EMPTY) { | 
					
						
							|  |  |  |         if (index != (uint16_t)index) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-08 18:04:16 -07:00
										 |  |  |         uint32_t keys_version = _PyDictKeys_GetVersionForCurrentState( | 
					
						
							|  |  |  |                 interp, globals_keys); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         if (keys_version == 0) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |             SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-10 17:01:16 -08:00
										 |  |  |         if (keys_version != (uint16_t)keys_version) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-02-28 12:56:29 +00:00
										 |  |  |         cache->index = (uint16_t)index; | 
					
						
							| 
									
										
										
										
											2023-03-10 17:01:16 -08:00
										 |  |  |         cache->module_keys_version = (uint16_t)keys_version; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = LOAD_GLOBAL_MODULE; | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!PyDict_CheckExact(builtins)) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_LOAD_GLOBAL_NON_DICT); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-17 12:20:51 +01:00
										 |  |  |     PyDictKeysObject * builtin_keys = ((PyDictObject *)builtins)->ma_keys; | 
					
						
							| 
									
										
										
										
											2022-03-03 15:17:18 +00:00
										 |  |  |     if (!DK_IS_UNICODE(builtin_keys)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_LOAD_GLOBAL_NON_STRING_OR_SPLIT); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-17 12:20:51 +01:00
										 |  |  |     index = _PyDictKeys_StringLookup(builtin_keys, name); | 
					
						
							|  |  |  |     if (index == DKIX_ERROR) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (index != (uint16_t)index) { | 
					
						
							| 
									
										
										
										
											2022-12-14 15:50:02 +00:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-08 18:04:16 -07:00
										 |  |  |     uint32_t globals_version = _PyDictKeys_GetVersionForCurrentState( | 
					
						
							|  |  |  |             interp, globals_keys); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     if (globals_version == 0) { | 
					
						
							| 
									
										
										
										
											2021-09-17 12:20:51 +01:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-10 17:01:16 -08:00
										 |  |  |     if (globals_version != (uint16_t)globals_version) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-08 18:04:16 -07:00
										 |  |  |     uint32_t builtins_version = _PyDictKeys_GetVersionForCurrentState( | 
					
						
							|  |  |  |             interp, builtin_keys); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     if (builtins_version == 0) { | 
					
						
							| 
									
										
										
										
											2021-09-17 12:20:51 +01:00
										 |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-28 12:56:29 +00:00
										 |  |  |     if (builtins_version > UINT16_MAX) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(LOAD_GLOBAL, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     cache->index = (uint16_t)index; | 
					
						
							| 
									
										
										
										
											2023-03-10 17:01:16 -08:00
										 |  |  |     cache->module_keys_version = (uint16_t)globals_version; | 
					
						
							| 
									
										
										
										
											2022-02-28 12:56:29 +00:00
										 |  |  |     cache->builtin_keys_version = (uint16_t)builtins_version; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = LOAD_GLOBAL_BUILTIN; | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     goto success; | 
					
						
							|  |  |  | fail: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(LOAD_GLOBAL, failure); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = LOAD_GLOBAL; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | success: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(LOAD_GLOBAL, success); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-06-14 11:04:09 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-15 15:32:32 +00:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  | binary_subscr_fail_kind(PyTypeObject *container_type, PyObject *sub) | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-08-08 13:42:43 -07:00
										 |  |  |     if (strcmp(container_type->tp_name, "array.array") == 0) { | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             return SPEC_FAIL_SUBSCR_ARRAY_INT; | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (PySlice_Check(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             return SPEC_FAIL_SUBSCR_ARRAY_SLICE; | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (container_type->tp_as_buffer) { | 
					
						
							|  |  |  |         if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             return SPEC_FAIL_SUBSCR_BUFFER_INT; | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (PySlice_Check(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             return SPEC_FAIL_SUBSCR_BUFFER_SLICE; | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (container_type->tp_as_sequence) { | 
					
						
							|  |  |  |         if (PyLong_CheckExact(sub) && container_type->tp_as_sequence->sq_item) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             return SPEC_FAIL_SUBSCR_SEQUENCE_INT; | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2021-09-15 14:55:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | function_kind(PyCodeObject *code) { | 
					
						
							|  |  |  |     int flags = code->co_flags; | 
					
						
							|  |  |  |     if ((flags & (CO_VARKEYWORDS | CO_VARARGS)) || code->co_kwonlyargcount) { | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  |         return SPEC_FAIL_CODE_COMPLEX_PARAMETERS; | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if ((flags & CO_OPTIMIZED) == 0) { | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  |         return SPEC_FAIL_CODE_NOT_OPTIMIZED; | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return SIMPLE_FUNCTION; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-17 19:37:07 +08:00
										 |  |  | /* Returning false indicates a failure. */ | 
					
						
							|  |  |  | static bool | 
					
						
							|  |  |  | function_check_args(PyObject *o, int expected_argcount, int opcode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(Py_IS_TYPE(o, &PyFunction_Type)); | 
					
						
							|  |  |  |     PyFunctionObject *func = (PyFunctionObject *)o; | 
					
						
							|  |  |  |     PyCodeObject *fcode = (PyCodeObject *)func->func_code; | 
					
						
							|  |  |  |     int kind = function_kind(fcode); | 
					
						
							|  |  |  |     if (kind != SIMPLE_FUNCTION) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(opcode, kind); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (fcode->co_argcount != expected_argcount) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(opcode, SPEC_FAIL_WRONG_NUMBER_ARGUMENTS); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Returning 0 indicates a failure. */ | 
					
						
							|  |  |  | static uint32_t | 
					
						
							|  |  |  | function_get_version(PyObject *o, int opcode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(Py_IS_TYPE(o, &PyFunction_Type)); | 
					
						
							|  |  |  |     PyFunctionObject *func = (PyFunctionObject *)o; | 
					
						
							|  |  |  |     uint32_t version = _PyFunction_GetVersionForCurrentState(func); | 
					
						
							|  |  |  |     if (version == 0) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(opcode, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return version; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  | /* Returning 0 indicates a failure. */ | 
					
						
							|  |  |  | static uint32_t | 
					
						
							|  |  |  | type_get_version(PyTypeObject *t, int opcode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t version = t->tp_version_tag; | 
					
						
							|  |  |  |     if (version == 0) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(opcode, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return version; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  | _Py_Specialize_BinarySubscr( | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |      _PyStackRef container_st, _PyStackRef sub_st, _Py_CODEUNIT *instr) | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *container = PyStackRef_AsPyObjectBorrow(container_st); | 
					
						
							|  |  |  |     PyObject *sub = PyStackRef_AsPyObjectBorrow(sub_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  |     assert(_PyOpcode_Caches[BINARY_SUBSCR] == | 
					
						
							| 
									
										
										
										
											2022-03-01 16:00:34 +00:00
										 |  |  |            INLINE_CACHE_ENTRIES_BINARY_SUBSCR); | 
					
						
							|  |  |  |     _PyBinarySubscrCache *cache = (_PyBinarySubscrCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |     PyTypeObject *container_type = Py_TYPE(container); | 
					
						
							|  |  |  |     if (container_type == &PyList_Type) { | 
					
						
							|  |  |  |         if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2023-03-22 14:49:51 +00:00
										 |  |  |             if (_PyLong_IsNonNegativeCompact((PyLongObject *)sub)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_SUBSCR_LIST_INT; | 
					
						
							| 
									
										
										
										
											2022-12-20 15:46:16 -05:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         SPECIALIZATION_FAIL(BINARY_SUBSCR, | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             PySlice_Check(sub) ? SPEC_FAIL_SUBSCR_LIST_SLICE : SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         goto fail; | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (container_type == &PyTuple_Type) { | 
					
						
							|  |  |  |         if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2023-03-22 14:49:51 +00:00
										 |  |  |             if (_PyLong_IsNonNegativeCompact((PyLongObject *)sub)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_SUBSCR_TUPLE_INT; | 
					
						
							| 
									
										
										
										
											2022-12-20 15:46:16 -05:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         SPECIALIZATION_FAIL(BINARY_SUBSCR, | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             PySlice_Check(sub) ? SPEC_FAIL_SUBSCR_TUPLE_SLICE : SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |         goto fail; | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-08-08 13:42:43 -07:00
										 |  |  |     if (container_type == &PyUnicode_Type) { | 
					
						
							|  |  |  |         if (PyLong_CheckExact(sub)) { | 
					
						
							|  |  |  |             if (_PyLong_IsNonNegativeCompact((PyLongObject *)sub)) { | 
					
						
							|  |  |  |                 instr->op.code = BINARY_SUBSCR_STR_INT; | 
					
						
							|  |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(BINARY_SUBSCR, | 
					
						
							|  |  |  |             PySlice_Check(sub) ? SPEC_FAIL_SUBSCR_STRING_SLICE : SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |     if (container_type == &PyDict_Type) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = BINARY_SUBSCR_DICT; | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyTypeObject *cls = Py_TYPE(container); | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |     PyObject *descriptor = _PyType_Lookup(cls, &_Py_ID(__getitem__)); | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |     if (descriptor && Py_TYPE(descriptor) == &PyFunction_Type) { | 
					
						
							| 
									
										
										
										
											2022-03-01 16:00:34 +00:00
										 |  |  |         if (!(container_type->tp_flags & Py_TPFLAGS_HEAPTYPE)) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_SUBSCR_NOT_HEAP_TYPE); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |         PyFunctionObject *func = (PyFunctionObject *)descriptor; | 
					
						
							| 
									
										
										
										
											2022-03-01 16:00:34 +00:00
										 |  |  |         PyCodeObject *fcode = (PyCodeObject *)func->func_code; | 
					
						
							|  |  |  |         int kind = function_kind(fcode); | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |         if (kind != SIMPLE_FUNCTION) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, kind); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-03-01 16:00:34 +00:00
										 |  |  |         if (fcode->co_argcount != 2) { | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_WRONG_NUMBER_ARGUMENTS); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-29 15:53:30 -07:00
										 |  |  |         uint32_t version = _PyFunction_GetVersionForCurrentState(func); | 
					
						
							|  |  |  |         if (version == 0) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |         if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(BINARY_SUBSCR, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-29 15:53:30 -07:00
										 |  |  |         PyHeapTypeObject *ht = (PyHeapTypeObject *)container_type; | 
					
						
							|  |  |  |         // This pointer is invalidated by PyType_Modified (see the comment on
 | 
					
						
							|  |  |  |         // struct _specialization_cache):
 | 
					
						
							|  |  |  |         ht->_spec_cache.getitem = descriptor; | 
					
						
							|  |  |  |         ht->_spec_cache.getitem_version = version; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = BINARY_SUBSCR_GETITEM; | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-27 12:01:22 +01:00
										 |  |  |     SPECIALIZATION_FAIL(BINARY_SUBSCR, | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |                         binary_subscr_fail_kind(container_type, sub)); | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  | fail: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(BINARY_SUBSCR, failure); | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = BINARY_SUBSCR; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  | success: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(BINARY_SUBSCR, success); | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-07-15 13:13:12 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_StoreSubscr(_PyStackRef container_st, _PyStackRef sub_st, _Py_CODEUNIT *instr) | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *container = PyStackRef_AsPyObjectBorrow(container_st); | 
					
						
							|  |  |  |     PyObject *sub = PyStackRef_AsPyObjectBorrow(sub_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-03-08 07:53:22 -08:00
										 |  |  |     _PyStoreSubscrCache *cache = (_PyStoreSubscrCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |     PyTypeObject *container_type = Py_TYPE(container); | 
					
						
							|  |  |  |     if (container_type == &PyList_Type) { | 
					
						
							|  |  |  |         if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2023-03-22 14:49:51 +00:00
										 |  |  |             if (_PyLong_IsNonNegativeCompact((PyLongObject *)sub) | 
					
						
							| 
									
										
										
										
											2023-01-30 10:03:04 +00:00
										 |  |  |                 && ((PyLongObject *)sub)->long_value.ob_digit[0] < (size_t)PyList_GET_SIZE(container)) | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = STORE_SUBSCR_LIST_INT; | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |                 goto fail; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (PySlice_Check(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_LIST_SLICE); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             goto fail; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |     if (container_type == &PyDict_Type) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = STORE_SUBSCR_DICT; | 
					
						
							|  |  |  |         goto success; | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  |     PyMappingMethods *as_mapping = container_type->tp_as_mapping; | 
					
						
							|  |  |  |     if (as_mapping && (as_mapping->mp_ass_subscript | 
					
						
							|  |  |  |                        == PyDict_Type.tp_as_mapping->mp_ass_subscript)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |         SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_DICT_SUBCLASS_NO_OVERRIDE); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |     if (PyObject_CheckBuffer(container)) { | 
					
						
							| 
									
										
										
										
											2023-05-09 08:53:19 -06:00
										 |  |  |         if (PyLong_CheckExact(sub) && (!_PyLong_IsNonNegativeCompact((PyLongObject *)sub))) { | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OUT_OF_RANGE); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (strcmp(container_type->tp_name, "array.array") == 0) { | 
					
						
							|  |  |  |             if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_ARRAY_INT); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             else if (PySlice_Check(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_ARRAY_SLICE); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (PyByteArray_CheckExact(container)) { | 
					
						
							|  |  |  |             if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_BYTEARRAY_INT); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             else if (PySlice_Check(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_BYTEARRAY_SLICE); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             if (PyLong_CheckExact(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_BUFFER_INT); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             else if (PySlice_Check(sub)) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_BUFFER_SLICE); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |     PyObject *descriptor = _PyType_Lookup(container_type, &_Py_ID(__setitem__)); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |     if (descriptor && Py_TYPE(descriptor) == &PyFunction_Type) { | 
					
						
							|  |  |  |         PyFunctionObject *func = (PyFunctionObject *)descriptor; | 
					
						
							|  |  |  |         PyCodeObject *code = (PyCodeObject *)func->func_code; | 
					
						
							|  |  |  |         int kind = function_kind(code); | 
					
						
							|  |  |  |         if (kind == SIMPLE_FUNCTION) { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_PY_SIMPLE); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2022-02-09 12:30:26 +00:00
										 |  |  |             SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_SUBSCR_PY_OTHER); | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |         goto fail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2022-01-04 13:05:09 -05:00
										 |  |  |     SPECIALIZATION_FAIL(STORE_SUBSCR, SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  | fail: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(STORE_SUBSCR, failure); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = STORE_SUBSCR; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  | success: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(STORE_SUBSCR, success); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-11-19 05:30:37 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  | /* Returns a borrowed reference.
 | 
					
						
							|  |  |  |  * The reference is only valid if guarded by a type version check. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static PyFunctionObject * | 
					
						
							|  |  |  | get_init_for_simple_managed_python_class(PyTypeObject *tp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(tp->tp_new == PyBaseObject_Type.tp_new); | 
					
						
							|  |  |  |     if (tp->tp_alloc != PyType_GenericAlloc) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_OVERRIDDEN); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-02 11:59:21 +01:00
										 |  |  |     if ((tp->tp_flags & Py_TPFLAGS_INLINE_VALUES) == 0) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CALL_INIT_NOT_INLINE_VALUES); | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!(tp->tp_flags & Py_TPFLAGS_HEAPTYPE)) { | 
					
						
							|  |  |  |         /* Is this possible? */ | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyObject *init = _PyType_Lookup(tp, &_Py_ID(__init__)); | 
					
						
							|  |  |  |     if (init == NULL || !PyFunction_Check(init)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CALL_INIT_NOT_PYTHON); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int kind = function_kind((PyCodeObject *)PyFunction_GET_CODE(init)); | 
					
						
							|  |  |  |     if (kind != SIMPLE_FUNCTION) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CALL_INIT_NOT_SIMPLE); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ((PyHeapTypeObject *)tp)->_spec_cache.init = init; | 
					
						
							|  |  |  |     return (PyFunctionObject *)init; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  | specialize_class_call(PyObject *callable, _Py_CODEUNIT *instr, int nargs) | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |     assert(PyType_Check(callable)); | 
					
						
							| 
									
										
										
										
											2022-01-21 23:30:17 +01:00
										 |  |  |     PyTypeObject *tp = _PyType_CAST(callable); | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |     if (tp->tp_flags & Py_TPFLAGS_IMMUTABLETYPE) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         int oparg = instr->op.arg; | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |         if (nargs == 1 && oparg == 1) { | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |             if (tp == &PyUnicode_Type) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                 instr->op.code = CALL_STR_1; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |                 return 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (tp == &PyType_Type) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                 instr->op.code = CALL_TYPE_1; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |                 return 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (tp == &PyTuple_Type) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                 instr->op.code = CALL_TUPLE_1; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |                 return 0; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-12-15 15:03:42 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |         if (tp->tp_vectorcall != NULL) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = CALL_BUILTIN_CLASS; | 
					
						
							| 
									
										
										
										
											2021-12-15 15:03:42 +00:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-08-22 03:50:55 -07:00
										 |  |  |         goto generic; | 
					
						
							| 
									
										
										
										
											2021-12-15 15:03:42 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-01 19:39:32 +00:00
										 |  |  |     if (Py_TYPE(tp) != &PyType_Type) { | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |         goto generic; | 
					
						
							| 
									
										
										
										
											2024-02-01 19:39:32 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |     if (tp->tp_new == PyBaseObject_Type.tp_new) { | 
					
						
							|  |  |  |         PyFunctionObject *init = get_init_for_simple_managed_python_class(tp); | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  |         if (type_get_version(tp, CALL) == 0) { | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |         if (init != NULL) { | 
					
						
							|  |  |  |             _PyCallCache *cache = (_PyCallCache *)(instr + 1); | 
					
						
							|  |  |  |             write_u32(cache->func_version, tp->tp_version_tag); | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |             _Py_SET_OPCODE(*instr, CALL_ALLOC_AND_ENTER_INIT); | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  | generic: | 
					
						
							|  |  |  |     instr->op.code = CALL_NON_PY_GENERAL; | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  | specialize_method_descriptor(PyMethodDescrObject *descr, _Py_CODEUNIT *instr, | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                              int nargs) | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (descr->d_method->ml_flags & | 
					
						
							|  |  |  |         (METH_VARARGS | METH_FASTCALL | METH_NOARGS | METH_O | | 
					
						
							|  |  |  |         METH_KEYWORDS | METH_METHOD)) { | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |         case METH_NOARGS: { | 
					
						
							|  |  |  |             if (nargs != 1) { | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |                 SPECIALIZATION_FAIL(CALL, SPEC_FAIL_WRONG_NUMBER_ARGUMENTS); | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |                 return -1; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |             instr->op.code = CALL_METHOD_DESCRIPTOR_NOARGS; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |         case METH_O: { | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |             if (nargs != 2) { | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |                 SPECIALIZATION_FAIL(CALL, SPEC_FAIL_WRONG_NUMBER_ARGUMENTS); | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  |             PyInterpreterState *interp = _PyInterpreterState_GET(); | 
					
						
							|  |  |  |             PyObject *list_append = interp->callable_cache.list_append; | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |             _Py_CODEUNIT next = instr[INLINE_CACHE_ENTRIES_CALL + 1]; | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             bool pop = (next.op.code == POP_TOP); | 
					
						
							|  |  |  |             int oparg = instr->op.arg; | 
					
						
							| 
									
										
										
										
											2022-04-05 06:18:30 -04:00
										 |  |  |             if ((PyObject *)descr == list_append && oparg == 1 && pop) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                 instr->op.code = CALL_LIST_APPEND; | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  |                 return 0; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |             instr->op.code = CALL_METHOD_DESCRIPTOR_O; | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case METH_FASTCALL: { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |             instr->op.code = CALL_METHOD_DESCRIPTOR_FAST; | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-12-22 20:44:59 +08:00
										 |  |  |         case METH_FASTCALL | METH_KEYWORDS: { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = CALL_METHOD_DESCRIPTOR_FAST_WITH_KEYWORDS; | 
					
						
							| 
									
										
										
										
											2022-03-28 01:23:25 +05:30
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |     instr->op.code = CALL_NON_PY_GENERAL; | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  | specialize_py_call(PyFunctionObject *func, _Py_CODEUNIT *instr, int nargs, | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                    bool bound_method) | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  |     _PyCallCache *cache = (_PyCallCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |     PyCodeObject *code = (PyCodeObject *)func->func_code; | 
					
						
							| 
									
										
										
										
											2021-11-18 11:02:14 +00:00
										 |  |  |     int kind = function_kind(code); | 
					
						
							| 
									
										
										
										
											2022-05-04 09:31:21 -06:00
										 |  |  |     /* Don't specialize if PEP 523 is active */ | 
					
						
							|  |  |  |     if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CALL_PEP_523); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |     int argcount = -1; | 
					
						
							|  |  |  |     if (kind == SPEC_FAIL_CODE_NOT_OPTIMIZED) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CODE_NOT_OPTIMIZED); | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |     if (kind == SIMPLE_FUNCTION) { | 
					
						
							|  |  |  |         argcount = code->co_argcount; | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     int version = _PyFunction_GetVersionForCurrentState(func); | 
					
						
							|  |  |  |     if (version == 0) { | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  |     write_u32(cache->func_version, version); | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |     if (argcount == nargs + bound_method) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = bound_method ? CALL_BOUND_METHOD_EXACT_ARGS : CALL_PY_EXACT_ARGS; | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |         instr->op.code = bound_method ? CALL_BOUND_METHOD_GENERAL : CALL_PY_GENERAL; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-16 17:11:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | specialize_py_call_kw(PyFunctionObject *func, _Py_CODEUNIT *instr, int nargs, | 
					
						
							|  |  |  |                    bool bound_method) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyCallCache *cache = (_PyCallCache *)(instr + 1); | 
					
						
							|  |  |  |     PyCodeObject *code = (PyCodeObject *)func->func_code; | 
					
						
							|  |  |  |     int kind = function_kind(code); | 
					
						
							|  |  |  |     /* Don't specialize if PEP 523 is active */ | 
					
						
							|  |  |  |     if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CALL_PEP_523); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (kind == SPEC_FAIL_CODE_NOT_OPTIMIZED) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CODE_NOT_OPTIMIZED); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int version = _PyFunction_GetVersionForCurrentState(func); | 
					
						
							|  |  |  |     if (version == 0) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     write_u32(cache->func_version, version); | 
					
						
							|  |  |  |     instr->op.code = bound_method ? CALL_KW_BOUND_METHOD : CALL_KW_PY; | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  | specialize_c_call(PyObject *callable, _Py_CODEUNIT *instr, int nargs) | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  | { | 
					
						
							|  |  |  |     if (PyCFunction_GET_FUNCTION(callable) == NULL) { | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |         SPECIALIZATION_FAIL(CALL, SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     switch (PyCFunction_GET_FLAGS(callable) & | 
					
						
							|  |  |  |         (METH_VARARGS | METH_FASTCALL | METH_NOARGS | METH_O | | 
					
						
							|  |  |  |         METH_KEYWORDS | METH_METHOD)) { | 
					
						
							|  |  |  |         case METH_O: { | 
					
						
							|  |  |  |             if (nargs != 1) { | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |                 SPECIALIZATION_FAIL(CALL, SPEC_FAIL_WRONG_NUMBER_ARGUMENTS); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |                 return 1; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             /* len(o) */ | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  |             PyInterpreterState *interp = _PyInterpreterState_GET(); | 
					
						
							|  |  |  |             if (callable == interp->callable_cache.len) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                 instr->op.code = CALL_LEN; | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |                 return 0; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |             instr->op.code = CALL_BUILTIN_O; | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         case METH_FASTCALL: { | 
					
						
							|  |  |  |             if (nargs == 2) { | 
					
						
							|  |  |  |                 /* isinstance(o1, o2) */ | 
					
						
							| 
									
										
										
										
											2022-03-07 11:45:00 -08:00
										 |  |  |                 PyInterpreterState *interp = _PyInterpreterState_GET(); | 
					
						
							|  |  |  |                 if (callable == interp->callable_cache.isinstance) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |                     instr->op.code = CALL_ISINSTANCE; | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |                     return 0; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |             instr->op.code = CALL_BUILTIN_FAST; | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |         case METH_FASTCALL | METH_KEYWORDS: { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = CALL_BUILTIN_FAST_WITH_KEYWORDS; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |             instr->op.code = CALL_NON_PY_GENERAL; | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-02-22 14:57:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_Call(_PyStackRef callable_st, _Py_CODEUNIT *instr, int nargs) | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *callable = PyStackRef_AsPyObjectBorrow(callable_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |     assert(_PyOpcode_Caches[CALL] == INLINE_CACHE_ENTRIES_CALL); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(_Py_OPCODE(*instr) != INSTRUMENTED_CALL); | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |     _PyCallCache *cache = (_PyCallCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |     int fail; | 
					
						
							|  |  |  |     if (PyCFunction_CheckExact(callable)) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |         fail = specialize_c_call(callable, instr, nargs); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |     else if (PyFunction_Check(callable)) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |         fail = specialize_py_call((PyFunctionObject *)callable, instr, nargs, false); | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (PyType_Check(callable)) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |         fail = specialize_class_call(callable, instr, nargs); | 
					
						
							| 
									
										
										
										
											2021-10-20 16:08:28 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |     else if (Py_IS_TYPE(callable, &PyMethodDescr_Type)) { | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |         fail = specialize_method_descriptor((PyMethodDescrObject *)callable, instr, nargs); | 
					
						
							| 
									
										
										
										
											2021-12-14 18:22:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-12-16 15:43:04 +00:00
										 |  |  |     else if (PyMethod_Check(callable)) { | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |         PyObject *func = ((PyMethodObject *)callable)->im_func; | 
					
						
							|  |  |  |         if (PyFunction_Check(func)) { | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |             fail = specialize_py_call((PyFunctionObject *)func, instr, nargs, true); | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2022-05-19 11:05:26 +01:00
										 |  |  |             SPECIALIZATION_FAIL(CALL, SPEC_FAIL_CALL_BOUND_METHOD); | 
					
						
							|  |  |  |             fail = -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-02-22 14:57:01 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |     else { | 
					
						
							| 
									
										
										
										
											2024-05-04 12:11:11 +01:00
										 |  |  |         instr->op.code = CALL_NON_PY_GENERAL; | 
					
						
							|  |  |  |         fail = 0; | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (fail) { | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |         STAT_INC(CALL, failure); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |         assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = CALL; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |         cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2022-01-28 12:42:30 +00:00
										 |  |  |         STAT_INC(CALL, success); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |         assert(!PyErr_Occurred()); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |         cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-10-20 07:16:36 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-16 17:11:24 +01:00
										 |  |  | void | 
					
						
							|  |  |  | _Py_Specialize_CallKw(_PyStackRef callable_st, _Py_CODEUNIT *instr, int nargs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyObject *callable = PyStackRef_AsPyObjectBorrow(callable_st); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							|  |  |  |     assert(_PyOpcode_Caches[CALL_KW] == INLINE_CACHE_ENTRIES_CALL_KW); | 
					
						
							|  |  |  |     assert(_Py_OPCODE(*instr) != INSTRUMENTED_CALL_KW); | 
					
						
							|  |  |  |     _PyCallCache *cache = (_PyCallCache *)(instr + 1); | 
					
						
							|  |  |  |     int fail; | 
					
						
							|  |  |  |     if (PyFunction_Check(callable)) { | 
					
						
							|  |  |  |         fail = specialize_py_call_kw((PyFunctionObject *)callable, instr, nargs, false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (PyMethod_Check(callable)) { | 
					
						
							|  |  |  |         PyObject *func = ((PyMethodObject *)callable)->im_func; | 
					
						
							|  |  |  |         if (PyFunction_Check(func)) { | 
					
						
							|  |  |  |             fail = specialize_py_call_kw((PyFunctionObject *)func, instr, nargs, true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(CALL_KW, SPEC_FAIL_CALL_BOUND_METHOD); | 
					
						
							|  |  |  |             fail = -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         instr->op.code = CALL_KW_NON_PY; | 
					
						
							|  |  |  |         fail = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (fail) { | 
					
						
							|  |  |  |         STAT_INC(CALL, failure); | 
					
						
							|  |  |  |         assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |         instr->op.code = CALL_KW; | 
					
						
							|  |  |  |         cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         STAT_INC(CALL, success); | 
					
						
							|  |  |  |         assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |         cache->counter = adaptive_counter_cooldown(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 08:49:58 -08:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | binary_op_fail_kind(int oparg, PyObject *lhs, PyObject *rhs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (oparg) { | 
					
						
							|  |  |  |         case NB_ADD: | 
					
						
							|  |  |  |         case NB_INPLACE_ADD: | 
					
						
							|  |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_ADD_DIFFERENT_TYPES; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_ADD_OTHER; | 
					
						
							|  |  |  |         case NB_AND: | 
					
						
							|  |  |  |         case NB_INPLACE_AND: | 
					
						
							|  |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_AND_DIFFERENT_TYPES; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyLong_CheckExact(lhs)) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_AND_INT; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_AND_OTHER; | 
					
						
							|  |  |  |         case NB_FLOOR_DIVIDE: | 
					
						
							|  |  |  |         case NB_INPLACE_FLOOR_DIVIDE: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_FLOOR_DIVIDE; | 
					
						
							|  |  |  |         case NB_LSHIFT: | 
					
						
							|  |  |  |         case NB_INPLACE_LSHIFT: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_LSHIFT; | 
					
						
							|  |  |  |         case NB_MATRIX_MULTIPLY: | 
					
						
							|  |  |  |         case NB_INPLACE_MATRIX_MULTIPLY: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_MATRIX_MULTIPLY; | 
					
						
							|  |  |  |         case NB_MULTIPLY: | 
					
						
							|  |  |  |         case NB_INPLACE_MULTIPLY: | 
					
						
							|  |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_MULTIPLY_DIFFERENT_TYPES; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_MULTIPLY_OTHER; | 
					
						
							|  |  |  |         case NB_OR: | 
					
						
							|  |  |  |         case NB_INPLACE_OR: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_OR; | 
					
						
							|  |  |  |         case NB_POWER: | 
					
						
							|  |  |  |         case NB_INPLACE_POWER: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_POWER; | 
					
						
							|  |  |  |         case NB_REMAINDER: | 
					
						
							|  |  |  |         case NB_INPLACE_REMAINDER: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_REMAINDER; | 
					
						
							|  |  |  |         case NB_RSHIFT: | 
					
						
							|  |  |  |         case NB_INPLACE_RSHIFT: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_RSHIFT; | 
					
						
							|  |  |  |         case NB_SUBTRACT: | 
					
						
							|  |  |  |         case NB_INPLACE_SUBTRACT: | 
					
						
							|  |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_SUBTRACT_DIFFERENT_TYPES; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_SUBTRACT_OTHER; | 
					
						
							|  |  |  |         case NB_TRUE_DIVIDE: | 
					
						
							|  |  |  |         case NB_INPLACE_TRUE_DIVIDE: | 
					
						
							|  |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_TRUE_DIVIDE_DIFFERENT_TYPES; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyFloat_CheckExact(lhs)) { | 
					
						
							|  |  |  |                 return SPEC_FAIL_BINARY_OP_TRUE_DIVIDE_FLOAT; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_TRUE_DIVIDE_OTHER; | 
					
						
							|  |  |  |         case NB_XOR: | 
					
						
							|  |  |  |         case NB_INPLACE_XOR: | 
					
						
							|  |  |  |             return SPEC_FAIL_BINARY_OP_XOR; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_UNREACHABLE(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2022-02-16 08:49:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_BinaryOp(_PyStackRef lhs_st, _PyStackRef rhs_st, _Py_CODEUNIT *instr, | 
					
						
							|  |  |  |                         int oparg, _PyStackRef *locals) | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *lhs = PyStackRef_AsPyObjectBorrow(lhs_st); | 
					
						
							|  |  |  |     PyObject *rhs = PyStackRef_AsPyObjectBorrow(rhs_st); | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  |     assert(_PyOpcode_Caches[BINARY_OP] == INLINE_CACHE_ENTRIES_BINARY_OP); | 
					
						
							| 
									
										
										
										
											2022-02-25 04:11:34 -08:00
										 |  |  |     _PyBinaryOpCache *cache = (_PyBinaryOpCache *)(instr + 1); | 
					
						
							|  |  |  |     switch (oparg) { | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |         case NB_ADD: | 
					
						
							|  |  |  |         case NB_INPLACE_ADD: | 
					
						
							| 
									
										
										
										
											2021-11-15 08:58:23 -08:00
										 |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							| 
									
										
										
										
											2022-02-16 08:49:58 -08:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2021-11-15 08:58:23 -08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |             if (PyUnicode_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2022-03-04 10:51:27 -08:00
										 |  |  |                 _Py_CODEUNIT next = instr[INLINE_CACHE_ENTRIES_BINARY_OP + 1]; | 
					
						
							| 
									
										
										
										
											2023-06-05 11:07:04 +01:00
										 |  |  |                 bool to_store = (next.op.code == STORE_FAST); | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |                 if (to_store && PyStackRef_AsPyObjectBorrow(locals[next.op.arg]) == lhs) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                     instr->op.code = BINARY_OP_INPLACE_ADD_UNICODE; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |                     goto success; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_ADD_UNICODE; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyLong_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_ADD_INT; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyFloat_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_ADD_FLOAT; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case NB_MULTIPLY: | 
					
						
							|  |  |  |         case NB_INPLACE_MULTIPLY: | 
					
						
							| 
									
										
										
										
											2021-11-15 08:58:23 -08:00
										 |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							| 
									
										
										
										
											2022-02-16 08:49:58 -08:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2021-11-15 08:58:23 -08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |             if (PyLong_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_MULTIPLY_INT; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyFloat_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_MULTIPLY_FLOAT; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-11-18 18:19:58 +09:00
										 |  |  |         case NB_SUBTRACT: | 
					
						
							|  |  |  |         case NB_INPLACE_SUBTRACT: | 
					
						
							| 
									
										
										
										
											2021-12-08 17:17:16 -08:00
										 |  |  |             if (!Py_IS_TYPE(lhs, Py_TYPE(rhs))) { | 
					
						
							| 
									
										
										
										
											2022-02-16 08:49:58 -08:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2021-12-08 17:17:16 -08:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-11-18 18:19:58 +09:00
										 |  |  |             if (PyLong_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_SUBTRACT_INT; | 
					
						
							| 
									
										
										
										
											2021-11-18 18:19:58 +09:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (PyFloat_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |                 instr->op.code = BINARY_OP_SUBTRACT_FLOAT; | 
					
						
							| 
									
										
										
										
											2021-11-18 18:19:58 +09:00
										 |  |  |                 goto success; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-25 04:11:34 -08:00
										 |  |  |     SPECIALIZATION_FAIL(BINARY_OP, binary_op_fail_kind(oparg, lhs, rhs)); | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(BINARY_OP, failure); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = BINARY_OP; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							| 
									
										
										
										
											2021-12-17 14:48:01 +00:00
										 |  |  |     STAT_INC(BINARY_OP, success); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-11-10 22:56:22 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | compare_op_fail_kind(PyObject *lhs, PyObject *rhs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (Py_TYPE(lhs) != Py_TYPE(rhs)) { | 
					
						
							|  |  |  |         if (PyFloat_CheckExact(lhs) && PyLong_CheckExact(rhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |             return SPEC_FAIL_COMPARE_OP_FLOAT_LONG; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (PyLong_CheckExact(lhs) && PyFloat_CheckExact(rhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |             return SPEC_FAIL_COMPARE_OP_LONG_FLOAT; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_DIFFERENT_TYPES; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyBytes_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_BYTES; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyTuple_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_TUPLE; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyList_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_LIST; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PySet_CheckExact(lhs) || PyFrozenSet_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_SET; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyBool_Check(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_BOOL; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (Py_TYPE(lhs)->tp_richcompare == PyBaseObject_Type.tp_richcompare) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         return SPEC_FAIL_COMPARE_OP_BASEOBJECT; | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2022-02-03 08:32:52 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_CompareOp(_PyStackRef lhs_st, _PyStackRef rhs_st, _Py_CODEUNIT *instr, | 
					
						
							| 
									
										
										
										
											2022-03-01 05:53:13 -08:00
										 |  |  |                          int oparg) | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *lhs = PyStackRef_AsPyObjectBorrow(lhs_st); | 
					
						
							|  |  |  |     PyObject *rhs = PyStackRef_AsPyObjectBorrow(rhs_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |     assert(_PyOpcode_Caches[COMPARE_OP] == INLINE_CACHE_ENTRIES_COMPARE_OP); | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  |     // All of these specializations compute boolean values, so they're all valid
 | 
					
						
							|  |  |  |     // regardless of the fifth-lowest oparg bit.
 | 
					
						
							| 
									
										
										
										
											2022-03-01 05:53:13 -08:00
										 |  |  |     _PyCompareOpCache *cache = (_PyCompareOpCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |     if (Py_TYPE(lhs) != Py_TYPE(rhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         SPECIALIZATION_FAIL(COMPARE_OP, compare_op_fail_kind(lhs, rhs)); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyFloat_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |         instr->op.code = COMPARE_OP_FLOAT; | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyLong_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-22 14:49:51 +00:00
										 |  |  |         if (_PyLong_IsCompact((PyLongObject *)lhs) && _PyLong_IsCompact((PyLongObject *)rhs)) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |             instr->op.code = COMPARE_OP_INT; | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |             goto success; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |             SPECIALIZATION_FAIL(COMPARE_OP, SPEC_FAIL_COMPARE_OP_BIG_INT); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyUnicode_CheckExact(lhs)) { | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  |         int cmp = oparg >> 5; | 
					
						
							| 
									
										
										
										
											2023-01-11 20:40:43 +00:00
										 |  |  |         if (cmp != Py_EQ && cmp != Py_NE) { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |             SPECIALIZATION_FAIL(COMPARE_OP, SPEC_FAIL_COMPARE_OP_STRING); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |             instr->op.code = COMPARE_OP_STR; | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |             goto success; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |     SPECIALIZATION_FAIL(COMPARE_OP, compare_op_fail_kind(lhs, rhs)); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  | failure: | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |     STAT_INC(COMPARE_OP, failure); | 
					
						
							|  |  |  |     instr->op.code = COMPARE_OP; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							| 
									
										
										
										
											2023-03-23 15:25:09 -07:00
										 |  |  |     STAT_INC(COMPARE_OP, success); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2021-12-03 04:29:12 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | unpack_sequence_fail_kind(PyObject *seq) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (PySequence_Check(seq)) { | 
					
						
							|  |  |  |         return SPEC_FAIL_UNPACK_SEQUENCE_SEQUENCE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyIter_Check(seq)) { | 
					
						
							|  |  |  |         return SPEC_FAIL_UNPACK_SEQUENCE_ITERATOR; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_UnpackSequence(_PyStackRef seq_st, _Py_CODEUNIT *instr, int oparg) | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *seq = PyStackRef_AsPyObjectBorrow(seq_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-03-21 04:11:17 -07:00
										 |  |  |     assert(_PyOpcode_Caches[UNPACK_SEQUENCE] == | 
					
						
							| 
									
										
										
										
											2022-02-28 03:54:14 -08:00
										 |  |  |            INLINE_CACHE_ENTRIES_UNPACK_SEQUENCE); | 
					
						
							|  |  |  |     _PyUnpackSequenceCache *cache = (_PyUnpackSequenceCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |     if (PyTuple_CheckExact(seq)) { | 
					
						
							| 
									
										
										
										
											2022-02-28 03:54:14 -08:00
										 |  |  |         if (PyTuple_GET_SIZE(seq) != oparg) { | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |             SPECIALIZATION_FAIL(UNPACK_SEQUENCE, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (PyTuple_GET_SIZE(seq) == 2) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |             instr->op.code = UNPACK_SEQUENCE_TWO_TUPLE; | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |             goto success; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = UNPACK_SEQUENCE_TUPLE; | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyList_CheckExact(seq)) { | 
					
						
							| 
									
										
										
										
											2022-02-28 03:54:14 -08:00
										 |  |  |         if (PyList_GET_SIZE(seq) != oparg) { | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |             SPECIALIZATION_FAIL(UNPACK_SEQUENCE, SPEC_FAIL_EXPECTED_ERROR); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = UNPACK_SEQUENCE_LIST; | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SPECIALIZATION_FAIL(UNPACK_SEQUENCE, unpack_sequence_fail_kind(seq)); | 
					
						
							|  |  |  | failure: | 
					
						
							|  |  |  |     STAT_INC(UNPACK_SEQUENCE, failure); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = UNPACK_SEQUENCE; | 
					
						
							| 
									
										
										
										
											2022-05-31 11:58:26 +01:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							|  |  |  |     STAT_INC(UNPACK_SEQUENCE, success); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2022-02-16 08:48:16 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 11:47:52 +00:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  | int | 
					
						
							|  |  |  |  _PySpecialization_ClassifyIterator(PyObject *iter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (PyGen_CheckExact(iter)) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_GENERATOR; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyCoro_CheckExact(iter)) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_COROUTINE; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyAsyncGen_CheckExact(iter)) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_ASYNC_GENERATOR; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyAsyncGenASend_CheckExact(iter)) { | 
					
						
							|  |  |  |         return SPEC_FAIL_ITER_ASYNC_GENERATOR_SEND; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-11 17:01:10 +01:00
										 |  |  |     PyTypeObject *t = Py_TYPE(iter); | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     if (t == &PyListIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_LIST; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyTupleIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_TUPLE; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyDictIterKey_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_DICT_KEYS; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyDictIterValue_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_DICT_VALUES; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyDictIterItem_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_DICT_ITEMS; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PySetIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_SET; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyUnicodeIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_STRING; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyBytesIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_BYTES; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyRangeIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_RANGE; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyEnum_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_ENUMERATE; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     if (t == &PyMap_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_MAP; | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyZip_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_ZIP; | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PySeqIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_SEQ_ITER; | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &PyListRevIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_REVERSED_LIST; | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (t == &_PyUnicodeASCIIIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_ASCII_STRING; | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     const char *name = t->tp_name; | 
					
						
							|  |  |  |     if (strncmp(name, "itertools", 9) == 0) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_ITERTOOLS; | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     if (strncmp(name, "callable_iterator", 17) == 0) { | 
					
						
							| 
									
										
										
										
											2023-05-11 06:40:59 +08:00
										 |  |  |         return SPEC_FAIL_ITER_CALLABLE; | 
					
						
							| 
									
										
										
										
											2022-06-13 01:40:54 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-02-02 15:56:47 +00:00
										 |  |  |     return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_ForIter(_PyStackRef iter, _Py_CODEUNIT *instr, int oparg) | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2023-01-19 18:14:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  |     assert(_PyOpcode_Caches[FOR_ITER] == INLINE_CACHE_ENTRIES_FOR_ITER); | 
					
						
							|  |  |  |     _PyForIterCache *cache = (_PyForIterCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2024-06-27 19:35:25 +08:00
										 |  |  |     PyObject *iter_o = PyStackRef_AsPyObjectBorrow(iter); | 
					
						
							|  |  |  |     PyTypeObject *tp = Py_TYPE(iter_o); | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  |     if (tp == &PyListIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = FOR_ITER_LIST; | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-12-09 18:27:01 +08:00
										 |  |  |     else if (tp == &PyTupleIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = FOR_ITER_TUPLE; | 
					
						
							| 
									
										
										
										
											2022-12-09 18:27:01 +08:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-22 11:11:57 +00:00
										 |  |  |     else if (tp == &PyRangeIter_Type) { | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = FOR_ITER_RANGE; | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-07 06:49:51 -08:00
										 |  |  |     else if (tp == &PyGen_Type && oparg <= SHRT_MAX) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         assert(instr[oparg + INLINE_CACHE_ENTRIES_FOR_ITER + 1].op.code == END_FOR  || | 
					
						
							|  |  |  |             instr[oparg + INLINE_CACHE_ENTRIES_FOR_ITER + 1].op.code == INSTRUMENTED_END_FOR | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |         if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(FOR_ITER, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = FOR_ITER_GEN; | 
					
						
							| 
									
										
										
										
											2022-11-07 06:49:51 -08:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-10-27 03:55:03 -07:00
										 |  |  |     SPECIALIZATION_FAIL(FOR_ITER, | 
					
						
							| 
									
										
										
										
											2024-06-27 19:35:25 +08:00
										 |  |  |                         _PySpecialization_ClassifyIterator(iter_o)); | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  | failure: | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  |     STAT_INC(FOR_ITER, failure); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = FOR_ITER; | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							|  |  |  |     STAT_INC(FOR_ITER, success); | 
					
						
							| 
									
										
										
										
											2022-11-09 10:50:09 -08:00
										 |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							| 
									
										
										
										
											2022-06-21 06:19:26 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_Send(_PyStackRef receiver_st, _Py_CODEUNIT *instr) | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *receiver = PyStackRef_AsPyObjectBorrow(receiver_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							|  |  |  |     assert(_PyOpcode_Caches[SEND] == INLINE_CACHE_ENTRIES_SEND); | 
					
						
							|  |  |  |     _PySendCache *cache = (_PySendCache *)(instr + 1); | 
					
						
							|  |  |  |     PyTypeObject *tp = Py_TYPE(receiver); | 
					
						
							|  |  |  |     if (tp == &PyGen_Type || tp == &PyCoro_Type) { | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  |         if (_PyInterpreterState_GET()->eval_frame) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(SEND, SPEC_FAIL_OTHER); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |         instr->op.code = SEND_GEN; | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SPECIALIZATION_FAIL(SEND, | 
					
						
							|  |  |  |                         _PySpecialization_ClassifyIterator(receiver)); | 
					
						
							| 
									
										
										
										
											2023-05-12 15:23:13 -07:00
										 |  |  | failure: | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  |     STAT_INC(SEND, failure); | 
					
						
							| 
									
										
										
										
											2023-02-20 14:56:48 +00:00
										 |  |  |     instr->op.code = SEND; | 
					
						
							| 
									
										
										
										
											2023-02-13 11:24:55 +00:00
										 |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							|  |  |  |     STAT_INC(SEND, success); | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_ToBool(_PyStackRef value_o, _Py_CODEUNIT *instr) | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							|  |  |  |     assert(_PyOpcode_Caches[TO_BOOL] == INLINE_CACHE_ENTRIES_TO_BOOL); | 
					
						
							|  |  |  |     _PyToBoolCache *cache = (_PyToBoolCache *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *value = PyStackRef_AsPyObjectBorrow(value_o); | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  |     if (PyBool_Check(value)) { | 
					
						
							|  |  |  |         instr->op.code = TO_BOOL_BOOL; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyLong_CheckExact(value)) { | 
					
						
							|  |  |  |         instr->op.code = TO_BOOL_INT; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyList_CheckExact(value)) { | 
					
						
							|  |  |  |         instr->op.code = TO_BOOL_LIST; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (Py_IsNone(value)) { | 
					
						
							|  |  |  |         instr->op.code = TO_BOOL_NONE; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyUnicode_CheckExact(value)) { | 
					
						
							|  |  |  |         instr->op.code = TO_BOOL_STR; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyType_HasFeature(Py_TYPE(value), Py_TPFLAGS_HEAPTYPE)) { | 
					
						
							|  |  |  |         PyNumberMethods *nb = Py_TYPE(value)->tp_as_number; | 
					
						
							|  |  |  |         if (nb && nb->nb_bool) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_NUMBER); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyMappingMethods *mp = Py_TYPE(value)->tp_as_mapping; | 
					
						
							|  |  |  |         if (mp && mp->mp_length) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_MAPPING); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PySequenceMethods *sq = Py_TYPE(value)->tp_as_sequence; | 
					
						
							|  |  |  |         if (sq && sq->sq_length) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_SEQUENCE); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!PyUnstable_Type_AssignVersionTag(Py_TYPE(value))) { | 
					
						
							|  |  |  |             SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_OUT_OF_VERSIONS); | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-01-10 21:33:05 -08:00
										 |  |  |         uint32_t version = type_get_version(Py_TYPE(value), TO_BOOL); | 
					
						
							|  |  |  |         if (version == 0) { | 
					
						
							|  |  |  |             goto failure; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  |         instr->op.code = TO_BOOL_ALWAYS_TRUE; | 
					
						
							|  |  |  |         write_u32(cache->version, version); | 
					
						
							|  |  |  |         assert(version); | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  |     if (PyByteArray_CheckExact(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_BYTEARRAY); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyBytes_CheckExact(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_BYTES); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyDict_CheckExact(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_DICT); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyFloat_CheckExact(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_FLOAT); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyMemoryView_Check(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_MEMORY_VIEW); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyAnySet_CheckExact(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_SET); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyTuple_CheckExact(value)) { | 
					
						
							|  |  |  |         SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_TUPLE); | 
					
						
							|  |  |  |         goto failure; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_OTHER); | 
					
						
							| 
									
										
										
										
											2023-09-06 17:54:59 +02:00
										 |  |  | #endif   // Py_STATS
 | 
					
						
							| 
									
										
										
										
											2023-06-29 13:49:54 -07:00
										 |  |  | failure: | 
					
						
							|  |  |  |     STAT_INC(TO_BOOL, failure); | 
					
						
							|  |  |  |     instr->op.code = TO_BOOL; | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							|  |  |  |     STAT_INC(TO_BOOL, success); | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  | #ifdef Py_STATS
 | 
					
						
							|  |  |  | static int containsop_fail_kind(PyObject *value) { | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     if (PyUnicode_CheckExact(value)) { | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |         return SPEC_FAIL_CONTAINS_OP_STR; | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyList_CheckExact(value)) { | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |         return SPEC_FAIL_CONTAINS_OP_LIST; | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (PyTuple_CheckExact(value)) { | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |         return SPEC_FAIL_CONTAINS_OP_TUPLE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyType_Check(value)) { | 
					
						
							|  |  |  |         return SPEC_FAIL_CONTAINS_OP_USER_CLASS; | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |     return SPEC_FAIL_OTHER; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif   // Py_STATS
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  | _Py_Specialize_ContainsOp(_PyStackRef value_st, _Py_CODEUNIT *instr) | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-27 03:10:43 +08:00
										 |  |  |     PyObject *value = PyStackRef_AsPyObjectBorrow(value_st); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |     assert(ENABLE_SPECIALIZATION); | 
					
						
							|  |  |  |     assert(_PyOpcode_Caches[CONTAINS_OP] == INLINE_CACHE_ENTRIES_COMPARE_OP); | 
					
						
							|  |  |  |     _PyContainsOpCache *cache = (_PyContainsOpCache  *)(instr + 1); | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     if (PyDict_CheckExact(value)) { | 
					
						
							|  |  |  |         instr->op.code = CONTAINS_OP_DICT; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |     if (PySet_CheckExact(value) || PyFrozenSet_CheckExact(value)) { | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |         instr->op.code = CONTAINS_OP_SET; | 
					
						
							|  |  |  |         goto success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-08 00:21:21 +08:00
										 |  |  |     SPECIALIZATION_FAIL(CONTAINS_OP, containsop_fail_kind(value)); | 
					
						
							| 
									
										
										
										
											2024-03-07 03:30:11 +08:00
										 |  |  |     STAT_INC(CONTAINS_OP, failure); | 
					
						
							|  |  |  |     instr->op.code = CONTAINS_OP; | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_backoff(cache->counter); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | success: | 
					
						
							|  |  |  |     STAT_INC(CONTAINS_OP, success); | 
					
						
							|  |  |  |     cache->counter = adaptive_counter_cooldown(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  | /* Code init cleanup.
 | 
					
						
							| 
									
										
										
										
											2023-09-13 10:25:45 -07:00
										 |  |  |  * CALL_ALLOC_AND_ENTER_INIT will set up | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |  * the frame to execute the EXIT_INIT_CHECK | 
					
						
							|  |  |  |  * instruction. | 
					
						
							|  |  |  |  * Ends with a RESUME so that it is not traced. | 
					
						
							|  |  |  |  * This is used as a plain code object, not a function, | 
					
						
							|  |  |  |  * so must not access globals or builtins. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define NO_LOC_4 (128 | (PY_CODE_LOCATION_INFO_NONE << 3) | 3)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const PyBytesObject no_location = { | 
					
						
							|  |  |  |     PyVarObject_HEAD_INIT(&PyBytes_Type, 1) | 
					
						
							|  |  |  |     .ob_sval = { NO_LOC_4 } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-20 16:52:58 +01:00
										 |  |  | const struct _PyCode8 _Py_InitCleanup = { | 
					
						
							| 
									
										
										
										
											2023-10-30 12:06:09 -04:00
										 |  |  |     _PyVarObject_HEAD_INIT(&PyCode_Type, 3), | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |     .co_consts = (PyObject *)&_Py_SINGLETON(tuple_empty), | 
					
						
							|  |  |  |     .co_names = (PyObject *)&_Py_SINGLETON(tuple_empty), | 
					
						
							|  |  |  |     .co_exceptiontable = (PyObject *)&_Py_SINGLETON(bytes_empty), | 
					
						
							| 
									
										
										
										
											2024-01-05 09:45:22 +00:00
										 |  |  |     .co_flags = CO_OPTIMIZED | CO_NO_MONITORING_EVENTS, | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |     .co_localsplusnames = (PyObject *)&_Py_SINGLETON(tuple_empty), | 
					
						
							|  |  |  |     .co_localspluskinds = (PyObject *)&_Py_SINGLETON(bytes_empty), | 
					
						
							|  |  |  |     .co_filename = &_Py_ID(__init__), | 
					
						
							|  |  |  |     .co_name = &_Py_ID(__init__), | 
					
						
							|  |  |  |     .co_qualname = &_Py_ID(__init__), | 
					
						
							|  |  |  |     .co_linetable = (PyObject *)&no_location, | 
					
						
							|  |  |  |     ._co_firsttraceable = 4, | 
					
						
							|  |  |  |     .co_stacksize = 2, | 
					
						
							|  |  |  |     .co_framesize = 2 + FRAME_SPECIALS_SIZE, | 
					
						
							|  |  |  |     .co_code_adaptive = { | 
					
						
							|  |  |  |         EXIT_INIT_CHECK, 0, | 
					
						
							|  |  |  |         RETURN_VALUE, 0, | 
					
						
							| 
									
										
										
										
											2023-10-26 16:30:18 +01:00
										 |  |  |         RESUME, RESUME_AT_FUNC_START, | 
					
						
							| 
									
										
										
										
											2023-06-22 09:48:19 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | }; |