mirror of
https://github.com/python/cpython.git
synced 2026-01-21 22:59:06 +00:00
gh-142913: Add test case for interpreter generator w/ overridden opcodes (#142911)
Add test case for interpreter generator w/ overridden opcodes
This commit is contained in:
parent
3e93225798
commit
4d5a676aa0
22 changed files with 14963 additions and 645 deletions
|
|
@ -563,7 +563,7 @@ _PyCode_GetTLBCFast(PyThreadState *tstate, PyCodeObject *co)
|
|||
|
||||
// Return a pointer to the thread-local bytecode for the current thread,
|
||||
// creating it if necessary.
|
||||
extern _Py_CODEUNIT *_PyCode_GetTLBC(PyCodeObject *co);
|
||||
PyAPI_FUNC(_Py_CODEUNIT *) _PyCode_GetTLBC(PyCodeObject *co);
|
||||
|
||||
// Reserve an index for the current thread into thread-local bytecode
|
||||
// arrays
|
||||
|
|
|
|||
|
|
@ -67,8 +67,8 @@ _Py_call_instrumentation_exc2(PyThreadState *tstate, int event,
|
|||
extern int
|
||||
_Py_Instrumentation_GetLine(PyCodeObject *code, int index);
|
||||
|
||||
extern PyObject _PyInstrumentation_MISSING;
|
||||
extern PyObject _PyInstrumentation_DISABLE;
|
||||
PyAPI_DATA(PyObject) _PyInstrumentation_MISSING;
|
||||
PyAPI_DATA(PyObject) _PyInstrumentation_DISABLE;
|
||||
|
||||
|
||||
/* Total tool ids available */
|
||||
|
|
|
|||
|
|
@ -279,7 +279,7 @@ _PyThreadState_GetFrame(PyThreadState *tstate)
|
|||
|
||||
/* For use by _PyFrame_GetFrameObject
|
||||
Do not call directly. */
|
||||
PyFrameObject *
|
||||
PyAPI_FUNC(PyFrameObject *)
|
||||
_PyFrame_MakeAndSetFrameObject(_PyInterpreterFrame *frame);
|
||||
|
||||
/* Gets the PyFrameObject for this frame, lazily
|
||||
|
|
|
|||
|
|
@ -17,12 +17,12 @@ PyAPI_FUNC(PyObject) *_PyList_SliceSubscript(PyObject*, PyObject*);
|
|||
extern void _PyList_DebugMallocStats(FILE *out);
|
||||
// _PyList_GetItemRef should be used only when the object is known as a list
|
||||
// because it doesn't raise TypeError when the object is not a list, whereas PyList_GetItemRef does.
|
||||
extern PyObject* _PyList_GetItemRef(PyListObject *, Py_ssize_t i);
|
||||
PyAPI_FUNC(PyObject *) _PyList_GetItemRef(PyListObject *, Py_ssize_t i);
|
||||
|
||||
|
||||
#ifdef Py_GIL_DISABLED
|
||||
// Returns -1 in case of races with other threads.
|
||||
extern int _PyList_GetItemRefNoLock(PyListObject *, Py_ssize_t, _PyStackRef *);
|
||||
PyAPI_FUNC(int) _PyList_GetItemRefNoLock(PyListObject *, Py_ssize_t, _PyStackRef *);
|
||||
#endif
|
||||
|
||||
#define _PyList_ITEMS(op) _Py_RVALUE(_PyList_CAST(op)->ob_item)
|
||||
|
|
|
|||
8
Include/internal/pycore_opcode_metadata.h
generated
8
Include/internal/pycore_opcode_metadata.h
generated
|
|
@ -1087,7 +1087,7 @@ struct opcode_metadata {
|
|||
uint32_t flags;
|
||||
};
|
||||
|
||||
extern const struct opcode_metadata _PyOpcode_opcode_metadata[267];
|
||||
PyAPI_DATA(const struct opcode_metadata) _PyOpcode_opcode_metadata[267];
|
||||
#ifdef NEED_OPCODE_METADATA
|
||||
const struct opcode_metadata _PyOpcode_opcode_metadata[267] = {
|
||||
[BINARY_OP] = { true, INSTR_FMT_IBC0000, HAS_ARG_FLAG | HAS_ERROR_FLAG | HAS_ERROR_NO_POP_FLAG | HAS_ESCAPES_FLAG },
|
||||
|
|
@ -1534,7 +1534,7 @@ _PyOpcode_macro_expansion[256] = {
|
|||
};
|
||||
#endif // NEED_OPCODE_METADATA
|
||||
|
||||
extern const char *_PyOpcode_OpName[267];
|
||||
PyAPI_DATA(const char) *_PyOpcode_OpName[267];
|
||||
#ifdef NEED_OPCODE_METADATA
|
||||
const char *_PyOpcode_OpName[267] = {
|
||||
[ANNOTATIONS_PLACEHOLDER] = "ANNOTATIONS_PLACEHOLDER",
|
||||
|
|
@ -1780,7 +1780,7 @@ const char *_PyOpcode_OpName[267] = {
|
|||
};
|
||||
#endif
|
||||
|
||||
extern const uint8_t _PyOpcode_Caches[256];
|
||||
PyAPI_DATA(const uint8_t) _PyOpcode_Caches[256];
|
||||
#ifdef NEED_OPCODE_METADATA
|
||||
const uint8_t _PyOpcode_Caches[256] = {
|
||||
[TO_BOOL] = 3,
|
||||
|
|
@ -1806,7 +1806,7 @@ const uint8_t _PyOpcode_Caches[256] = {
|
|||
};
|
||||
#endif
|
||||
|
||||
extern const uint8_t _PyOpcode_Deopt[256];
|
||||
PyAPI_DATA(const uint8_t) _PyOpcode_Deopt[256];
|
||||
#ifdef NEED_OPCODE_METADATA
|
||||
const uint8_t _PyOpcode_Deopt[256] = {
|
||||
[121] = 121,
|
||||
|
|
|
|||
|
|
@ -11,8 +11,8 @@ extern "C" {
|
|||
|
||||
/* Error handling definitions */
|
||||
|
||||
extern _PyErr_StackItem* _PyErr_GetTopmostException(PyThreadState *tstate);
|
||||
extern PyObject* _PyErr_GetHandledException(PyThreadState *);
|
||||
PyAPI_FUNC(_PyErr_StackItem*) _PyErr_GetTopmostException(PyThreadState *tstate);
|
||||
PyAPI_FUNC(PyObject*) _PyErr_GetHandledException(PyThreadState *);
|
||||
extern void _PyErr_SetHandledException(PyThreadState *, PyObject *);
|
||||
extern void _PyErr_GetExcInfo(PyThreadState *, PyObject **, PyObject **, PyObject **);
|
||||
|
||||
|
|
@ -108,7 +108,7 @@ extern void _PyErr_Restore(
|
|||
PyObject *value,
|
||||
PyObject *traceback);
|
||||
|
||||
extern void _PyErr_SetObject(
|
||||
PyAPI_FUNC(void) _PyErr_SetObject(
|
||||
PyThreadState *tstate,
|
||||
PyObject *type,
|
||||
PyObject *value);
|
||||
|
|
|
|||
|
|
@ -2857,6 +2857,24 @@ def func():
|
|||
names = ["func", "outer", "outer", "inner", "inner", "outer", "inner"]
|
||||
self.do_test(func, names)
|
||||
|
||||
def test_replaced_interpreter(self):
|
||||
def inner():
|
||||
yield 'abc'
|
||||
def outer():
|
||||
yield from inner()
|
||||
def func():
|
||||
list(outer())
|
||||
_testinternalcapi.set_eval_frame_interp()
|
||||
try:
|
||||
func()
|
||||
finally:
|
||||
_testinternalcapi.set_eval_frame_default()
|
||||
|
||||
stats = _testinternalcapi.get_eval_frame_stats()
|
||||
|
||||
self.assertEqual(stats["resumes"], 5)
|
||||
self.assertEqual(stats["loads"], 5)
|
||||
|
||||
|
||||
@unittest.skipUnless(support.Py_GIL_DISABLED, 'need Py_GIL_DISABLED')
|
||||
class TestPyThreadId(unittest.TestCase):
|
||||
|
|
|
|||
|
|
@ -2150,7 +2150,7 @@ Objects/mimalloc/page.o: $(srcdir)/Objects/mimalloc/page-queue.c
|
|||
regen-cases: \
|
||||
regen-opcode-ids regen-opcode-targets regen-uop-ids regen-opcode-metadata-py \
|
||||
regen-generated-cases regen-executor-cases regen-optimizer-cases \
|
||||
regen-opcode-metadata regen-uop-metadata
|
||||
regen-opcode-metadata regen-uop-metadata regen-test-cases regen-test-opcode-targets
|
||||
|
||||
.PHONY: regen-opcode-ids
|
||||
regen-opcode-ids:
|
||||
|
|
@ -2182,6 +2182,20 @@ regen-generated-cases:
|
|||
-o $(srcdir)/Python/generated_cases.c.h.new $(srcdir)/Python/bytecodes.c
|
||||
$(UPDATE_FILE) $(srcdir)/Python/generated_cases.c.h $(srcdir)/Python/generated_cases.c.h.new
|
||||
|
||||
.PHONY: regen-test-cases
|
||||
regen-test-cases:
|
||||
$(PYTHON_FOR_REGEN) $(srcdir)/Tools/cases_generator/tier1_generator.py \
|
||||
-o $(srcdir)/Modules/_testinternalcapi/test_cases.c.h.new $(srcdir)/Python/bytecodes.c \
|
||||
$(srcdir)/Modules/_testinternalcapi/testbytecodes.c
|
||||
$(UPDATE_FILE) $(srcdir)/Modules/_testinternalcapi/test_cases.c.h $(srcdir)/Modules/_testinternalcapi/test_cases.c.h.new
|
||||
|
||||
.PHONY: regen-test-opcode-targets
|
||||
regen-test-opcode-targets:
|
||||
$(PYTHON_FOR_REGEN) $(srcdir)/Tools/cases_generator/target_generator.py \
|
||||
-o $(srcdir)/Modules/_testinternalcapi/test_targets.h.new $(srcdir)/Python/bytecodes.c \
|
||||
$(srcdir)/Modules/_testinternalcapi/testbytecodes.c
|
||||
$(UPDATE_FILE) $(srcdir)/Modules/_testinternalcapi/test_targets.h $(srcdir)/Modules/_testinternalcapi/test_targets.h.new
|
||||
|
||||
.PHONY: regen-executor-cases
|
||||
regen-executor-cases:
|
||||
$(PYTHON_FOR_REGEN) $(srcdir)/Tools/cases_generator/tier2_generator.py \
|
||||
|
|
@ -3428,7 +3442,7 @@ MODULE__SOCKET_DEPS=$(srcdir)/Modules/socketmodule.h $(srcdir)/Modules/addrinfo.
|
|||
MODULE__SSL_DEPS=$(srcdir)/Modules/_ssl.h $(srcdir)/Modules/_ssl/cert.c $(srcdir)/Modules/_ssl/debughelpers.c $(srcdir)/Modules/_ssl/misc.c $(srcdir)/Modules/_ssl_data_111.h $(srcdir)/Modules/_ssl_data_300.h $(srcdir)/Modules/socketmodule.h
|
||||
MODULE__TESTCAPI_DEPS=$(srcdir)/Modules/_testcapi/parts.h $(srcdir)/Modules/_testcapi/util.h
|
||||
MODULE__TESTLIMITEDCAPI_DEPS=$(srcdir)/Modules/_testlimitedcapi/testcapi_long.h $(srcdir)/Modules/_testlimitedcapi/parts.h $(srcdir)/Modules/_testlimitedcapi/util.h
|
||||
MODULE__TESTINTERNALCAPI_DEPS=$(srcdir)/Modules/_testinternalcapi/parts.h
|
||||
MODULE__TESTINTERNALCAPI_DEPS=$(srcdir)/Modules/_testinternalcapi/parts.h $(srcdir)/Python/ceval.h $(srcdir)/Modules/_testinternalcapi/test_targets.h $(srcdir)/Modules/_testinternalcapi/test_cases.c.h
|
||||
MODULE__SQLITE3_DEPS=$(srcdir)/Modules/_sqlite/connection.h $(srcdir)/Modules/_sqlite/cursor.h $(srcdir)/Modules/_sqlite/microprotocols.h $(srcdir)/Modules/_sqlite/module.h $(srcdir)/Modules/_sqlite/prepare_protocol.h $(srcdir)/Modules/_sqlite/row.h $(srcdir)/Modules/_sqlite/util.h
|
||||
MODULE__ZSTD_DEPS=$(srcdir)/Modules/_zstd/_zstdmodule.h $(srcdir)/Modules/_zstd/buffer.h $(srcdir)/Modules/_zstd/zstddict.h
|
||||
|
||||
|
|
|
|||
|
|
@ -174,7 +174,7 @@
|
|||
@MODULE_XXSUBTYPE_TRUE@xxsubtype xxsubtype.c
|
||||
@MODULE__XXTESTFUZZ_TRUE@_xxtestfuzz _xxtestfuzz/_xxtestfuzz.c _xxtestfuzz/fuzzer.c
|
||||
@MODULE__TESTBUFFER_TRUE@_testbuffer _testbuffer.c
|
||||
@MODULE__TESTINTERNALCAPI_TRUE@_testinternalcapi _testinternalcapi.c _testinternalcapi/test_lock.c _testinternalcapi/pytime.c _testinternalcapi/set.c _testinternalcapi/test_critical_sections.c _testinternalcapi/complex.c
|
||||
@MODULE__TESTINTERNALCAPI_TRUE@_testinternalcapi _testinternalcapi.c _testinternalcapi/test_lock.c _testinternalcapi/pytime.c _testinternalcapi/set.c _testinternalcapi/test_critical_sections.c _testinternalcapi/complex.c _testinternalcapi/interpreter.c
|
||||
@MODULE__TESTCAPI_TRUE@_testcapi _testcapimodule.c _testcapi/vectorcall.c _testcapi/heaptype.c _testcapi/abstract.c _testcapi/unicode.c _testcapi/dict.c _testcapi/set.c _testcapi/list.c _testcapi/tuple.c _testcapi/getargs.c _testcapi/datetime.c _testcapi/docstring.c _testcapi/mem.c _testcapi/watchers.c _testcapi/long.c _testcapi/float.c _testcapi/complex.c _testcapi/numbers.c _testcapi/structmember.c _testcapi/exceptions.c _testcapi/code.c _testcapi/buffer.c _testcapi/pyatomic.c _testcapi/run.c _testcapi/file.c _testcapi/codec.c _testcapi/immortal.c _testcapi/gc.c _testcapi/hash.c _testcapi/time.c _testcapi/bytes.c _testcapi/object.c _testcapi/modsupport.c _testcapi/monitoring.c _testcapi/config.c _testcapi/import.c _testcapi/frame.c _testcapi/type.c _testcapi/function.c _testcapi/module.c
|
||||
@MODULE__TESTLIMITEDCAPI_TRUE@_testlimitedcapi _testlimitedcapi.c _testlimitedcapi/abstract.c _testlimitedcapi/bytearray.c _testlimitedcapi/bytes.c _testlimitedcapi/codec.c _testlimitedcapi/complex.c _testlimitedcapi/dict.c _testlimitedcapi/eval.c _testlimitedcapi/float.c _testlimitedcapi/heaptype_relative.c _testlimitedcapi/import.c _testlimitedcapi/list.c _testlimitedcapi/long.c _testlimitedcapi/object.c _testlimitedcapi/pyos.c _testlimitedcapi/set.c _testlimitedcapi/sys.c _testlimitedcapi/tuple.c _testlimitedcapi/unicode.c _testlimitedcapi/vectorcall_limited.c _testlimitedcapi/version.c _testlimitedcapi/file.c
|
||||
@MODULE__TESTCLINIC_TRUE@_testclinic _testclinic.c
|
||||
|
|
|
|||
|
|
@ -693,6 +693,43 @@ set_eval_frame_record(PyObject *self, PyObject *list)
|
|||
Py_RETURN_NONE;
|
||||
}
|
||||
|
||||
// Defined in interpreter.c
|
||||
extern PyObject*
|
||||
Test_EvalFrame(PyThreadState *tstate, _PyInterpreterFrame *frame, int throwflag);
|
||||
extern int Test_EvalFrame_Resumes, Test_EvalFrame_Loads;
|
||||
|
||||
static PyObject *
|
||||
get_eval_frame_stats(PyObject *self, PyObject *Py_UNUSED(args))
|
||||
{
|
||||
PyObject *res = PyDict_New();
|
||||
if (res == NULL) {
|
||||
return NULL;
|
||||
}
|
||||
PyObject *resumes = PyLong_FromLong(Test_EvalFrame_Resumes);
|
||||
if (resumes == NULL || PyDict_SetItemString(res, "resumes", resumes) < 0) {
|
||||
Py_XDECREF(resumes);
|
||||
Py_DECREF(res);
|
||||
return NULL;
|
||||
}
|
||||
Py_DECREF(resumes);
|
||||
PyObject *loads = PyLong_FromLong(Test_EvalFrame_Loads);
|
||||
if (loads == NULL || PyDict_SetItemString(res, "loads", loads) < 0) {
|
||||
Py_XDECREF(loads);
|
||||
Py_DECREF(res);
|
||||
return NULL;
|
||||
}
|
||||
Py_DECREF(loads);
|
||||
Test_EvalFrame_Resumes = Test_EvalFrame_Loads = 0;
|
||||
return res;
|
||||
}
|
||||
|
||||
static PyObject *
|
||||
set_eval_frame_interp(PyObject *self, PyObject *Py_UNUSED(args))
|
||||
{
|
||||
_PyInterpreterState_SetEvalFrameFunc(_PyInterpreterState_GET(), Test_EvalFrame);
|
||||
Py_RETURN_NONE;
|
||||
}
|
||||
|
||||
/*[clinic input]
|
||||
|
||||
_testinternalcapi.compiler_cleandoc -> object
|
||||
|
|
@ -2531,6 +2568,7 @@ test_threadstate_set_stack_protection(PyObject *self, PyObject *Py_UNUSED(args))
|
|||
|
||||
static PyMethodDef module_functions[] = {
|
||||
{"get_configs", get_configs, METH_NOARGS},
|
||||
{"get_eval_frame_stats", get_eval_frame_stats, METH_NOARGS, NULL},
|
||||
{"get_recursion_depth", get_recursion_depth, METH_NOARGS},
|
||||
{"get_c_recursion_remaining", get_c_recursion_remaining, METH_NOARGS},
|
||||
{"get_stack_pointer", get_stack_pointer, METH_NOARGS},
|
||||
|
|
@ -2547,6 +2585,7 @@ static PyMethodDef module_functions[] = {
|
|||
{"EncodeLocaleEx", encode_locale_ex, METH_VARARGS},
|
||||
{"DecodeLocaleEx", decode_locale_ex, METH_VARARGS},
|
||||
{"set_eval_frame_default", set_eval_frame_default, METH_NOARGS, NULL},
|
||||
{"set_eval_frame_interp", set_eval_frame_interp, METH_NOARGS, NULL},
|
||||
{"set_eval_frame_record", set_eval_frame_record, METH_O, NULL},
|
||||
_TESTINTERNALCAPI_COMPILER_CLEANDOC_METHODDEF
|
||||
_TESTINTERNALCAPI_NEW_INSTRUCTION_SEQUENCE_METHODDEF
|
||||
|
|
|
|||
254
Modules/_testinternalcapi/interpreter.c
Normal file
254
Modules/_testinternalcapi/interpreter.c
Normal file
|
|
@ -0,0 +1,254 @@
|
|||
#ifndef Py_BUILD_CORE
|
||||
#define Py_BUILD_CORE
|
||||
#endif
|
||||
#ifndef Py_BUILD_CORE_MODULE
|
||||
#define Py_BUILD_CORE_MODULE
|
||||
#endif
|
||||
|
||||
#include "../../Python/ceval.h"
|
||||
|
||||
#include "../../Python/ceval_macros.h"
|
||||
|
||||
int Test_EvalFrame_Resumes, Test_EvalFrame_Loads;
|
||||
|
||||
#if _Py_TAIL_CALL_INTERP
|
||||
#include "test_targets.h"
|
||||
#include "test_cases.c.h"
|
||||
#endif
|
||||
|
||||
PyObject* _Py_HOT_FUNCTION
|
||||
Test_EvalFrame(PyThreadState *tstate, _PyInterpreterFrame *frame, int throwflag)
|
||||
{
|
||||
_Py_EnsureTstateNotNULL(tstate);
|
||||
check_invalid_reentrancy();
|
||||
CALL_STAT_INC(pyeval_calls);
|
||||
|
||||
#if USE_COMPUTED_GOTOS && !_Py_TAIL_CALL_INTERP
|
||||
/* Import the static jump table */
|
||||
#include "test_targets.h"
|
||||
void **opcode_targets = opcode_targets_table;
|
||||
#endif
|
||||
|
||||
#ifdef Py_STATS
|
||||
int lastopcode = 0;
|
||||
#endif
|
||||
#if !_Py_TAIL_CALL_INTERP
|
||||
uint8_t opcode; /* Current opcode */
|
||||
int oparg; /* Current opcode argument, if any */
|
||||
assert(tstate->current_frame == NULL || tstate->current_frame->stackpointer != NULL);
|
||||
#if !USE_COMPUTED_GOTOS
|
||||
uint8_t tracing_mode = 0;
|
||||
uint8_t dispatch_code;
|
||||
#endif
|
||||
#endif
|
||||
_PyEntryFrame entry;
|
||||
|
||||
if (_Py_EnterRecursiveCallTstate(tstate, "")) {
|
||||
assert(frame->owner != FRAME_OWNED_BY_INTERPRETER);
|
||||
_PyEval_FrameClearAndPop(tstate, frame);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/* Local "register" variables.
|
||||
* These are cached values from the frame and code object. */
|
||||
_Py_CODEUNIT *next_instr;
|
||||
_PyStackRef *stack_pointer;
|
||||
entry.stack[0] = PyStackRef_NULL;
|
||||
#ifdef Py_STACKREF_DEBUG
|
||||
entry.frame.f_funcobj = PyStackRef_None;
|
||||
#elif defined(Py_DEBUG)
|
||||
/* Set these to invalid but identifiable values for debugging. */
|
||||
entry.frame.f_funcobj = (_PyStackRef){.bits = 0xaaa0};
|
||||
entry.frame.f_locals = (PyObject*)0xaaa1;
|
||||
entry.frame.frame_obj = (PyFrameObject*)0xaaa2;
|
||||
entry.frame.f_globals = (PyObject*)0xaaa3;
|
||||
entry.frame.f_builtins = (PyObject*)0xaaa4;
|
||||
#endif
|
||||
entry.frame.f_executable = PyStackRef_None;
|
||||
entry.frame.instr_ptr = (_Py_CODEUNIT *)_Py_INTERPRETER_TRAMPOLINE_INSTRUCTIONS_PTR + 1;
|
||||
entry.frame.stackpointer = entry.stack;
|
||||
entry.frame.owner = FRAME_OWNED_BY_INTERPRETER;
|
||||
entry.frame.visited = 0;
|
||||
entry.frame.return_offset = 0;
|
||||
#ifdef Py_DEBUG
|
||||
entry.frame.lltrace = 0;
|
||||
#endif
|
||||
/* Push frame */
|
||||
entry.frame.previous = tstate->current_frame;
|
||||
frame->previous = &entry.frame;
|
||||
tstate->current_frame = frame;
|
||||
entry.frame.localsplus[0] = PyStackRef_NULL;
|
||||
#ifdef _Py_TIER2
|
||||
if (tstate->current_executor != NULL) {
|
||||
entry.frame.localsplus[0] = PyStackRef_FromPyObjectNew(tstate->current_executor);
|
||||
tstate->current_executor = NULL;
|
||||
}
|
||||
#endif
|
||||
|
||||
/* support for generator.throw() */
|
||||
if (throwflag) {
|
||||
if (_Py_EnterRecursivePy(tstate)) {
|
||||
goto early_exit;
|
||||
}
|
||||
#ifdef Py_GIL_DISABLED
|
||||
/* Load thread-local bytecode */
|
||||
if (frame->tlbc_index != ((_PyThreadStateImpl *)tstate)->tlbc_index) {
|
||||
_Py_CODEUNIT *bytecode =
|
||||
_PyEval_GetExecutableCode(tstate, _PyFrame_GetCode(frame));
|
||||
if (bytecode == NULL) {
|
||||
goto early_exit;
|
||||
}
|
||||
ptrdiff_t off = frame->instr_ptr - _PyFrame_GetBytecode(frame);
|
||||
frame->tlbc_index = ((_PyThreadStateImpl *)tstate)->tlbc_index;
|
||||
frame->instr_ptr = bytecode + off;
|
||||
}
|
||||
#endif
|
||||
/* Because this avoids the RESUME, we need to update instrumentation */
|
||||
_Py_Instrument(_PyFrame_GetCode(frame), tstate->interp);
|
||||
next_instr = frame->instr_ptr;
|
||||
monitor_throw(tstate, frame, next_instr);
|
||||
stack_pointer = _PyFrame_GetStackPointer(frame);
|
||||
#if _Py_TAIL_CALL_INTERP
|
||||
# if Py_STATS
|
||||
return _TAIL_CALL_error(frame, stack_pointer, tstate, next_instr, instruction_funcptr_handler_table, 0, lastopcode);
|
||||
# else
|
||||
return _TAIL_CALL_error(frame, stack_pointer, tstate, next_instr, instruction_funcptr_handler_table, 0);
|
||||
# endif
|
||||
#else
|
||||
goto error;
|
||||
#endif
|
||||
}
|
||||
|
||||
#if defined(_Py_TIER2) && !defined(_Py_JIT)
|
||||
/* Tier 2 interpreter state */
|
||||
_PyExecutorObject *current_executor = NULL;
|
||||
const _PyUOpInstruction *next_uop = NULL;
|
||||
#endif
|
||||
#if _Py_TAIL_CALL_INTERP
|
||||
# if Py_STATS
|
||||
return _TAIL_CALL_start_frame(frame, NULL, tstate, NULL, instruction_funcptr_handler_table, 0, lastopcode);
|
||||
# else
|
||||
return _TAIL_CALL_start_frame(frame, NULL, tstate, NULL, instruction_funcptr_handler_table, 0);
|
||||
# endif
|
||||
#else
|
||||
goto start_frame;
|
||||
#include "test_cases.c.h"
|
||||
#endif
|
||||
|
||||
|
||||
#ifdef _Py_TIER2
|
||||
|
||||
// Tier 2 is also here!
|
||||
enter_tier_two:
|
||||
|
||||
#ifdef _Py_JIT
|
||||
assert(0);
|
||||
#else
|
||||
|
||||
#undef LOAD_IP
|
||||
#define LOAD_IP(UNUSED) (void)0
|
||||
|
||||
#ifdef Py_STATS
|
||||
// Disable these macros that apply to Tier 1 stats when we are in Tier 2
|
||||
#undef STAT_INC
|
||||
#define STAT_INC(opname, name) ((void)0)
|
||||
#undef STAT_DEC
|
||||
#define STAT_DEC(opname, name) ((void)0)
|
||||
#endif
|
||||
|
||||
#undef ENABLE_SPECIALIZATION
|
||||
#define ENABLE_SPECIALIZATION 0
|
||||
#undef ENABLE_SPECIALIZATION_FT
|
||||
#define ENABLE_SPECIALIZATION_FT 0
|
||||
|
||||
; // dummy statement after a label, before a declaration
|
||||
uint16_t uopcode;
|
||||
#ifdef Py_STATS
|
||||
int lastuop = 0;
|
||||
uint64_t trace_uop_execution_counter = 0;
|
||||
#endif
|
||||
|
||||
assert(next_uop->opcode == _START_EXECUTOR);
|
||||
tier2_dispatch:
|
||||
for (;;) {
|
||||
uopcode = next_uop->opcode;
|
||||
#ifdef Py_DEBUG
|
||||
if (frame->lltrace >= 3) {
|
||||
dump_stack(frame, stack_pointer);
|
||||
if (next_uop->opcode == _START_EXECUTOR) {
|
||||
printf("%4d uop: ", 0);
|
||||
}
|
||||
else {
|
||||
printf("%4d uop: ", (int)(next_uop - current_executor->trace));
|
||||
}
|
||||
_PyUOpPrint(next_uop);
|
||||
printf("\n");
|
||||
}
|
||||
#endif
|
||||
next_uop++;
|
||||
OPT_STAT_INC(uops_executed);
|
||||
UOP_STAT_INC(uopcode, execution_count);
|
||||
UOP_PAIR_INC(uopcode, lastuop);
|
||||
#ifdef Py_STATS
|
||||
trace_uop_execution_counter++;
|
||||
((_PyUOpInstruction *)next_uop)[-1].execution_count++;
|
||||
#endif
|
||||
|
||||
switch (uopcode) {
|
||||
|
||||
#include "executor_cases.c.h"
|
||||
|
||||
default:
|
||||
#ifdef Py_DEBUG
|
||||
{
|
||||
printf("Unknown uop: ");
|
||||
_PyUOpPrint(&next_uop[-1]);
|
||||
printf(" @ %d\n", (int)(next_uop - current_executor->trace - 1));
|
||||
Py_FatalError("Unknown uop");
|
||||
}
|
||||
#else
|
||||
Py_UNREACHABLE();
|
||||
#endif
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
jump_to_error_target:
|
||||
#ifdef Py_DEBUG
|
||||
if (frame->lltrace >= 2) {
|
||||
printf("Error: [UOp ");
|
||||
_PyUOpPrint(&next_uop[-1]);
|
||||
printf(" @ %d -> %s]\n",
|
||||
(int)(next_uop - current_executor->trace - 1),
|
||||
_PyOpcode_OpName[frame->instr_ptr->op.code]);
|
||||
}
|
||||
#endif
|
||||
assert(next_uop[-1].format == UOP_FORMAT_JUMP);
|
||||
uint16_t target = uop_get_error_target(&next_uop[-1]);
|
||||
next_uop = current_executor->trace + target;
|
||||
goto tier2_dispatch;
|
||||
|
||||
jump_to_jump_target:
|
||||
assert(next_uop[-1].format == UOP_FORMAT_JUMP);
|
||||
target = uop_get_jump_target(&next_uop[-1]);
|
||||
next_uop = current_executor->trace + target;
|
||||
goto tier2_dispatch;
|
||||
|
||||
#endif // _Py_JIT
|
||||
|
||||
#endif // _Py_TIER2
|
||||
|
||||
early_exit:
|
||||
assert(_PyErr_Occurred(tstate));
|
||||
_Py_LeaveRecursiveCallPy(tstate);
|
||||
assert(frame->owner != FRAME_OWNED_BY_INTERPRETER);
|
||||
// GH-99729: We need to unlink the frame *before* clearing it:
|
||||
_PyInterpreterFrame *dying = frame;
|
||||
frame = tstate->current_frame = dying->previous;
|
||||
_PyEval_FrameClearAndPop(tstate, dying);
|
||||
frame->return_offset = 0;
|
||||
assert(frame->owner == FRAME_OWNED_BY_INTERPRETER);
|
||||
/* Restore previous frame and exit */
|
||||
tstate->current_frame = frame->previous;
|
||||
return NULL;
|
||||
}
|
||||
12513
Modules/_testinternalcapi/test_cases.c.h
Normal file
12513
Modules/_testinternalcapi/test_cases.c.h
Normal file
File diff suppressed because it is too large
Load diff
1289
Modules/_testinternalcapi/test_targets.h
Normal file
1289
Modules/_testinternalcapi/test_targets.h
Normal file
File diff suppressed because it is too large
Load diff
177
Modules/_testinternalcapi/testbytecodes.c
Normal file
177
Modules/_testinternalcapi/testbytecodes.c
Normal file
|
|
@ -0,0 +1,177 @@
|
|||
// This file contains instruction definitions.
|
||||
// It is read by generators stored in Tools/cases_generator/
|
||||
// to generate Python/generated_cases.c.h and others.
|
||||
// Note that there is some dummy C code at the top and bottom of the file
|
||||
// to fool text editors like VS Code into believing this is valid C code.
|
||||
// The actual instruction definitions start at // BEGIN BYTECODES //.
|
||||
// See Tools/cases_generator/README.md for more information.
|
||||
|
||||
#include "Python.h"
|
||||
#include "pycore_abstract.h" // _PyIndex_Check()
|
||||
#include "pycore_audit.h" // _PySys_Audit()
|
||||
#include "pycore_backoff.h"
|
||||
#include "pycore_cell.h" // PyCell_GetRef()
|
||||
#include "pycore_code.h"
|
||||
#include "pycore_emscripten_signal.h" // _Py_CHECK_EMSCRIPTEN_SIGNALS
|
||||
#include "pycore_function.h"
|
||||
#include "pycore_instruments.h"
|
||||
#include "pycore_interpolation.h" // _PyInterpolation_Build()
|
||||
#include "pycore_intrinsics.h"
|
||||
#include "pycore_long.h" // _PyLong_ExactDealloc(), _PyLong_GetZero()
|
||||
#include "pycore_moduleobject.h" // PyModuleObject
|
||||
#include "pycore_object.h" // _PyObject_GC_TRACK()
|
||||
#include "pycore_opcode_metadata.h" // uop names
|
||||
#include "pycore_opcode_utils.h" // MAKE_FUNCTION_*
|
||||
#include "pycore_pyatomic_ft_wrappers.h" // FT_ATOMIC_*
|
||||
#include "pycore_pyerrors.h" // _PyErr_GetRaisedException()
|
||||
#include "pycore_pystate.h" // _PyInterpreterState_GET()
|
||||
#include "pycore_range.h" // _PyRangeIterObject
|
||||
#include "pycore_setobject.h" // _PySet_NextEntry()
|
||||
#include "pycore_sliceobject.h" // _PyBuildSlice_ConsumeRefs
|
||||
#include "pycore_stackref.h"
|
||||
#include "pycore_template.h" // _PyTemplate_Build()
|
||||
#include "pycore_tuple.h" // _PyTuple_ITEMS()
|
||||
#include "pycore_typeobject.h" // _PySuper_Lookup()
|
||||
|
||||
#include "pycore_dict.h"
|
||||
#include "dictobject.h"
|
||||
#include "pycore_frame.h"
|
||||
#include "opcode.h"
|
||||
#include "optimizer.h"
|
||||
#include "pydtrace.h"
|
||||
#include "setobject.h"
|
||||
|
||||
|
||||
#define USE_COMPUTED_GOTOS 0
|
||||
#include "ceval_macros.h"
|
||||
|
||||
/* Flow control macros */
|
||||
|
||||
#define inst(name, ...) case name:
|
||||
#define op(name, ...) /* NAME is ignored */
|
||||
#define macro(name) static int MACRO_##name
|
||||
#define super(name) static int SUPER_##name
|
||||
#define family(name, ...) static int family_##name
|
||||
#define pseudo(name) static int pseudo_##name
|
||||
#define label(name) name:
|
||||
|
||||
/* Annotations */
|
||||
#define guard
|
||||
#define override
|
||||
#define specializing
|
||||
#define replicate(TIMES)
|
||||
#define tier1
|
||||
#define no_save_ip
|
||||
|
||||
// Dummy variables for stack effects.
|
||||
static PyObject *value, *value1, *value2, *left, *right, *res, *sum, *prod, *sub;
|
||||
static PyObject *container, *start, *stop, *v, *lhs, *rhs, *res2;
|
||||
static PyObject *list, *tuple, *dict, *owner, *set, *str, *tup, *map, *keys;
|
||||
static PyObject *exit_func, *lasti, *val, *retval, *obj, *iter, *exhausted;
|
||||
static PyObject *aiter, *awaitable, *iterable, *w, *exc_value, *bc, *locals;
|
||||
static PyObject *orig, *excs, *update, *b, *fromlist, *level, *from;
|
||||
static PyObject **pieces, **values;
|
||||
static size_t jump;
|
||||
// Dummy variables for cache effects
|
||||
static uint16_t invert, counter, index, hint;
|
||||
#define unused 0 // Used in a macro def, can't be static
|
||||
static uint32_t type_version;
|
||||
static _PyExecutorObject *current_executor;
|
||||
|
||||
static PyObject *
|
||||
dummy_func(
|
||||
PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
unsigned char opcode,
|
||||
unsigned int oparg,
|
||||
_Py_CODEUNIT *next_instr,
|
||||
PyObject **stack_pointer,
|
||||
int throwflag,
|
||||
PyObject *args[]
|
||||
)
|
||||
{
|
||||
// Dummy labels.
|
||||
pop_1_error:
|
||||
// Dummy locals.
|
||||
PyObject *dummy;
|
||||
_Py_CODEUNIT *this_instr;
|
||||
PyObject *attr;
|
||||
PyObject *attrs;
|
||||
PyObject *bottom;
|
||||
PyObject *callable;
|
||||
PyObject *callargs;
|
||||
PyObject *codeobj;
|
||||
PyObject *cond;
|
||||
PyObject *descr;
|
||||
PyObject *exc;
|
||||
PyObject *exit;
|
||||
PyObject *fget;
|
||||
PyObject *fmt_spec;
|
||||
PyObject *func;
|
||||
uint32_t func_version;
|
||||
PyObject *getattribute;
|
||||
PyObject *kwargs;
|
||||
PyObject *kwdefaults;
|
||||
PyObject *len_o;
|
||||
PyObject *match;
|
||||
PyObject *match_type;
|
||||
PyObject *method;
|
||||
PyObject *mgr;
|
||||
Py_ssize_t min_args;
|
||||
PyObject *names;
|
||||
PyObject *new_exc;
|
||||
PyObject *next;
|
||||
PyObject *none;
|
||||
PyObject *null;
|
||||
PyObject *prev_exc;
|
||||
PyObject *receiver;
|
||||
PyObject *rest;
|
||||
int result;
|
||||
PyObject *self;
|
||||
PyObject *seq;
|
||||
PyObject *slice;
|
||||
PyObject *step;
|
||||
PyObject *subject;
|
||||
PyObject *top;
|
||||
PyObject *type;
|
||||
PyObject *typevars;
|
||||
PyObject *val0;
|
||||
PyObject *val1;
|
||||
int values_or_none;
|
||||
|
||||
switch (opcode) {
|
||||
|
||||
// BEGIN BYTECODES //
|
||||
|
||||
// Override an op
|
||||
override op(_CHECK_PERIODIC_IF_NOT_YIELD_FROM, (--)) {
|
||||
Test_EvalFrame_Resumes++;
|
||||
if ((oparg & RESUME_OPARG_LOCATION_MASK) < RESUME_AFTER_YIELD_FROM) {
|
||||
int err = check_periodics(tstate);
|
||||
ERROR_IF(err != 0);
|
||||
}
|
||||
}
|
||||
|
||||
// Override an inst
|
||||
override inst(LOAD_CONST, (-- value)) {
|
||||
Test_EvalFrame_Loads++;
|
||||
PyObject *obj = GETITEM(FRAME_CO_CONSTS, oparg);
|
||||
value = PyStackRef_FromPyObjectBorrow(obj);
|
||||
}
|
||||
|
||||
|
||||
// END BYTECODES //
|
||||
|
||||
}
|
||||
dispatch_opcode:
|
||||
error:
|
||||
exception_unwind:
|
||||
exit_unwind:
|
||||
handle_eval_breaker:
|
||||
resume_frame:
|
||||
start_frame:
|
||||
unbound_local_error:
|
||||
;
|
||||
}
|
||||
|
||||
// Future families go below this point //
|
||||
|
|
@ -19,6 +19,7 @@ this type and there is exactly one in existence.
|
|||
#include "pycore_long.h" // _PyLong_GetZero()
|
||||
#include "pycore_modsupport.h" // _PyArg_NoKeywords()
|
||||
#include "pycore_object.h" // _PyObject_GC_TRACK()
|
||||
#include "pycore_sliceobject.h" // _PyBuildSlice_ConsumeRefs
|
||||
|
||||
|
||||
#define _PySlice_CAST(op) _Py_CAST(PySliceObject*, (op))
|
||||
|
|
|
|||
|
|
@ -99,6 +99,7 @@
|
|||
<ClCompile Include="..\Modules\_testinternalcapi\test_lock.c" />
|
||||
<ClCompile Include="..\Modules\_testinternalcapi\set.c" />
|
||||
<ClCompile Include="..\Modules\_testinternalcapi\complex.c" />
|
||||
<ClCompile Include="..\Modules\_testinternalcapi\interpreter.c" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ResourceCompile Include="..\PC\python_nt.rc" />
|
||||
|
|
|
|||
|
|
@ -112,6 +112,10 @@
|
|||
WorkingDirectory="$(PySourcePath)" />
|
||||
<Exec Command="$(PythonForBuild) Tools\cases_generator\uop_metadata_generator.py Python\bytecodes.c"
|
||||
WorkingDirectory="$(PySourcePath)" />
|
||||
<Exec Command="$(PythonForBuild) Tools\cases_generator\target_generator.py -o Modules\_testinternalcapi\test_targets.h Python\bytecodes.c Modules\_testinternalcapi\testbytecodes.c"
|
||||
WorkingDirectory="$(PySourcePath)" />
|
||||
<Exec Command="$(PythonForBuild) Tools\cases_generator\tier1_generator.py -o Modules\_testinternalcapi\test_cases.c.h Python\bytecodes.c Modules\_testinternalcapi\testbytecodes.c"
|
||||
WorkingDirectory="$(PySourcePath)" />
|
||||
</Target>
|
||||
|
||||
<Target Name="_RegenJIT"
|
||||
|
|
|
|||
626
Python/ceval.c
626
Python/ceval.c
|
|
@ -1,338 +1,6 @@
|
|||
/* Execute compiled code */
|
||||
|
||||
#define _PY_INTERPRETER
|
||||
|
||||
#include "Python.h"
|
||||
#include "pycore_abstract.h" // _PyIndex_Check()
|
||||
#include "pycore_audit.h" // _PySys_Audit()
|
||||
#include "pycore_backoff.h"
|
||||
#include "pycore_call.h" // _PyObject_CallNoArgs()
|
||||
#include "pycore_cell.h" // PyCell_GetRef()
|
||||
#include "pycore_ceval.h" // SPECIAL___ENTER__
|
||||
#include "pycore_code.h"
|
||||
#include "pycore_dict.h"
|
||||
#include "pycore_emscripten_signal.h" // _Py_CHECK_EMSCRIPTEN_SIGNALS
|
||||
#include "pycore_floatobject.h" // _PyFloat_ExactDealloc()
|
||||
#include "pycore_frame.h"
|
||||
#include "pycore_function.h"
|
||||
#include "pycore_genobject.h" // _PyCoro_GetAwaitableIter()
|
||||
#include "pycore_import.h" // _PyImport_IsDefaultImportFunc()
|
||||
#include "pycore_instruments.h"
|
||||
#include "pycore_interpframe.h" // _PyFrame_SetStackPointer()
|
||||
#include "pycore_interpolation.h" // _PyInterpolation_Build()
|
||||
#include "pycore_intrinsics.h"
|
||||
#include "pycore_jit.h"
|
||||
#include "pycore_list.h" // _PyList_GetItemRef()
|
||||
#include "pycore_long.h" // _PyLong_GetZero()
|
||||
#include "pycore_moduleobject.h" // PyModuleObject
|
||||
#include "pycore_object.h" // _PyObject_GC_TRACK()
|
||||
#include "pycore_opcode_metadata.h" // EXTRA_CASES
|
||||
#include "pycore_opcode_utils.h" // MAKE_FUNCTION_*
|
||||
#include "pycore_optimizer.h" // _PyUOpExecutor_Type
|
||||
#include "pycore_pyatomic_ft_wrappers.h" // FT_ATOMIC_*
|
||||
#include "pycore_pyerrors.h" // _PyErr_GetRaisedException()
|
||||
#include "pycore_pystate.h" // _PyInterpreterState_GET()
|
||||
#include "pycore_range.h" // _PyRangeIterObject
|
||||
#include "pycore_setobject.h" // _PySet_Update()
|
||||
#include "pycore_sliceobject.h" // _PyBuildSlice_ConsumeRefs
|
||||
#include "pycore_sysmodule.h" // _PySys_GetOptionalAttrString()
|
||||
#include "pycore_template.h" // _PyTemplate_Build()
|
||||
#include "pycore_traceback.h" // _PyTraceBack_FromFrame
|
||||
#include "pycore_tuple.h" // _PyTuple_ITEMS()
|
||||
#include "pycore_uop_ids.h" // Uops
|
||||
|
||||
#include "dictobject.h"
|
||||
#include "frameobject.h" // _PyInterpreterFrame_GetLine
|
||||
#include "opcode.h"
|
||||
#include "pydtrace.h"
|
||||
#include "setobject.h"
|
||||
#include "pycore_stackref.h"
|
||||
|
||||
#include <stdbool.h> // bool
|
||||
|
||||
#if !defined(Py_BUILD_CORE)
|
||||
# error "ceval.c must be build with Py_BUILD_CORE define for best performance"
|
||||
#endif
|
||||
|
||||
#if !defined(Py_DEBUG) && !defined(Py_TRACE_REFS)
|
||||
// GH-89279: The MSVC compiler does not inline these static inline functions
|
||||
// in PGO build in _PyEval_EvalFrameDefault(), because this function is over
|
||||
// the limit of PGO, and that limit cannot be configured.
|
||||
// Define them as macros to make sure that they are always inlined by the
|
||||
// preprocessor.
|
||||
|
||||
#undef Py_IS_TYPE
|
||||
#define Py_IS_TYPE(ob, type) \
|
||||
(_PyObject_CAST(ob)->ob_type == (type))
|
||||
|
||||
#undef Py_XDECREF
|
||||
#define Py_XDECREF(arg) \
|
||||
do { \
|
||||
PyObject *xop = _PyObject_CAST(arg); \
|
||||
if (xop != NULL) { \
|
||||
Py_DECREF(xop); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#ifndef Py_GIL_DISABLED
|
||||
|
||||
#undef Py_DECREF
|
||||
#define Py_DECREF(arg) \
|
||||
do { \
|
||||
PyObject *op = _PyObject_CAST(arg); \
|
||||
if (_Py_IsImmortal(op)) { \
|
||||
_Py_DECREF_IMMORTAL_STAT_INC(); \
|
||||
break; \
|
||||
} \
|
||||
_Py_DECREF_STAT_INC(); \
|
||||
if (--op->ob_refcnt == 0) { \
|
||||
_PyReftracerTrack(op, PyRefTracer_DESTROY); \
|
||||
destructor dealloc = Py_TYPE(op)->tp_dealloc; \
|
||||
(*dealloc)(op); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#undef _Py_DECREF_SPECIALIZED
|
||||
#define _Py_DECREF_SPECIALIZED(arg, dealloc) \
|
||||
do { \
|
||||
PyObject *op = _PyObject_CAST(arg); \
|
||||
if (_Py_IsImmortal(op)) { \
|
||||
_Py_DECREF_IMMORTAL_STAT_INC(); \
|
||||
break; \
|
||||
} \
|
||||
_Py_DECREF_STAT_INC(); \
|
||||
if (--op->ob_refcnt == 0) { \
|
||||
_PyReftracerTrack(op, PyRefTracer_DESTROY); \
|
||||
destructor d = (destructor)(dealloc); \
|
||||
d(op); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#else // Py_GIL_DISABLED
|
||||
|
||||
#undef Py_DECREF
|
||||
#define Py_DECREF(arg) \
|
||||
do { \
|
||||
PyObject *op = _PyObject_CAST(arg); \
|
||||
uint32_t local = _Py_atomic_load_uint32_relaxed(&op->ob_ref_local); \
|
||||
if (local == _Py_IMMORTAL_REFCNT_LOCAL) { \
|
||||
_Py_DECREF_IMMORTAL_STAT_INC(); \
|
||||
break; \
|
||||
} \
|
||||
_Py_DECREF_STAT_INC(); \
|
||||
if (_Py_IsOwnedByCurrentThread(op)) { \
|
||||
local--; \
|
||||
_Py_atomic_store_uint32_relaxed(&op->ob_ref_local, local); \
|
||||
if (local == 0) { \
|
||||
_Py_MergeZeroLocalRefcount(op); \
|
||||
} \
|
||||
} \
|
||||
else { \
|
||||
_Py_DecRefShared(op); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#undef _Py_DECREF_SPECIALIZED
|
||||
#define _Py_DECREF_SPECIALIZED(arg, dealloc) Py_DECREF(arg)
|
||||
|
||||
#endif
|
||||
#endif
|
||||
|
||||
|
||||
static void
|
||||
check_invalid_reentrancy(void)
|
||||
{
|
||||
#if defined(Py_DEBUG) && defined(Py_GIL_DISABLED)
|
||||
// In the free-threaded build, the interpreter must not be re-entered if
|
||||
// the world-is-stopped. If so, that's a bug somewhere (quite likely in
|
||||
// the painfully complex typeobject code).
|
||||
PyInterpreterState *interp = _PyInterpreterState_GET();
|
||||
assert(!interp->stoptheworld.world_stopped);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
#ifdef Py_DEBUG
|
||||
static void
|
||||
dump_item(_PyStackRef item)
|
||||
{
|
||||
if (PyStackRef_IsNull(item)) {
|
||||
printf("<NULL>");
|
||||
return;
|
||||
}
|
||||
if (PyStackRef_IsMalformed(item)) {
|
||||
printf("<INVALID>");
|
||||
return;
|
||||
}
|
||||
if (PyStackRef_IsTaggedInt(item)) {
|
||||
printf("%" PRId64, (int64_t)PyStackRef_UntagInt(item));
|
||||
return;
|
||||
}
|
||||
PyObject *obj = PyStackRef_AsPyObjectBorrow(item);
|
||||
if (obj == NULL) {
|
||||
printf("<nil>");
|
||||
return;
|
||||
}
|
||||
// Don't call __repr__(), it might recurse into the interpreter.
|
||||
printf("<%s at %p>", Py_TYPE(obj)->tp_name, (void *)obj);
|
||||
}
|
||||
|
||||
static void
|
||||
dump_stack(_PyInterpreterFrame *frame, _PyStackRef *stack_pointer)
|
||||
{
|
||||
_PyFrame_SetStackPointer(frame, stack_pointer);
|
||||
_PyStackRef *locals_base = _PyFrame_GetLocalsArray(frame);
|
||||
_PyStackRef *stack_base = _PyFrame_Stackbase(frame);
|
||||
PyObject *exc = PyErr_GetRaisedException();
|
||||
printf(" locals=[");
|
||||
for (_PyStackRef *ptr = locals_base; ptr < stack_base; ptr++) {
|
||||
if (ptr != locals_base) {
|
||||
printf(", ");
|
||||
}
|
||||
dump_item(*ptr);
|
||||
}
|
||||
printf("]\n");
|
||||
if (stack_pointer < stack_base) {
|
||||
printf(" stack=%d\n", (int)(stack_pointer-stack_base));
|
||||
}
|
||||
else {
|
||||
printf(" stack=[");
|
||||
for (_PyStackRef *ptr = stack_base; ptr < stack_pointer; ptr++) {
|
||||
if (ptr != stack_base) {
|
||||
printf(", ");
|
||||
}
|
||||
dump_item(*ptr);
|
||||
}
|
||||
printf("]\n");
|
||||
}
|
||||
fflush(stdout);
|
||||
PyErr_SetRaisedException(exc);
|
||||
_PyFrame_GetStackPointer(frame);
|
||||
}
|
||||
|
||||
#if defined(_Py_TIER2) && !defined(_Py_JIT) && defined(Py_DEBUG)
|
||||
static void
|
||||
dump_cache_item(_PyStackRef cache, int position, int depth)
|
||||
{
|
||||
if (position < depth) {
|
||||
dump_item(cache);
|
||||
}
|
||||
else {
|
||||
printf("---");
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
static void
|
||||
lltrace_instruction(_PyInterpreterFrame *frame,
|
||||
_PyStackRef *stack_pointer,
|
||||
_Py_CODEUNIT *next_instr,
|
||||
int opcode,
|
||||
int oparg)
|
||||
{
|
||||
int offset = 0;
|
||||
if (frame->owner < FRAME_OWNED_BY_INTERPRETER) {
|
||||
dump_stack(frame, stack_pointer);
|
||||
offset = (int)(next_instr - _PyFrame_GetBytecode(frame));
|
||||
}
|
||||
const char *opname = _PyOpcode_OpName[opcode];
|
||||
assert(opname != NULL);
|
||||
if (OPCODE_HAS_ARG((int)_PyOpcode_Deopt[opcode])) {
|
||||
printf("%d: %s %d\n", offset * 2, opname, oparg);
|
||||
}
|
||||
else {
|
||||
printf("%d: %s\n", offset * 2, opname);
|
||||
}
|
||||
fflush(stdout);
|
||||
}
|
||||
|
||||
static void
|
||||
lltrace_resume_frame(_PyInterpreterFrame *frame)
|
||||
{
|
||||
PyObject *fobj = PyStackRef_AsPyObjectBorrow(frame->f_funcobj);
|
||||
if (!PyStackRef_CodeCheck(frame->f_executable) ||
|
||||
fobj == NULL ||
|
||||
!PyFunction_Check(fobj)
|
||||
) {
|
||||
printf("\nResuming frame.\n");
|
||||
return;
|
||||
}
|
||||
PyFunctionObject *f = (PyFunctionObject *)fobj;
|
||||
PyObject *exc = PyErr_GetRaisedException();
|
||||
PyObject *name = f->func_qualname;
|
||||
if (name == NULL) {
|
||||
name = f->func_name;
|
||||
}
|
||||
printf("\nResuming frame");
|
||||
if (name) {
|
||||
printf(" for ");
|
||||
if (PyObject_Print(name, stdout, 0) < 0) {
|
||||
PyErr_Clear();
|
||||
}
|
||||
}
|
||||
if (f->func_module) {
|
||||
printf(" in module ");
|
||||
if (PyObject_Print(f->func_module, stdout, 0) < 0) {
|
||||
PyErr_Clear();
|
||||
}
|
||||
}
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
PyErr_SetRaisedException(exc);
|
||||
}
|
||||
|
||||
static int
|
||||
maybe_lltrace_resume_frame(_PyInterpreterFrame *frame, PyObject *globals)
|
||||
{
|
||||
if (globals == NULL) {
|
||||
return 0;
|
||||
}
|
||||
if (frame->owner >= FRAME_OWNED_BY_INTERPRETER) {
|
||||
return 0;
|
||||
}
|
||||
int r = PyDict_Contains(globals, &_Py_ID(__lltrace__));
|
||||
if (r < 0) {
|
||||
PyErr_Clear();
|
||||
return 0;
|
||||
}
|
||||
int lltrace = r * 5; // Levels 1-4 only trace uops
|
||||
if (!lltrace) {
|
||||
// Can also be controlled by environment variable
|
||||
char *python_lltrace = Py_GETENV("PYTHON_LLTRACE");
|
||||
if (python_lltrace != NULL && *python_lltrace >= '0') {
|
||||
lltrace = *python_lltrace - '0'; // TODO: Parse an int and all that
|
||||
}
|
||||
}
|
||||
if (lltrace >= 5) {
|
||||
lltrace_resume_frame(frame);
|
||||
}
|
||||
return lltrace;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
static void monitor_reraise(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr);
|
||||
static int monitor_stop_iteration(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr,
|
||||
PyObject *value);
|
||||
static void monitor_unwind(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr);
|
||||
static int monitor_handled(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, PyObject *exc);
|
||||
static void monitor_throw(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr);
|
||||
|
||||
static int get_exception_handler(PyCodeObject *, int, int*, int*, int*);
|
||||
|
||||
#ifdef HAVE_ERRNO_H
|
||||
#include <errno.h>
|
||||
#endif
|
||||
#include "ceval.h"
|
||||
|
||||
int
|
||||
Py_GetRecursionLimit(void)
|
||||
|
|
@ -1401,19 +1069,6 @@ extern void _PyUOpPrint(const _PyUOpInstruction *uop);
|
|||
#endif
|
||||
|
||||
|
||||
/* Disable unused label warnings. They are handy for debugging, even
|
||||
if computed gotos aren't used. */
|
||||
|
||||
/* TBD - what about other compilers? */
|
||||
#if defined(__GNUC__) || defined(__clang__)
|
||||
# pragma GCC diagnostic push
|
||||
# pragma GCC diagnostic ignored "-Wunused-label"
|
||||
#elif defined(_MSC_VER) /* MS_WINDOWS */
|
||||
# pragma warning(push)
|
||||
# pragma warning(disable:4102)
|
||||
#endif
|
||||
|
||||
|
||||
PyObject **
|
||||
_PyObjectArray_FromStackRefArray(_PyStackRef *input, Py_ssize_t nargs, PyObject **scratch)
|
||||
{
|
||||
|
|
@ -1444,12 +1099,6 @@ _PyObjectArray_Free(PyObject **array, PyObject **scratch)
|
|||
}
|
||||
}
|
||||
|
||||
#ifdef Py_DEBUG
|
||||
#define ASSERT_WITHIN_STACK_BOUNDS(F, L) _Py_assert_within_stack_bounds(frame, stack_pointer, (F), (L))
|
||||
#else
|
||||
#define ASSERT_WITHIN_STACK_BOUNDS(F, L) (void)0
|
||||
#endif
|
||||
|
||||
#if _Py_TIER2
|
||||
// 0 for success, -1 for error.
|
||||
static int
|
||||
|
|
@ -1498,11 +1147,6 @@ stop_tracing_and_jit(PyThreadState *tstate, _PyInterpreterFrame *frame)
|
|||
#define DONT_SLP_VECTORIZE
|
||||
#endif
|
||||
|
||||
typedef struct {
|
||||
_PyInterpreterFrame frame;
|
||||
_PyStackRef stack[1];
|
||||
} _PyEntryFrame;
|
||||
|
||||
PyObject* _Py_HOT_FUNCTION DONT_SLP_VECTORIZE
|
||||
_PyEval_EvalFrameDefault(PyThreadState *tstate, _PyInterpreterFrame *frame, int throwflag)
|
||||
{
|
||||
|
|
@ -2001,74 +1645,6 @@ positional_only_passed_as_keyword(PyThreadState *tstate, PyCodeObject *co,
|
|||
|
||||
}
|
||||
|
||||
|
||||
static inline unsigned char *
|
||||
scan_back_to_entry_start(unsigned char *p) {
|
||||
for (; (p[0]&128) == 0; p--);
|
||||
return p;
|
||||
}
|
||||
|
||||
static inline unsigned char *
|
||||
skip_to_next_entry(unsigned char *p, unsigned char *end) {
|
||||
while (p < end && ((p[0] & 128) == 0)) {
|
||||
p++;
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
|
||||
#define MAX_LINEAR_SEARCH 40
|
||||
|
||||
static Py_NO_INLINE int
|
||||
get_exception_handler(PyCodeObject *code, int index, int *level, int *handler, int *lasti)
|
||||
{
|
||||
unsigned char *start = (unsigned char *)PyBytes_AS_STRING(code->co_exceptiontable);
|
||||
unsigned char *end = start + PyBytes_GET_SIZE(code->co_exceptiontable);
|
||||
/* Invariants:
|
||||
* start_table == end_table OR
|
||||
* start_table points to a legal entry and end_table points
|
||||
* beyond the table or to a legal entry that is after index.
|
||||
*/
|
||||
if (end - start > MAX_LINEAR_SEARCH) {
|
||||
int offset;
|
||||
parse_varint(start, &offset);
|
||||
if (offset > index) {
|
||||
return 0;
|
||||
}
|
||||
do {
|
||||
unsigned char * mid = start + ((end-start)>>1);
|
||||
mid = scan_back_to_entry_start(mid);
|
||||
parse_varint(mid, &offset);
|
||||
if (offset > index) {
|
||||
end = mid;
|
||||
}
|
||||
else {
|
||||
start = mid;
|
||||
}
|
||||
|
||||
} while (end - start > MAX_LINEAR_SEARCH);
|
||||
}
|
||||
unsigned char *scan = start;
|
||||
while (scan < end) {
|
||||
int start_offset, size;
|
||||
scan = parse_varint(scan, &start_offset);
|
||||
if (start_offset > index) {
|
||||
break;
|
||||
}
|
||||
scan = parse_varint(scan, &size);
|
||||
if (start_offset + size > index) {
|
||||
scan = parse_varint(scan, handler);
|
||||
int depth_and_lasti;
|
||||
parse_varint(scan, &depth_and_lasti);
|
||||
*level = depth_and_lasti >> 1;
|
||||
*lasti = depth_and_lasti & 1;
|
||||
return 1;
|
||||
}
|
||||
scan = skip_to_next_entry(scan, end);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
initialize_locals(PyThreadState *tstate, PyFunctionObject *func,
|
||||
_PyStackRef *localsplus, _PyStackRef const *args,
|
||||
|
|
@ -2590,109 +2166,6 @@ PyEval_EvalCodeEx(PyObject *_co, PyObject *globals, PyObject *locals,
|
|||
return res;
|
||||
}
|
||||
|
||||
|
||||
/* Logic for the raise statement (too complicated for inlining).
|
||||
This *consumes* a reference count to each of its arguments. */
|
||||
static int
|
||||
do_raise(PyThreadState *tstate, PyObject *exc, PyObject *cause)
|
||||
{
|
||||
PyObject *type = NULL, *value = NULL;
|
||||
|
||||
if (exc == NULL) {
|
||||
/* Reraise */
|
||||
_PyErr_StackItem *exc_info = _PyErr_GetTopmostException(tstate);
|
||||
exc = exc_info->exc_value;
|
||||
if (Py_IsNone(exc) || exc == NULL) {
|
||||
_PyErr_SetString(tstate, PyExc_RuntimeError,
|
||||
"No active exception to reraise");
|
||||
return 0;
|
||||
}
|
||||
Py_INCREF(exc);
|
||||
assert(PyExceptionInstance_Check(exc));
|
||||
_PyErr_SetRaisedException(tstate, exc);
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* We support the following forms of raise:
|
||||
raise
|
||||
raise <instance>
|
||||
raise <type> */
|
||||
|
||||
if (PyExceptionClass_Check(exc)) {
|
||||
type = exc;
|
||||
value = _PyObject_CallNoArgs(exc);
|
||||
if (value == NULL)
|
||||
goto raise_error;
|
||||
if (!PyExceptionInstance_Check(value)) {
|
||||
_PyErr_Format(tstate, PyExc_TypeError,
|
||||
"calling %R should have returned an instance of "
|
||||
"BaseException, not %R",
|
||||
type, Py_TYPE(value));
|
||||
goto raise_error;
|
||||
}
|
||||
}
|
||||
else if (PyExceptionInstance_Check(exc)) {
|
||||
value = exc;
|
||||
type = PyExceptionInstance_Class(exc);
|
||||
Py_INCREF(type);
|
||||
}
|
||||
else {
|
||||
/* Not something you can raise. You get an exception
|
||||
anyway, just not what you specified :-) */
|
||||
Py_DECREF(exc);
|
||||
_PyErr_SetString(tstate, PyExc_TypeError,
|
||||
"exceptions must derive from BaseException");
|
||||
goto raise_error;
|
||||
}
|
||||
|
||||
assert(type != NULL);
|
||||
assert(value != NULL);
|
||||
|
||||
if (cause) {
|
||||
PyObject *fixed_cause;
|
||||
if (PyExceptionClass_Check(cause)) {
|
||||
fixed_cause = _PyObject_CallNoArgs(cause);
|
||||
if (fixed_cause == NULL)
|
||||
goto raise_error;
|
||||
if (!PyExceptionInstance_Check(fixed_cause)) {
|
||||
_PyErr_Format(tstate, PyExc_TypeError,
|
||||
"calling %R should have returned an instance of "
|
||||
"BaseException, not %R",
|
||||
cause, Py_TYPE(fixed_cause));
|
||||
Py_DECREF(fixed_cause);
|
||||
goto raise_error;
|
||||
}
|
||||
Py_DECREF(cause);
|
||||
}
|
||||
else if (PyExceptionInstance_Check(cause)) {
|
||||
fixed_cause = cause;
|
||||
}
|
||||
else if (Py_IsNone(cause)) {
|
||||
Py_DECREF(cause);
|
||||
fixed_cause = NULL;
|
||||
}
|
||||
else {
|
||||
_PyErr_SetString(tstate, PyExc_TypeError,
|
||||
"exception causes must derive from "
|
||||
"BaseException");
|
||||
goto raise_error;
|
||||
}
|
||||
PyException_SetCause(value, fixed_cause);
|
||||
}
|
||||
|
||||
_PyErr_SetObject(tstate, type, value);
|
||||
/* _PyErr_SetObject incref's its arguments */
|
||||
Py_DECREF(value);
|
||||
Py_DECREF(type);
|
||||
return 0;
|
||||
|
||||
raise_error:
|
||||
Py_XDECREF(value);
|
||||
Py_XDECREF(type);
|
||||
Py_XDECREF(cause);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Logic for matching an exception in an except* clause (too
|
||||
complicated for inlining).
|
||||
*/
|
||||
|
|
@ -2890,45 +2363,7 @@ _PyEval_UnpackIterableStackRef(PyThreadState *tstate, PyObject *v,
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
do_monitor_exc(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, int event)
|
||||
{
|
||||
assert(event < _PY_MONITORING_UNGROUPED_EVENTS);
|
||||
if (_PyFrame_GetCode(frame)->co_flags & CO_NO_MONITORING_EVENTS) {
|
||||
return 0;
|
||||
}
|
||||
PyObject *exc = PyErr_GetRaisedException();
|
||||
assert(exc != NULL);
|
||||
int err = _Py_call_instrumentation_arg(tstate, event, frame, instr, exc);
|
||||
if (err == 0) {
|
||||
PyErr_SetRaisedException(exc);
|
||||
}
|
||||
else {
|
||||
assert(PyErr_Occurred());
|
||||
Py_DECREF(exc);
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline bool
|
||||
no_tools_for_global_event(PyThreadState *tstate, int event)
|
||||
{
|
||||
return tstate->interp->monitors.tools[event] == 0;
|
||||
}
|
||||
|
||||
static inline bool
|
||||
no_tools_for_local_event(PyThreadState *tstate, _PyInterpreterFrame *frame, int event)
|
||||
{
|
||||
assert(event < _PY_MONITORING_LOCAL_EVENTS);
|
||||
_PyCoMonitoringData *data = _PyFrame_GetCode(frame)->_co_monitoring;
|
||||
if (data) {
|
||||
return data->active_monitors.tools[event] == 0;
|
||||
}
|
||||
else {
|
||||
return no_tools_for_global_event(tstate, event);
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
_PyEval_MonitorRaise(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
|
|
@ -2940,70 +2375,11 @@ _PyEval_MonitorRaise(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
|||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_RAISE);
|
||||
}
|
||||
|
||||
static void
|
||||
monitor_reraise(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_RERAISE)) {
|
||||
return;
|
||||
}
|
||||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_RERAISE);
|
||||
}
|
||||
|
||||
static int
|
||||
monitor_stop_iteration(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, PyObject *value)
|
||||
{
|
||||
if (no_tools_for_local_event(tstate, frame, PY_MONITORING_EVENT_STOP_ITERATION)) {
|
||||
return 0;
|
||||
}
|
||||
assert(!PyErr_Occurred());
|
||||
PyErr_SetObject(PyExc_StopIteration, value);
|
||||
int res = do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_STOP_ITERATION);
|
||||
if (res < 0) {
|
||||
return res;
|
||||
}
|
||||
PyErr_SetRaisedException(NULL);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
monitor_unwind(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_PY_UNWIND)) {
|
||||
return;
|
||||
}
|
||||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_PY_UNWIND);
|
||||
}
|
||||
|
||||
bool
|
||||
_PyEval_NoToolsForUnwind(PyThreadState *tstate) {
|
||||
return no_tools_for_global_event(tstate, PY_MONITORING_EVENT_PY_UNWIND);
|
||||
}
|
||||
|
||||
static int
|
||||
monitor_handled(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, PyObject *exc)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_EXCEPTION_HANDLED)) {
|
||||
return 0;
|
||||
}
|
||||
return _Py_call_instrumentation_arg(tstate, PY_MONITORING_EVENT_EXCEPTION_HANDLED, frame, instr, exc);
|
||||
}
|
||||
|
||||
static void
|
||||
monitor_throw(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_PY_THROW)) {
|
||||
return;
|
||||
}
|
||||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_PY_THROW);
|
||||
}
|
||||
|
||||
void
|
||||
PyThreadState_EnterTracing(PyThreadState *tstate)
|
||||
|
|
|
|||
625
Python/ceval.h
Normal file
625
Python/ceval.h
Normal file
|
|
@ -0,0 +1,625 @@
|
|||
#define _PY_INTERPRETER
|
||||
|
||||
#include "Python.h"
|
||||
#include "pycore_abstract.h" // _PyIndex_Check()
|
||||
#include "pycore_audit.h" // _PySys_Audit()
|
||||
#include "pycore_backoff.h"
|
||||
#include "pycore_call.h" // _PyObject_CallNoArgs()
|
||||
#include "pycore_cell.h" // PyCell_GetRef()
|
||||
#include "pycore_ceval.h" // SPECIAL___ENTER__
|
||||
#include "pycore_code.h"
|
||||
#include "pycore_dict.h"
|
||||
#include "pycore_emscripten_signal.h" // _Py_CHECK_EMSCRIPTEN_SIGNALS
|
||||
#include "pycore_floatobject.h" // _PyFloat_ExactDealloc()
|
||||
#include "pycore_frame.h"
|
||||
#include "pycore_function.h"
|
||||
#include "pycore_genobject.h" // _PyCoro_GetAwaitableIter()
|
||||
#include "pycore_import.h" // _PyImport_IsDefaultImportFunc()
|
||||
#include "pycore_instruments.h"
|
||||
#include "pycore_interpframe.h" // _PyFrame_SetStackPointer()
|
||||
#include "pycore_interpolation.h" // _PyInterpolation_Build()
|
||||
#include "pycore_intrinsics.h"
|
||||
#include "pycore_jit.h"
|
||||
#include "pycore_list.h" // _PyList_GetItemRef()
|
||||
#include "pycore_long.h" // _PyLong_GetZero()
|
||||
#include "pycore_moduleobject.h" // PyModuleObject
|
||||
#include "pycore_object.h" // _PyObject_GC_TRACK()
|
||||
#include "pycore_opcode_metadata.h" // EXTRA_CASES
|
||||
#include "pycore_opcode_utils.h" // MAKE_FUNCTION_*
|
||||
#include "pycore_optimizer.h" // _PyUOpExecutor_Type
|
||||
#include "pycore_pyatomic_ft_wrappers.h" // FT_ATOMIC_*
|
||||
#include "pycore_pyerrors.h" // _PyErr_GetRaisedException()
|
||||
#include "pycore_pystate.h" // _PyInterpreterState_GET()
|
||||
#include "pycore_range.h" // _PyRangeIterObject
|
||||
#include "pycore_setobject.h" // _PySet_Update()
|
||||
#include "pycore_sliceobject.h" // _PyBuildSlice_ConsumeRefs
|
||||
#include "pycore_sysmodule.h" // _PySys_GetOptionalAttrString()
|
||||
#include "pycore_template.h" // _PyTemplate_Build()
|
||||
#include "pycore_traceback.h" // _PyTraceBack_FromFrame
|
||||
#include "pycore_tuple.h" // _PyTuple_ITEMS()
|
||||
#include "pycore_uop_ids.h" // Uops
|
||||
|
||||
#include "dictobject.h"
|
||||
#include "frameobject.h" // _PyInterpreterFrame_GetLine
|
||||
#include "opcode.h"
|
||||
#include "pydtrace.h"
|
||||
#include "setobject.h"
|
||||
#include "pycore_stackref.h"
|
||||
|
||||
#include <stdbool.h> // bool
|
||||
|
||||
#if !defined(Py_BUILD_CORE)
|
||||
# error "ceval.c must be build with Py_BUILD_CORE define for best performance"
|
||||
#endif
|
||||
|
||||
#if !defined(Py_DEBUG) && !defined(Py_TRACE_REFS)
|
||||
// GH-89279: The MSVC compiler does not inline these static inline functions
|
||||
// in PGO build in _PyEval_EvalFrameDefault(), because this function is over
|
||||
// the limit of PGO, and that limit cannot be configured.
|
||||
// Define them as macros to make sure that they are always inlined by the
|
||||
// preprocessor.
|
||||
|
||||
#undef Py_IS_TYPE
|
||||
#define Py_IS_TYPE(ob, type) \
|
||||
(_PyObject_CAST(ob)->ob_type == (type))
|
||||
|
||||
#undef Py_XDECREF
|
||||
#define Py_XDECREF(arg) \
|
||||
do { \
|
||||
PyObject *xop = _PyObject_CAST(arg); \
|
||||
if (xop != NULL) { \
|
||||
Py_DECREF(xop); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#ifndef Py_GIL_DISABLED
|
||||
|
||||
#undef Py_DECREF
|
||||
#define Py_DECREF(arg) \
|
||||
do { \
|
||||
PyObject *op = _PyObject_CAST(arg); \
|
||||
if (_Py_IsImmortal(op)) { \
|
||||
_Py_DECREF_IMMORTAL_STAT_INC(); \
|
||||
break; \
|
||||
} \
|
||||
_Py_DECREF_STAT_INC(); \
|
||||
if (--op->ob_refcnt == 0) { \
|
||||
_PyReftracerTrack(op, PyRefTracer_DESTROY); \
|
||||
destructor dealloc = Py_TYPE(op)->tp_dealloc; \
|
||||
(*dealloc)(op); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#undef _Py_DECREF_SPECIALIZED
|
||||
#define _Py_DECREF_SPECIALIZED(arg, dealloc) \
|
||||
do { \
|
||||
PyObject *op = _PyObject_CAST(arg); \
|
||||
if (_Py_IsImmortal(op)) { \
|
||||
_Py_DECREF_IMMORTAL_STAT_INC(); \
|
||||
break; \
|
||||
} \
|
||||
_Py_DECREF_STAT_INC(); \
|
||||
if (--op->ob_refcnt == 0) { \
|
||||
_PyReftracerTrack(op, PyRefTracer_DESTROY); \
|
||||
destructor d = (destructor)(dealloc); \
|
||||
d(op); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#else // Py_GIL_DISABLED
|
||||
|
||||
#undef Py_DECREF
|
||||
#define Py_DECREF(arg) \
|
||||
do { \
|
||||
PyObject *op = _PyObject_CAST(arg); \
|
||||
uint32_t local = _Py_atomic_load_uint32_relaxed(&op->ob_ref_local); \
|
||||
if (local == _Py_IMMORTAL_REFCNT_LOCAL) { \
|
||||
_Py_DECREF_IMMORTAL_STAT_INC(); \
|
||||
break; \
|
||||
} \
|
||||
_Py_DECREF_STAT_INC(); \
|
||||
if (_Py_IsOwnedByCurrentThread(op)) { \
|
||||
local--; \
|
||||
_Py_atomic_store_uint32_relaxed(&op->ob_ref_local, local); \
|
||||
if (local == 0) { \
|
||||
_Py_MergeZeroLocalRefcount(op); \
|
||||
} \
|
||||
} \
|
||||
else { \
|
||||
_Py_DecRefShared(op); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#undef _Py_DECREF_SPECIALIZED
|
||||
#define _Py_DECREF_SPECIALIZED(arg, dealloc) Py_DECREF(arg)
|
||||
|
||||
#endif
|
||||
#endif
|
||||
|
||||
static void
|
||||
check_invalid_reentrancy(void)
|
||||
{
|
||||
#if defined(Py_DEBUG) && defined(Py_GIL_DISABLED)
|
||||
// In the free-threaded build, the interpreter must not be re-entered if
|
||||
// the world-is-stopped. If so, that's a bug somewhere (quite likely in
|
||||
// the painfully complex typeobject code).
|
||||
PyInterpreterState *interp = _PyInterpreterState_GET();
|
||||
assert(!interp->stoptheworld.world_stopped);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
#ifdef Py_DEBUG
|
||||
static void
|
||||
dump_item(_PyStackRef item)
|
||||
{
|
||||
if (PyStackRef_IsNull(item)) {
|
||||
printf("<NULL>");
|
||||
return;
|
||||
}
|
||||
if (PyStackRef_IsMalformed(item)) {
|
||||
printf("<INVALID>");
|
||||
return;
|
||||
}
|
||||
if (PyStackRef_IsTaggedInt(item)) {
|
||||
printf("%" PRId64, (int64_t)PyStackRef_UntagInt(item));
|
||||
return;
|
||||
}
|
||||
PyObject *obj = PyStackRef_AsPyObjectBorrow(item);
|
||||
if (obj == NULL) {
|
||||
printf("<nil>");
|
||||
return;
|
||||
}
|
||||
// Don't call __repr__(), it might recurse into the interpreter.
|
||||
printf("<%s at %p>", Py_TYPE(obj)->tp_name, (void *)obj);
|
||||
}
|
||||
|
||||
static void
|
||||
dump_stack(_PyInterpreterFrame *frame, _PyStackRef *stack_pointer)
|
||||
{
|
||||
_PyFrame_SetStackPointer(frame, stack_pointer);
|
||||
_PyStackRef *locals_base = _PyFrame_GetLocalsArray(frame);
|
||||
_PyStackRef *stack_base = _PyFrame_Stackbase(frame);
|
||||
PyObject *exc = PyErr_GetRaisedException();
|
||||
printf(" locals=[");
|
||||
for (_PyStackRef *ptr = locals_base; ptr < stack_base; ptr++) {
|
||||
if (ptr != locals_base) {
|
||||
printf(", ");
|
||||
}
|
||||
dump_item(*ptr);
|
||||
}
|
||||
printf("]\n");
|
||||
if (stack_pointer < stack_base) {
|
||||
printf(" stack=%d\n", (int)(stack_pointer-stack_base));
|
||||
}
|
||||
else {
|
||||
printf(" stack=[");
|
||||
for (_PyStackRef *ptr = stack_base; ptr < stack_pointer; ptr++) {
|
||||
if (ptr != stack_base) {
|
||||
printf(", ");
|
||||
}
|
||||
dump_item(*ptr);
|
||||
}
|
||||
printf("]\n");
|
||||
}
|
||||
fflush(stdout);
|
||||
PyErr_SetRaisedException(exc);
|
||||
_PyFrame_GetStackPointer(frame);
|
||||
}
|
||||
|
||||
#if defined(_Py_TIER2) && !defined(_Py_JIT) && defined(Py_DEBUG)
|
||||
static void
|
||||
dump_cache_item(_PyStackRef cache, int position, int depth)
|
||||
{
|
||||
if (position < depth) {
|
||||
dump_item(cache);
|
||||
}
|
||||
else {
|
||||
printf("---");
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
static void
|
||||
lltrace_instruction(_PyInterpreterFrame *frame,
|
||||
_PyStackRef *stack_pointer,
|
||||
_Py_CODEUNIT *next_instr,
|
||||
int opcode,
|
||||
int oparg)
|
||||
{
|
||||
int offset = 0;
|
||||
if (frame->owner < FRAME_OWNED_BY_INTERPRETER) {
|
||||
dump_stack(frame, stack_pointer);
|
||||
offset = (int)(next_instr - _PyFrame_GetBytecode(frame));
|
||||
}
|
||||
const char *opname = _PyOpcode_OpName[opcode];
|
||||
assert(opname != NULL);
|
||||
if (OPCODE_HAS_ARG((int)_PyOpcode_Deopt[opcode])) {
|
||||
printf("%d: %s %d\n", offset * 2, opname, oparg);
|
||||
}
|
||||
else {
|
||||
printf("%d: %s\n", offset * 2, opname);
|
||||
}
|
||||
fflush(stdout);
|
||||
}
|
||||
|
||||
static void
|
||||
lltrace_resume_frame(_PyInterpreterFrame *frame)
|
||||
{
|
||||
PyObject *fobj = PyStackRef_AsPyObjectBorrow(frame->f_funcobj);
|
||||
if (!PyStackRef_CodeCheck(frame->f_executable) ||
|
||||
fobj == NULL ||
|
||||
!PyFunction_Check(fobj)
|
||||
) {
|
||||
printf("\nResuming frame.\n");
|
||||
return;
|
||||
}
|
||||
PyFunctionObject *f = (PyFunctionObject *)fobj;
|
||||
PyObject *exc = PyErr_GetRaisedException();
|
||||
PyObject *name = f->func_qualname;
|
||||
if (name == NULL) {
|
||||
name = f->func_name;
|
||||
}
|
||||
printf("\nResuming frame");
|
||||
if (name) {
|
||||
printf(" for ");
|
||||
if (PyObject_Print(name, stdout, 0) < 0) {
|
||||
PyErr_Clear();
|
||||
}
|
||||
}
|
||||
if (f->func_module) {
|
||||
printf(" in module ");
|
||||
if (PyObject_Print(f->func_module, stdout, 0) < 0) {
|
||||
PyErr_Clear();
|
||||
}
|
||||
}
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
PyErr_SetRaisedException(exc);
|
||||
}
|
||||
|
||||
static int
|
||||
maybe_lltrace_resume_frame(_PyInterpreterFrame *frame, PyObject *globals)
|
||||
{
|
||||
if (globals == NULL) {
|
||||
return 0;
|
||||
}
|
||||
if (frame->owner >= FRAME_OWNED_BY_INTERPRETER) {
|
||||
return 0;
|
||||
}
|
||||
int r = PyDict_Contains(globals, &_Py_ID(__lltrace__));
|
||||
if (r < 0) {
|
||||
PyErr_Clear();
|
||||
return 0;
|
||||
}
|
||||
int lltrace = r * 5; // Levels 1-4 only trace uops
|
||||
if (!lltrace) {
|
||||
// Can also be controlled by environment variable
|
||||
char *python_lltrace = Py_GETENV("PYTHON_LLTRACE");
|
||||
if (python_lltrace != NULL && *python_lltrace >= '0') {
|
||||
lltrace = *python_lltrace - '0'; // TODO: Parse an int and all that
|
||||
}
|
||||
}
|
||||
if (lltrace >= 5) {
|
||||
lltrace_resume_frame(frame);
|
||||
}
|
||||
return lltrace;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
static void monitor_reraise(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr);
|
||||
static int monitor_stop_iteration(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr,
|
||||
PyObject *value);
|
||||
static void monitor_unwind(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr);
|
||||
static int monitor_handled(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, PyObject *exc);
|
||||
static void monitor_throw(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr);
|
||||
|
||||
static int get_exception_handler(PyCodeObject *, int, int*, int*, int*);
|
||||
|
||||
#ifdef HAVE_ERRNO_H
|
||||
#include <errno.h>
|
||||
#endif
|
||||
|
||||
typedef struct {
|
||||
_PyInterpreterFrame frame;
|
||||
_PyStackRef stack[1];
|
||||
} _PyEntryFrame;
|
||||
|
||||
static int
|
||||
do_monitor_exc(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, int event)
|
||||
{
|
||||
assert(event < _PY_MONITORING_UNGROUPED_EVENTS);
|
||||
if (_PyFrame_GetCode(frame)->co_flags & CO_NO_MONITORING_EVENTS) {
|
||||
return 0;
|
||||
}
|
||||
PyObject *exc = PyErr_GetRaisedException();
|
||||
assert(exc != NULL);
|
||||
int err = _Py_call_instrumentation_arg(tstate, event, frame, instr, exc);
|
||||
if (err == 0) {
|
||||
PyErr_SetRaisedException(exc);
|
||||
}
|
||||
else {
|
||||
assert(PyErr_Occurred());
|
||||
Py_DECREF(exc);
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
static inline bool
|
||||
no_tools_for_global_event(PyThreadState *tstate, int event)
|
||||
{
|
||||
return tstate->interp->monitors.tools[event] == 0;
|
||||
}
|
||||
|
||||
static inline bool
|
||||
no_tools_for_local_event(PyThreadState *tstate, _PyInterpreterFrame *frame, int event)
|
||||
{
|
||||
assert(event < _PY_MONITORING_LOCAL_EVENTS);
|
||||
_PyCoMonitoringData *data = _PyFrame_GetCode(frame)->_co_monitoring;
|
||||
if (data) {
|
||||
return data->active_monitors.tools[event] == 0;
|
||||
}
|
||||
else {
|
||||
return no_tools_for_global_event(tstate, event);
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
monitor_handled(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, PyObject *exc)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_EXCEPTION_HANDLED)) {
|
||||
return 0;
|
||||
}
|
||||
return _Py_call_instrumentation_arg(tstate, PY_MONITORING_EVENT_EXCEPTION_HANDLED, frame, instr, exc);
|
||||
}
|
||||
|
||||
static void
|
||||
monitor_throw(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_PY_THROW)) {
|
||||
return;
|
||||
}
|
||||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_PY_THROW);
|
||||
}
|
||||
|
||||
static void
|
||||
monitor_reraise(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_RERAISE)) {
|
||||
return;
|
||||
}
|
||||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_RERAISE);
|
||||
}
|
||||
|
||||
static int
|
||||
monitor_stop_iteration(PyThreadState *tstate, _PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr, PyObject *value)
|
||||
{
|
||||
if (no_tools_for_local_event(tstate, frame, PY_MONITORING_EVENT_STOP_ITERATION)) {
|
||||
return 0;
|
||||
}
|
||||
assert(!PyErr_Occurred());
|
||||
PyErr_SetObject(PyExc_StopIteration, value);
|
||||
int res = do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_STOP_ITERATION);
|
||||
if (res < 0) {
|
||||
return res;
|
||||
}
|
||||
PyErr_SetRaisedException(NULL);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void
|
||||
monitor_unwind(PyThreadState *tstate,
|
||||
_PyInterpreterFrame *frame,
|
||||
_Py_CODEUNIT *instr)
|
||||
{
|
||||
if (no_tools_for_global_event(tstate, PY_MONITORING_EVENT_PY_UNWIND)) {
|
||||
return;
|
||||
}
|
||||
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_PY_UNWIND);
|
||||
}
|
||||
|
||||
static inline unsigned char *
|
||||
scan_back_to_entry_start(unsigned char *p) {
|
||||
for (; (p[0]&128) == 0; p--);
|
||||
return p;
|
||||
}
|
||||
|
||||
static inline unsigned char *
|
||||
skip_to_next_entry(unsigned char *p, unsigned char *end) {
|
||||
while (p < end && ((p[0] & 128) == 0)) {
|
||||
p++;
|
||||
}
|
||||
return p;
|
||||
}
|
||||
|
||||
|
||||
#define MAX_LINEAR_SEARCH 40
|
||||
|
||||
static Py_NO_INLINE int
|
||||
get_exception_handler(PyCodeObject *code, int index, int *level, int *handler, int *lasti)
|
||||
{
|
||||
unsigned char *start = (unsigned char *)PyBytes_AS_STRING(code->co_exceptiontable);
|
||||
unsigned char *end = start + PyBytes_GET_SIZE(code->co_exceptiontable);
|
||||
/* Invariants:
|
||||
* start_table == end_table OR
|
||||
* start_table points to a legal entry and end_table points
|
||||
* beyond the table or to a legal entry that is after index.
|
||||
*/
|
||||
if (end - start > MAX_LINEAR_SEARCH) {
|
||||
int offset;
|
||||
parse_varint(start, &offset);
|
||||
if (offset > index) {
|
||||
return 0;
|
||||
}
|
||||
do {
|
||||
unsigned char * mid = start + ((end-start)>>1);
|
||||
mid = scan_back_to_entry_start(mid);
|
||||
parse_varint(mid, &offset);
|
||||
if (offset > index) {
|
||||
end = mid;
|
||||
}
|
||||
else {
|
||||
start = mid;
|
||||
}
|
||||
|
||||
} while (end - start > MAX_LINEAR_SEARCH);
|
||||
}
|
||||
unsigned char *scan = start;
|
||||
while (scan < end) {
|
||||
int start_offset, size;
|
||||
scan = parse_varint(scan, &start_offset);
|
||||
if (start_offset > index) {
|
||||
break;
|
||||
}
|
||||
scan = parse_varint(scan, &size);
|
||||
if (start_offset + size > index) {
|
||||
scan = parse_varint(scan, handler);
|
||||
int depth_and_lasti;
|
||||
parse_varint(scan, &depth_and_lasti);
|
||||
*level = depth_and_lasti >> 1;
|
||||
*lasti = depth_and_lasti & 1;
|
||||
return 1;
|
||||
}
|
||||
scan = skip_to_next_entry(scan, end);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
#ifdef Py_DEBUG
|
||||
#define ASSERT_WITHIN_STACK_BOUNDS(F, L) _Py_assert_within_stack_bounds(frame, stack_pointer, (F), (L))
|
||||
#else
|
||||
#define ASSERT_WITHIN_STACK_BOUNDS(F, L) (void)0
|
||||
#endif
|
||||
|
||||
/* Logic for the raise statement (too complicated for inlining).
|
||||
This *consumes* a reference count to each of its arguments. */
|
||||
static int
|
||||
do_raise(PyThreadState *tstate, PyObject *exc, PyObject *cause)
|
||||
{
|
||||
PyObject *type = NULL, *value = NULL;
|
||||
|
||||
if (exc == NULL) {
|
||||
/* Reraise */
|
||||
_PyErr_StackItem *exc_info = _PyErr_GetTopmostException(tstate);
|
||||
exc = exc_info->exc_value;
|
||||
if (Py_IsNone(exc) || exc == NULL) {
|
||||
_PyErr_SetString(tstate, PyExc_RuntimeError,
|
||||
"No active exception to reraise");
|
||||
return 0;
|
||||
}
|
||||
Py_INCREF(exc);
|
||||
assert(PyExceptionInstance_Check(exc));
|
||||
_PyErr_SetRaisedException(tstate, exc);
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* We support the following forms of raise:
|
||||
raise
|
||||
raise <instance>
|
||||
raise <type> */
|
||||
|
||||
if (PyExceptionClass_Check(exc)) {
|
||||
type = exc;
|
||||
value = _PyObject_CallNoArgs(exc);
|
||||
if (value == NULL)
|
||||
goto raise_error;
|
||||
if (!PyExceptionInstance_Check(value)) {
|
||||
_PyErr_Format(tstate, PyExc_TypeError,
|
||||
"calling %R should have returned an instance of "
|
||||
"BaseException, not %R",
|
||||
type, Py_TYPE(value));
|
||||
goto raise_error;
|
||||
}
|
||||
}
|
||||
else if (PyExceptionInstance_Check(exc)) {
|
||||
value = exc;
|
||||
type = PyExceptionInstance_Class(exc);
|
||||
Py_INCREF(type);
|
||||
}
|
||||
else {
|
||||
/* Not something you can raise. You get an exception
|
||||
anyway, just not what you specified :-) */
|
||||
Py_DECREF(exc);
|
||||
_PyErr_SetString(tstate, PyExc_TypeError,
|
||||
"exceptions must derive from BaseException");
|
||||
goto raise_error;
|
||||
}
|
||||
|
||||
assert(type != NULL);
|
||||
assert(value != NULL);
|
||||
|
||||
if (cause) {
|
||||
PyObject *fixed_cause;
|
||||
if (PyExceptionClass_Check(cause)) {
|
||||
fixed_cause = _PyObject_CallNoArgs(cause);
|
||||
if (fixed_cause == NULL)
|
||||
goto raise_error;
|
||||
if (!PyExceptionInstance_Check(fixed_cause)) {
|
||||
_PyErr_Format(tstate, PyExc_TypeError,
|
||||
"calling %R should have returned an instance of "
|
||||
"BaseException, not %R",
|
||||
cause, Py_TYPE(fixed_cause));
|
||||
Py_DECREF(fixed_cause);
|
||||
goto raise_error;
|
||||
}
|
||||
Py_DECREF(cause);
|
||||
}
|
||||
else if (PyExceptionInstance_Check(cause)) {
|
||||
fixed_cause = cause;
|
||||
}
|
||||
else if (Py_IsNone(cause)) {
|
||||
Py_DECREF(cause);
|
||||
fixed_cause = NULL;
|
||||
}
|
||||
else {
|
||||
_PyErr_SetString(tstate, PyExc_TypeError,
|
||||
"exception causes must derive from "
|
||||
"BaseException");
|
||||
goto raise_error;
|
||||
}
|
||||
PyException_SetCause(value, fixed_cause);
|
||||
}
|
||||
|
||||
_PyErr_SetObject(tstate, type, value);
|
||||
/* _PyErr_SetObject incref's its arguments */
|
||||
Py_DECREF(value);
|
||||
Py_DECREF(type);
|
||||
return 0;
|
||||
|
||||
raise_error:
|
||||
Py_XDECREF(value);
|
||||
Py_XDECREF(type);
|
||||
Py_XDECREF(cause);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Disable unused label warnings. They are handy for debugging, even
|
||||
if computed gotos aren't used. */
|
||||
|
||||
/* TBD - what about other compilers? */
|
||||
#if defined(__GNUC__) || defined(__clang__)
|
||||
# pragma GCC diagnostic push
|
||||
# pragma GCC diagnostic ignored "-Wunused-label"
|
||||
#elif defined(_MSC_VER) /* MS_WINDOWS */
|
||||
# pragma warning(push)
|
||||
# pragma warning(disable:4102)
|
||||
#endif
|
||||
|
|
@ -81,12 +81,15 @@ def format_tsv_lines(lines):
|
|||
'Python/executor_cases.c.h',
|
||||
'Python/optimizer_cases.c.h',
|
||||
'Python/opcode_targets.h',
|
||||
'Modules/_testinternalcapi/test_targets.h',
|
||||
'Modules/_testinternalcapi/test_cases.c.h',
|
||||
# XXX: Throws errors if PY_VERSION_HEX is not mocked out
|
||||
'Modules/clinic/_testclinic_depr.c.h',
|
||||
|
||||
# not actually source
|
||||
'Python/bytecodes.c',
|
||||
'Python/optimizer_bytecodes.c',
|
||||
'Modules/_testinternalcapi/testbytecodes.c',
|
||||
|
||||
# mimalloc
|
||||
'Objects/mimalloc/*.c',
|
||||
|
|
|
|||
|
|
@ -567,6 +567,10 @@ Modules/_testimportmultiple.c - _barmodule -
|
|||
Modules/_testimportmultiple.c - _foomodule -
|
||||
Modules/_testimportmultiple.c - _testimportmultiple -
|
||||
Modules/_testinternalcapi.c - pending_identify_result -
|
||||
Modules/_testinternalcapi.c - Test_EvalFrame_Resumes -
|
||||
Modules/_testinternalcapi.c - Test_EvalFrame_Loads -
|
||||
Modules/_testinternalcapi/interpreter.c - Test_EvalFrame_Resumes -
|
||||
Modules/_testinternalcapi/interpreter.c - Test_EvalFrame_Loads -
|
||||
Modules/_testmultiphase.c - Example_Type_slots -
|
||||
Modules/_testmultiphase.c - Example_Type_spec -
|
||||
Modules/_testmultiphase.c - Example_methods -
|
||||
|
|
|
|||
|
Can't render this file because it has a wrong number of fields in line 4.
|
|
|
@ -150,7 +150,7 @@ def generate_instruction_formats(analysis: Analysis, out: CWriter) -> None:
|
|||
|
||||
|
||||
def generate_deopt_table(analysis: Analysis, out: CWriter) -> None:
|
||||
out.emit("extern const uint8_t _PyOpcode_Deopt[256];\n")
|
||||
out.emit("PyAPI_DATA(const uint8_t) _PyOpcode_Deopt[256];\n")
|
||||
out.emit("#ifdef NEED_OPCODE_METADATA\n")
|
||||
out.emit("const uint8_t _PyOpcode_Deopt[256] = {\n")
|
||||
deopts: list[tuple[str, str]] = []
|
||||
|
|
@ -173,7 +173,7 @@ def generate_deopt_table(analysis: Analysis, out: CWriter) -> None:
|
|||
|
||||
|
||||
def generate_cache_table(analysis: Analysis, out: CWriter) -> None:
|
||||
out.emit("extern const uint8_t _PyOpcode_Caches[256];\n")
|
||||
out.emit("PyAPI_DATA(const uint8_t) _PyOpcode_Caches[256];\n")
|
||||
out.emit("#ifdef NEED_OPCODE_METADATA\n")
|
||||
out.emit("const uint8_t _PyOpcode_Caches[256] = {\n")
|
||||
for inst in analysis.instructions.values():
|
||||
|
|
@ -189,7 +189,7 @@ def generate_cache_table(analysis: Analysis, out: CWriter) -> None:
|
|||
|
||||
def generate_name_table(analysis: Analysis, out: CWriter) -> None:
|
||||
table_size = 256 + len(analysis.pseudos)
|
||||
out.emit(f"extern const char *_PyOpcode_OpName[{table_size}];\n")
|
||||
out.emit(f"PyAPI_DATA(const char) *_PyOpcode_OpName[{table_size}];\n")
|
||||
out.emit("#ifdef NEED_OPCODE_METADATA\n")
|
||||
out.emit(f"const char *_PyOpcode_OpName[{table_size}] = {{\n")
|
||||
names = list(analysis.instructions) + list(analysis.pseudos)
|
||||
|
|
@ -207,7 +207,7 @@ def generate_metadata_table(analysis: Analysis, out: CWriter) -> None:
|
|||
out.emit("uint32_t flags;\n")
|
||||
out.emit("};\n\n")
|
||||
out.emit(
|
||||
f"extern const struct opcode_metadata _PyOpcode_opcode_metadata[{table_size}];\n"
|
||||
f"PyAPI_DATA(const struct opcode_metadata) _PyOpcode_opcode_metadata[{table_size}];\n"
|
||||
)
|
||||
out.emit("#ifdef NEED_OPCODE_METADATA\n")
|
||||
out.emit(
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue