mirror of https://github.com/python/cpython
26 lines
1.1 KiB
C
26 lines
1.1 KiB
C
|
#include "Python.h"
|
||
|
|
||
|
#include "pycore_ceval.h"
|
||
|
#include "pycore_frame.h"
|
||
|
#include "pycore_jit.h"
|
||
|
|
||
|
// This is where the calling convention changes, on platforms that require it.
|
||
|
// The actual change is patched in while the JIT compiler is being built, in
|
||
|
// Tools/jit/_targets.py. On other platforms, this function compiles to nothing.
|
||
|
_Py_CODEUNIT *
|
||
|
_ENTRY(_PyInterpreterFrame *frame, PyObject **stack_pointer, PyThreadState *tstate)
|
||
|
{
|
||
|
// This is subtle. The actual trace will return to us once it exits, so we
|
||
|
// need to make sure that we stay alive until then. If our trace side-exits
|
||
|
// into another trace, and this trace is then invalidated, the code for
|
||
|
// *this function* will be freed and we'll crash upon return:
|
||
|
PyAPI_DATA(void) _JIT_EXECUTOR;
|
||
|
PyObject *executor = (PyObject *)(uintptr_t)&_JIT_EXECUTOR;
|
||
|
Py_INCREF(executor);
|
||
|
// Note that this is *not* a tail call:
|
||
|
PyAPI_DATA(void) _JIT_CONTINUE;
|
||
|
_Py_CODEUNIT *target = ((jit_func)&_JIT_CONTINUE)(frame, stack_pointer, tstate);
|
||
|
Py_SETREF(tstate->previous_executor, executor);
|
||
|
return target;
|
||
|
}
|