mirror of
https://github.com/python/cpython.git
synced 2025-10-13 02:13:03 +00:00
bpo-44525: Specialize for calls to type and other builtin classes with 1 argument. (GH-29942)
This commit is contained in:
parent
f025ae63dc
commit
3a60bfef49
6 changed files with 105 additions and 40 deletions
50
Include/opcode.h
generated
50
Include/opcode.h
generated
|
@ -142,30 +142,32 @@ extern "C" {
|
||||||
#define CALL_NO_KW_PY_SIMPLE 45
|
#define CALL_NO_KW_PY_SIMPLE 45
|
||||||
#define CALL_NO_KW_LIST_APPEND 46
|
#define CALL_NO_KW_LIST_APPEND 46
|
||||||
#define CALL_NO_KW_METHOD_DESCRIPTOR_O 47
|
#define CALL_NO_KW_METHOD_DESCRIPTOR_O 47
|
||||||
#define CALL_NO_KW_METHOD_DESCRIPTOR_FAST 48
|
#define CALL_NO_KW_TYPE_1 48
|
||||||
#define JUMP_ABSOLUTE_QUICK 55
|
#define CALL_NO_KW_BUILTIN_CLASS_1 55
|
||||||
#define LOAD_ATTR_ADAPTIVE 56
|
#define CALL_NO_KW_METHOD_DESCRIPTOR_FAST 56
|
||||||
#define LOAD_ATTR_INSTANCE_VALUE 57
|
#define JUMP_ABSOLUTE_QUICK 57
|
||||||
#define LOAD_ATTR_WITH_HINT 58
|
#define LOAD_ATTR_ADAPTIVE 58
|
||||||
#define LOAD_ATTR_SLOT 59
|
#define LOAD_ATTR_INSTANCE_VALUE 59
|
||||||
#define LOAD_ATTR_MODULE 62
|
#define LOAD_ATTR_WITH_HINT 62
|
||||||
#define LOAD_GLOBAL_ADAPTIVE 63
|
#define LOAD_ATTR_SLOT 63
|
||||||
#define LOAD_GLOBAL_MODULE 64
|
#define LOAD_ATTR_MODULE 64
|
||||||
#define LOAD_GLOBAL_BUILTIN 65
|
#define LOAD_GLOBAL_ADAPTIVE 65
|
||||||
#define LOAD_METHOD_ADAPTIVE 66
|
#define LOAD_GLOBAL_MODULE 66
|
||||||
#define LOAD_METHOD_CACHED 67
|
#define LOAD_GLOBAL_BUILTIN 67
|
||||||
#define LOAD_METHOD_CLASS 72
|
#define LOAD_METHOD_ADAPTIVE 72
|
||||||
#define LOAD_METHOD_MODULE 75
|
#define LOAD_METHOD_CACHED 75
|
||||||
#define LOAD_METHOD_NO_DICT 76
|
#define LOAD_METHOD_CLASS 76
|
||||||
#define STORE_ATTR_ADAPTIVE 77
|
#define LOAD_METHOD_MODULE 77
|
||||||
#define STORE_ATTR_INSTANCE_VALUE 78
|
#define LOAD_METHOD_NO_DICT 78
|
||||||
#define STORE_ATTR_SLOT 79
|
#define STORE_ATTR_ADAPTIVE 79
|
||||||
#define STORE_ATTR_WITH_HINT 80
|
#define STORE_ATTR_INSTANCE_VALUE 80
|
||||||
#define LOAD_FAST__LOAD_FAST 81
|
#define STORE_ATTR_SLOT 81
|
||||||
#define STORE_FAST__LOAD_FAST 87
|
#define STORE_ATTR_WITH_HINT 87
|
||||||
#define LOAD_FAST__LOAD_CONST 128
|
#define LOAD_FAST__LOAD_FAST 128
|
||||||
#define LOAD_CONST__LOAD_FAST 131
|
#define STORE_FAST__LOAD_FAST 131
|
||||||
#define STORE_FAST__STORE_FAST 134
|
#define LOAD_FAST__LOAD_CONST 134
|
||||||
|
#define LOAD_CONST__LOAD_FAST 140
|
||||||
|
#define STORE_FAST__STORE_FAST 141
|
||||||
#define DO_TRACING 255
|
#define DO_TRACING 255
|
||||||
#ifdef NEED_OPCODE_JUMP_TABLES
|
#ifdef NEED_OPCODE_JUMP_TABLES
|
||||||
static uint32_t _PyOpcode_RelativeJump[8] = {
|
static uint32_t _PyOpcode_RelativeJump[8] = {
|
||||||
|
|
|
@ -259,6 +259,8 @@ _specialized_instructions = [
|
||||||
"CALL_NO_KW_PY_SIMPLE",
|
"CALL_NO_KW_PY_SIMPLE",
|
||||||
"CALL_NO_KW_LIST_APPEND",
|
"CALL_NO_KW_LIST_APPEND",
|
||||||
"CALL_NO_KW_METHOD_DESCRIPTOR_O",
|
"CALL_NO_KW_METHOD_DESCRIPTOR_O",
|
||||||
|
"CALL_NO_KW_TYPE_1",
|
||||||
|
"CALL_NO_KW_BUILTIN_CLASS_1",
|
||||||
"CALL_NO_KW_METHOD_DESCRIPTOR_FAST",
|
"CALL_NO_KW_METHOD_DESCRIPTOR_FAST",
|
||||||
"JUMP_ABSOLUTE_QUICK",
|
"JUMP_ABSOLUTE_QUICK",
|
||||||
"LOAD_ATTR_ADAPTIVE",
|
"LOAD_ATTR_ADAPTIVE",
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
Specialize the CALL_FUNCTION instruction for calls to builtin types with a
|
||||||
|
single argument. Speeds up ``range(x)``, ``list(x)``, and specifically
|
||||||
|
``type(obj)``.
|
|
@ -4854,6 +4854,41 @@ check_eval_breaker:
|
||||||
goto start_frame;
|
goto start_frame;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TARGET(CALL_NO_KW_TYPE_1) {
|
||||||
|
assert(STACK_ADJUST_IS_RESET);
|
||||||
|
assert(GET_CACHE()->adaptive.original_oparg == 1);
|
||||||
|
PyObject *obj = TOP();
|
||||||
|
PyObject *callable = SECOND();
|
||||||
|
DEOPT_IF(callable != (PyObject *)&PyType_Type, CALL_NO_KW);
|
||||||
|
PyObject *res = Py_NewRef(Py_TYPE(obj));
|
||||||
|
STACK_SHRINK(1);
|
||||||
|
Py_DECREF(callable);
|
||||||
|
Py_DECREF(obj);
|
||||||
|
SET_TOP(res);
|
||||||
|
DISPATCH();
|
||||||
|
}
|
||||||
|
|
||||||
|
TARGET(CALL_NO_KW_BUILTIN_CLASS_1) {
|
||||||
|
assert(STACK_ADJUST_IS_RESET);
|
||||||
|
SpecializedCacheEntry *caches = GET_CACHE();
|
||||||
|
_PyAdaptiveEntry *cache0 = &caches[0].adaptive;
|
||||||
|
assert(cache0->original_oparg == 1);
|
||||||
|
PyObject *callable = SECOND();
|
||||||
|
PyObject *arg = TOP();
|
||||||
|
DEOPT_IF(!PyType_Check(callable), CALL_NO_KW);
|
||||||
|
PyTypeObject *tp = (PyTypeObject *)callable;
|
||||||
|
DEOPT_IF(tp->tp_version_tag != cache0->version, CALL_NO_KW);
|
||||||
|
STACK_SHRINK(1);
|
||||||
|
PyObject *res = tp->tp_vectorcall((PyObject *)tp, stack_pointer, 1, NULL);
|
||||||
|
SET_TOP(res);
|
||||||
|
Py_DECREF(tp);
|
||||||
|
Py_DECREF(arg);
|
||||||
|
if (res == NULL) {
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
DISPATCH();
|
||||||
|
}
|
||||||
|
|
||||||
TARGET(CALL_NO_KW_BUILTIN_O) {
|
TARGET(CALL_NO_KW_BUILTIN_O) {
|
||||||
assert(cframe.use_tracing == 0);
|
assert(cframe.use_tracing == 0);
|
||||||
assert(STACK_ADJUST_IS_RESET);
|
assert(STACK_ADJUST_IS_RESET);
|
||||||
|
|
28
Python/opcode_targets.h
generated
28
Python/opcode_targets.h
generated
|
@ -47,46 +47,46 @@ static void *opcode_targets[256] = {
|
||||||
&&TARGET_CALL_NO_KW_PY_SIMPLE,
|
&&TARGET_CALL_NO_KW_PY_SIMPLE,
|
||||||
&&TARGET_CALL_NO_KW_LIST_APPEND,
|
&&TARGET_CALL_NO_KW_LIST_APPEND,
|
||||||
&&TARGET_CALL_NO_KW_METHOD_DESCRIPTOR_O,
|
&&TARGET_CALL_NO_KW_METHOD_DESCRIPTOR_O,
|
||||||
&&TARGET_CALL_NO_KW_METHOD_DESCRIPTOR_FAST,
|
&&TARGET_CALL_NO_KW_TYPE_1,
|
||||||
&&TARGET_WITH_EXCEPT_START,
|
&&TARGET_WITH_EXCEPT_START,
|
||||||
&&TARGET_GET_AITER,
|
&&TARGET_GET_AITER,
|
||||||
&&TARGET_GET_ANEXT,
|
&&TARGET_GET_ANEXT,
|
||||||
&&TARGET_BEFORE_ASYNC_WITH,
|
&&TARGET_BEFORE_ASYNC_WITH,
|
||||||
&&TARGET_BEFORE_WITH,
|
&&TARGET_BEFORE_WITH,
|
||||||
&&TARGET_END_ASYNC_FOR,
|
&&TARGET_END_ASYNC_FOR,
|
||||||
|
&&TARGET_CALL_NO_KW_BUILTIN_CLASS_1,
|
||||||
|
&&TARGET_CALL_NO_KW_METHOD_DESCRIPTOR_FAST,
|
||||||
&&TARGET_JUMP_ABSOLUTE_QUICK,
|
&&TARGET_JUMP_ABSOLUTE_QUICK,
|
||||||
&&TARGET_LOAD_ATTR_ADAPTIVE,
|
&&TARGET_LOAD_ATTR_ADAPTIVE,
|
||||||
&&TARGET_LOAD_ATTR_INSTANCE_VALUE,
|
&&TARGET_LOAD_ATTR_INSTANCE_VALUE,
|
||||||
&&TARGET_LOAD_ATTR_WITH_HINT,
|
|
||||||
&&TARGET_LOAD_ATTR_SLOT,
|
|
||||||
&&TARGET_STORE_SUBSCR,
|
&&TARGET_STORE_SUBSCR,
|
||||||
&&TARGET_DELETE_SUBSCR,
|
&&TARGET_DELETE_SUBSCR,
|
||||||
|
&&TARGET_LOAD_ATTR_WITH_HINT,
|
||||||
|
&&TARGET_LOAD_ATTR_SLOT,
|
||||||
&&TARGET_LOAD_ATTR_MODULE,
|
&&TARGET_LOAD_ATTR_MODULE,
|
||||||
&&TARGET_LOAD_GLOBAL_ADAPTIVE,
|
&&TARGET_LOAD_GLOBAL_ADAPTIVE,
|
||||||
&&TARGET_LOAD_GLOBAL_MODULE,
|
&&TARGET_LOAD_GLOBAL_MODULE,
|
||||||
&&TARGET_LOAD_GLOBAL_BUILTIN,
|
&&TARGET_LOAD_GLOBAL_BUILTIN,
|
||||||
&&TARGET_LOAD_METHOD_ADAPTIVE,
|
|
||||||
&&TARGET_LOAD_METHOD_CACHED,
|
|
||||||
&&TARGET_GET_ITER,
|
&&TARGET_GET_ITER,
|
||||||
&&TARGET_GET_YIELD_FROM_ITER,
|
&&TARGET_GET_YIELD_FROM_ITER,
|
||||||
&&TARGET_PRINT_EXPR,
|
&&TARGET_PRINT_EXPR,
|
||||||
&&TARGET_LOAD_BUILD_CLASS,
|
&&TARGET_LOAD_BUILD_CLASS,
|
||||||
&&TARGET_LOAD_METHOD_CLASS,
|
&&TARGET_LOAD_METHOD_ADAPTIVE,
|
||||||
&&TARGET_GET_AWAITABLE,
|
&&TARGET_GET_AWAITABLE,
|
||||||
&&TARGET_LOAD_ASSERTION_ERROR,
|
&&TARGET_LOAD_ASSERTION_ERROR,
|
||||||
|
&&TARGET_LOAD_METHOD_CACHED,
|
||||||
|
&&TARGET_LOAD_METHOD_CLASS,
|
||||||
&&TARGET_LOAD_METHOD_MODULE,
|
&&TARGET_LOAD_METHOD_MODULE,
|
||||||
&&TARGET_LOAD_METHOD_NO_DICT,
|
&&TARGET_LOAD_METHOD_NO_DICT,
|
||||||
&&TARGET_STORE_ATTR_ADAPTIVE,
|
&&TARGET_STORE_ATTR_ADAPTIVE,
|
||||||
&&TARGET_STORE_ATTR_INSTANCE_VALUE,
|
&&TARGET_STORE_ATTR_INSTANCE_VALUE,
|
||||||
&&TARGET_STORE_ATTR_SLOT,
|
&&TARGET_STORE_ATTR_SLOT,
|
||||||
&&TARGET_STORE_ATTR_WITH_HINT,
|
|
||||||
&&TARGET_LOAD_FAST__LOAD_FAST,
|
|
||||||
&&TARGET_LIST_TO_TUPLE,
|
&&TARGET_LIST_TO_TUPLE,
|
||||||
&&TARGET_RETURN_VALUE,
|
&&TARGET_RETURN_VALUE,
|
||||||
&&TARGET_IMPORT_STAR,
|
&&TARGET_IMPORT_STAR,
|
||||||
&&TARGET_SETUP_ANNOTATIONS,
|
&&TARGET_SETUP_ANNOTATIONS,
|
||||||
&&TARGET_YIELD_VALUE,
|
&&TARGET_YIELD_VALUE,
|
||||||
&&TARGET_STORE_FAST__LOAD_FAST,
|
&&TARGET_STORE_ATTR_WITH_HINT,
|
||||||
&&TARGET_PREP_RERAISE_STAR,
|
&&TARGET_PREP_RERAISE_STAR,
|
||||||
&&TARGET_POP_EXCEPT,
|
&&TARGET_POP_EXCEPT,
|
||||||
&&TARGET_STORE_NAME,
|
&&TARGET_STORE_NAME,
|
||||||
|
@ -127,20 +127,20 @@ static void *opcode_targets[256] = {
|
||||||
&&TARGET_STORE_FAST,
|
&&TARGET_STORE_FAST,
|
||||||
&&TARGET_DELETE_FAST,
|
&&TARGET_DELETE_FAST,
|
||||||
&&TARGET_JUMP_IF_NOT_EG_MATCH,
|
&&TARGET_JUMP_IF_NOT_EG_MATCH,
|
||||||
&&TARGET_LOAD_FAST__LOAD_CONST,
|
&&TARGET_LOAD_FAST__LOAD_FAST,
|
||||||
&&TARGET_GEN_START,
|
&&TARGET_GEN_START,
|
||||||
&&TARGET_RAISE_VARARGS,
|
&&TARGET_RAISE_VARARGS,
|
||||||
&&TARGET_LOAD_CONST__LOAD_FAST,
|
&&TARGET_STORE_FAST__LOAD_FAST,
|
||||||
&&TARGET_MAKE_FUNCTION,
|
&&TARGET_MAKE_FUNCTION,
|
||||||
&&TARGET_BUILD_SLICE,
|
&&TARGET_BUILD_SLICE,
|
||||||
&&TARGET_STORE_FAST__STORE_FAST,
|
&&TARGET_LOAD_FAST__LOAD_CONST,
|
||||||
&&TARGET_MAKE_CELL,
|
&&TARGET_MAKE_CELL,
|
||||||
&&TARGET_LOAD_CLOSURE,
|
&&TARGET_LOAD_CLOSURE,
|
||||||
&&TARGET_LOAD_DEREF,
|
&&TARGET_LOAD_DEREF,
|
||||||
&&TARGET_STORE_DEREF,
|
&&TARGET_STORE_DEREF,
|
||||||
&&TARGET_DELETE_DEREF,
|
&&TARGET_DELETE_DEREF,
|
||||||
&&_unknown_opcode,
|
&&TARGET_LOAD_CONST__LOAD_FAST,
|
||||||
&&_unknown_opcode,
|
&&TARGET_STORE_FAST__STORE_FAST,
|
||||||
&&TARGET_CALL_FUNCTION_EX,
|
&&TARGET_CALL_FUNCTION_EX,
|
||||||
&&_unknown_opcode,
|
&&_unknown_opcode,
|
||||||
&&TARGET_EXTENDED_ARG,
|
&&TARGET_EXTENDED_ARG,
|
||||||
|
|
|
@ -491,8 +491,10 @@ initial_counter_value(void) {
|
||||||
#define SPEC_FAIL_PYCFUNCTION_NOARGS 16
|
#define SPEC_FAIL_PYCFUNCTION_NOARGS 16
|
||||||
#define SPEC_FAIL_BAD_CALL_FLAGS 17
|
#define SPEC_FAIL_BAD_CALL_FLAGS 17
|
||||||
#define SPEC_FAIL_CLASS 18
|
#define SPEC_FAIL_CLASS 18
|
||||||
#define SPEC_FAIL_C_METHOD_CALL 19
|
#define SPEC_FAIL_PYTHON_CLASS 19
|
||||||
#define SPEC_FAIL_METHDESCR_NON_METHOD 20
|
#define SPEC_FAIL_C_METHOD_CALL 20
|
||||||
|
#define SPEC_FAIL_METHDESCR_NON_METHOD 21
|
||||||
|
#define SPEC_FAIL_METHOD_CALL_CLASS 22
|
||||||
|
|
||||||
/* COMPARE_OP */
|
/* COMPARE_OP */
|
||||||
#define SPEC_FAIL_STRING_COMPARE 13
|
#define SPEC_FAIL_STRING_COMPARE 13
|
||||||
|
@ -1263,6 +1265,27 @@ specialize_class_call(
|
||||||
PyObject *callable, _Py_CODEUNIT *instr,
|
PyObject *callable, _Py_CODEUNIT *instr,
|
||||||
int nargs, SpecializedCacheEntry *cache)
|
int nargs, SpecializedCacheEntry *cache)
|
||||||
{
|
{
|
||||||
|
assert(PyType_Check(callable));
|
||||||
|
PyTypeObject *tp = (PyTypeObject *)callable;
|
||||||
|
if (_Py_OPCODE(instr[-1]) == PRECALL_METHOD) {
|
||||||
|
SPECIALIZATION_FAIL(CALL_NO_KW, SPEC_FAIL_METHOD_CALL_CLASS);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (tp->tp_new == PyBaseObject_Type.tp_new) {
|
||||||
|
SPECIALIZATION_FAIL(CALL_NO_KW, SPEC_FAIL_PYTHON_CLASS);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (nargs == 1) {
|
||||||
|
if (tp == &PyType_Type) {
|
||||||
|
*instr = _Py_MAKECODEUNIT(CALL_NO_KW_TYPE_1, _Py_OPARG(*instr));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if ((tp->tp_flags & Py_TPFLAGS_IMMUTABLETYPE) && tp->tp_vectorcall != NULL) {
|
||||||
|
cache->adaptive.version = tp->tp_version_tag;
|
||||||
|
*instr = _Py_MAKECODEUNIT(CALL_NO_KW_BUILTIN_CLASS_1, _Py_OPARG(*instr));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
SPECIALIZATION_FAIL(CALL_NO_KW, SPEC_FAIL_CLASS);
|
SPECIALIZATION_FAIL(CALL_NO_KW, SPEC_FAIL_CLASS);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue