Skip to content

Commit

Permalink
pythonGH-107724: Fix the signature of PY_THROW callback functions. (p…
Browse files Browse the repository at this point in the history
  • Loading branch information
markshannon committed Aug 9, 2023
1 parent 2fb484e commit 52fbcf6
Show file tree
Hide file tree
Showing 6 changed files with 39 additions and 18 deletions.
4 changes: 0 additions & 4 deletions Include/internal/pycore_instruments.h
Expand Up @@ -90,10 +90,6 @@ extern int
_Py_call_instrumentation_2args(PyThreadState *tstate, int event,
_PyInterpreterFrame *frame, _Py_CODEUNIT *instr, PyObject *arg0, PyObject *arg1);

extern void
_Py_call_instrumentation_exc0(PyThreadState *tstate, int event,
_PyInterpreterFrame *frame, _Py_CODEUNIT *instr);

extern void
_Py_call_instrumentation_exc2(PyThreadState *tstate, int event,
_PyInterpreterFrame *frame, _Py_CODEUNIT *instr, PyObject *arg0, PyObject *arg1);
Expand Down
32 changes: 32 additions & 0 deletions Lib/test/test_monitoring.py
Expand Up @@ -743,6 +743,13 @@ class ExceptionHandledRecorder(ExceptionRecorder):
def __call__(self, code, offset, exc):
self.events.append(("handled", type(exc)))

class ThrowRecorder(ExceptionRecorder):

event_type = E.PY_THROW

def __call__(self, code, offset, exc):
self.events.append(("throw", type(exc)))

class ExceptionMonitoringTest(CheckEvents):


Expand Down Expand Up @@ -888,6 +895,31 @@ async def async_loop():
func,
recorders = self.exception_recorders)

def test_throw(self):

def gen():
yield 1
yield 2

def func():
try:
g = gen()
next(g)
g.throw(IndexError)
except IndexError:
pass

self.check_balanced(
func,
recorders = self.exception_recorders)

events = self.get_events(
func,
TEST_TOOL,
self.exception_recorders + (ThrowRecorder,)
)
self.assertEqual(events[0], ("throw", IndexError))

class LineRecorder:

event_type = E.LINE
Expand Down
@@ -0,0 +1,3 @@
In pre-release versions of 3.12, up to rc1, the sys.monitoring callback
function for the ``PY_THROW`` event was missing the third, exception
argument. That is now fixed.
2 changes: 1 addition & 1 deletion Python/ceval.c
Expand Up @@ -2039,7 +2039,7 @@ monitor_throw(PyThreadState *tstate,
if (no_tools_for_event(tstate, frame, PY_MONITORING_EVENT_PY_THROW)) {
return;
}
_Py_call_instrumentation_exc0(tstate, PY_MONITORING_EVENT_PY_THROW, frame, instr);
do_monitor_exc(tstate, frame, instr, PY_MONITORING_EVENT_PY_THROW);
}

void
Expand Down
10 changes: 0 additions & 10 deletions Python/instrumentation.c
Expand Up @@ -1081,16 +1081,6 @@ call_instrumentation_vector_protected(
assert(_PyErr_Occurred(tstate));
}

void
_Py_call_instrumentation_exc0(
PyThreadState *tstate, int event,
_PyInterpreterFrame *frame, _Py_CODEUNIT *instr)
{
assert(_PyErr_Occurred(tstate));
PyObject *args[3] = { NULL, NULL, NULL };
call_instrumentation_vector_protected(tstate, event, frame, instr, 2, args);
}

void
_Py_call_instrumentation_exc2(
PyThreadState *tstate, int event,
Expand Down
6 changes: 3 additions & 3 deletions Python/legacy_tracing.c
Expand Up @@ -163,7 +163,7 @@ sys_trace_func2(
}

static PyObject *
sys_trace_unwind(
sys_trace_func3(
_PyLegacyEventHandler *self, PyObject *const *args,
size_t nargsf, PyObject *kwnames
) {
Expand Down Expand Up @@ -445,7 +445,7 @@ _PyEval_SetTrace(PyThreadState *tstate, Py_tracefunc func, PyObject *arg)
return -1;
}
if (set_callbacks(PY_MONITORING_SYS_TRACE_ID,
(vectorcallfunc)sys_trace_func2, PyTrace_CALL,
(vectorcallfunc)sys_trace_func3, PyTrace_CALL,
PY_MONITORING_EVENT_PY_THROW, -1)) {
return -1;
}
Expand All @@ -470,7 +470,7 @@ _PyEval_SetTrace(PyThreadState *tstate, Py_tracefunc func, PyObject *arg)
return -1;
}
if (set_callbacks(PY_MONITORING_SYS_TRACE_ID,
(vectorcallfunc)sys_trace_unwind, PyTrace_RETURN,
(vectorcallfunc)sys_trace_func3, PyTrace_RETURN,
PY_MONITORING_EVENT_PY_UNWIND, -1)) {
return -1;
}
Expand Down

0 comments on commit 52fbcf6

Please sign in to comment.