| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | /* Support for legacy tracing on top of PEP 669 instrumentation
 | 
					
						
							|  |  |  |  * Provides callables to forward PEP 669 events to legacy events. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2024-10-18 09:26:08 -06:00
										 |  |  | #include "pycore_audit.h"         // _PySys_Audit()
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | #include "pycore_ceval.h"         // export _PyEval_SetProfile()
 | 
					
						
							| 
									
										
										
										
											2025-03-19 18:17:44 +01:00
										 |  |  | #include "pycore_frame.h"         // PyFrameObject members
 | 
					
						
							| 
									
										
										
										
											2025-03-20 12:35:23 +01:00
										 |  |  | #include "pycore_interpframe.h"   // _PyFrame_GetCode()
 | 
					
						
							| 
									
										
										
										
											2023-09-06 15:56:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "opcode.h"
 | 
					
						
							|  |  |  | #include <stddef.h>
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-20 12:35:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | typedef struct _PyLegacyEventHandler { | 
					
						
							|  |  |  |     PyObject_HEAD | 
					
						
							|  |  |  |     vectorcallfunc vectorcall; | 
					
						
							|  |  |  |     int event; | 
					
						
							|  |  |  | } _PyLegacyEventHandler; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  | #define _PyLegacyEventHandler_CAST(op)  ((_PyLegacyEventHandler *)(op))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  | #ifdef Py_GIL_DISABLED
 | 
					
						
							|  |  |  | #define LOCK_SETUP()    PyMutex_Lock(&_PyRuntime.ceval.sys_trace_profile_mutex);
 | 
					
						
							|  |  |  | #define UNLOCK_SETUP()  PyMutex_Unlock(&_PyRuntime.ceval.sys_trace_profile_mutex);
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define LOCK_SETUP()
 | 
					
						
							|  |  |  | #define UNLOCK_SETUP()
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | /* The Py_tracefunc function expects the following arguments:
 | 
					
						
							|  |  |  |  *   obj: the trace object (PyObject *) | 
					
						
							|  |  |  |  *   frame: the current frame (PyFrameObject *) | 
					
						
							|  |  |  |  *   kind: the kind of event, see PyTrace_XXX #defines (int) | 
					
						
							|  |  |  |  *   arg: The arg (a PyObject *) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | call_profile_func(_PyLegacyEventHandler *self, PyObject *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyThreadState *tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     if (tstate->c_profilefunc == NULL) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyFrameObject *frame = PyEval_GetFrame(); | 
					
						
							|  |  |  |     if (frame == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_SystemError, | 
					
						
							|  |  |  |                         "Missing frame when calling profile function."); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_INCREF(frame); | 
					
						
							|  |  |  |     int err = tstate->c_profilefunc(tstate->c_profileobj, frame, self->event, arg); | 
					
						
							|  |  |  |     Py_DECREF(frame); | 
					
						
							|  |  |  |     if (err) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  | sys_profile_start( | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 2); | 
					
						
							|  |  |  |     return call_profile_func(self, Py_None); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  | sys_profile_throw( | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     return call_profile_func(self, Py_None); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | sys_profile_return( | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     return call_profile_func(self, args[2]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | sys_profile_unwind( | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |      _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |    return call_profile_func(self, NULL); | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | sys_profile_call_or_return( | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     PyObject *op, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(op); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 4); | 
					
						
							|  |  |  |     PyObject *callable = args[2]; | 
					
						
							|  |  |  |     if (PyCFunction_Check(callable)) { | 
					
						
							|  |  |  |         return call_profile_func(self, callable); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (Py_TYPE(callable) == &PyMethodDescr_Type) { | 
					
						
							|  |  |  |         PyObject *self_arg = args[3]; | 
					
						
							|  |  |  |         /* For backwards compatibility need to
 | 
					
						
							|  |  |  |          * convert to builtin method */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* If no arg, skip */ | 
					
						
							|  |  |  |         if (self_arg == &_PyInstrumentation_MISSING) { | 
					
						
							|  |  |  |             Py_RETURN_NONE; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyObject *meth = Py_TYPE(callable)->tp_descr_get( | 
					
						
							|  |  |  |             callable, self_arg, (PyObject*)Py_TYPE(self_arg)); | 
					
						
							|  |  |  |         if (meth == NULL) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyObject *res =  call_profile_func(self, meth); | 
					
						
							|  |  |  |         Py_DECREF(meth); | 
					
						
							|  |  |  |         return res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  | int | 
					
						
							|  |  |  | _PyEval_SetOpcodeTrace( | 
					
						
							|  |  |  |     PyFrameObject *frame, | 
					
						
							|  |  |  |     bool enable | 
					
						
							|  |  |  | ) { | 
					
						
							|  |  |  |     assert(frame != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-12 12:37:06 -04:00
										 |  |  |     PyCodeObject *code = _PyFrame_GetCode(frame->f_frame); | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  |     _PyMonitoringEventSet events = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (_PyMonitoring_GetLocalEvents(code, PY_MONITORING_SYS_TRACE_ID, &events) < 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (enable) { | 
					
						
							|  |  |  |         if (events & (1 << PY_MONITORING_EVENT_INSTRUCTION)) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         events |= (1 << PY_MONITORING_EVENT_INSTRUCTION); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (!(events & (1 << PY_MONITORING_EVENT_INSTRUCTION))) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         events &= (~(1 << PY_MONITORING_EVENT_INSTRUCTION)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return _PyMonitoring_SetLocalEvents(code, PY_MONITORING_SYS_TRACE_ID, events); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | call_trace_func(_PyLegacyEventHandler *self, PyObject *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyThreadState *tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     if (tstate->c_tracefunc == NULL) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyFrameObject *frame = PyEval_GetFrame(); | 
					
						
							|  |  |  |     if (frame == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_SystemError, | 
					
						
							|  |  |  |                         "Missing frame when calling trace function."); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  |     if (frame->f_trace_opcodes) { | 
					
						
							|  |  |  |         if (_PyEval_SetOpcodeTrace(frame, true) != 0) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     Py_INCREF(frame); | 
					
						
							|  |  |  |     int err = tstate->c_tracefunc(tstate->c_traceobj, frame, self->event, arg); | 
					
						
							| 
									
										
										
										
											2024-04-29 01:54:52 -07:00
										 |  |  |     frame->f_lineno = 0; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     Py_DECREF(frame); | 
					
						
							|  |  |  |     if (err) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_exception_func( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     PyObject *exc = args[2]; | 
					
						
							|  |  |  |     assert(PyExceptionInstance_Check(exc)); | 
					
						
							|  |  |  |     PyObject *type = (PyObject *)Py_TYPE(exc); | 
					
						
							|  |  |  |     PyObject *tb = PyException_GetTraceback(exc); | 
					
						
							|  |  |  |     if (tb == NULL) { | 
					
						
							|  |  |  |         tb = Py_NewRef(Py_None); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyObject *tuple = PyTuple_Pack(3, type, exc, tb); | 
					
						
							|  |  |  |     Py_DECREF(tb); | 
					
						
							|  |  |  |     if (tuple == NULL) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyObject *res = call_trace_func(self, tuple); | 
					
						
							|  |  |  |     Py_DECREF(tuple); | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  | sys_trace_start( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 2); | 
					
						
							|  |  |  |     return call_trace_func(self, Py_None); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  | static PyObject * | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  | sys_trace_throw( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-07-27 15:47:33 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     return call_trace_func(self, Py_None); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_unwind( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-11-02 09:38:08 -07:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     return call_trace_func(self, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_return( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(!PyErr_Occurred()); | 
					
						
							|  |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     assert(PyCode_Check(args[0])); | 
					
						
							|  |  |  |     PyObject *val = args[2]; | 
					
						
							|  |  |  |     PyObject *res = call_trace_func(self, val); | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_yield( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							|  |  |  |     return call_trace_func(self, args[2]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_instruction_func( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 2); | 
					
						
							|  |  |  |     PyFrameObject *frame = PyEval_GetFrame(); | 
					
						
							|  |  |  |     if (frame == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_SystemError, | 
					
						
							|  |  |  |                         "Missing frame when calling trace function."); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  |     PyThreadState *tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     if (!tstate->c_tracefunc || !frame->f_trace_opcodes) { | 
					
						
							|  |  |  |         if (_PyEval_SetOpcodeTrace(frame, false) != 0) { | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_INCREF(frame); | 
					
						
							|  |  |  |     int err = tstate->c_tracefunc(tstate->c_traceobj, frame, self->event, Py_None); | 
					
						
							|  |  |  |     frame->f_lineno = 0; | 
					
						
							|  |  |  |     Py_DECREF(frame); | 
					
						
							|  |  |  |     if (err) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | trace_line( | 
					
						
							|  |  |  |     PyThreadState *tstate, _PyLegacyEventHandler *self, | 
					
						
							|  |  |  |     PyFrameObject *frame, int line | 
					
						
							|  |  |  | ) { | 
					
						
							|  |  |  |     if (!frame->f_trace_lines) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (line < 0) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_INCREF(frame); | 
					
						
							|  |  |  |     frame->f_lineno = line; | 
					
						
							|  |  |  |     int err = tstate->c_tracefunc(tstate->c_traceobj, frame, self->event, Py_None); | 
					
						
							|  |  |  |     frame->f_lineno = 0; | 
					
						
							|  |  |  |     Py_DECREF(frame); | 
					
						
							|  |  |  |     if (err) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_RETURN_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_line_func( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     PyThreadState *tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     if (tstate->c_tracefunc == NULL) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 2); | 
					
						
							| 
									
										
										
										
											2023-08-25 01:01:30 +02:00
										 |  |  |     int line = PyLong_AsInt(args[1]); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(line >= 0); | 
					
						
							|  |  |  |     PyFrameObject *frame = PyEval_GetFrame(); | 
					
						
							|  |  |  |     if (frame == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_SystemError, | 
					
						
							|  |  |  |                         "Missing frame when calling trace function."); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-06-14 13:46:37 +01:00
										 |  |  |     assert(args[0] == (PyObject *)_PyFrame_GetCode(frame->f_frame)); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     return trace_line(tstate, self, frame, line); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-12 12:21:20 +01:00
										 |  |  | /* sys.settrace generates line events for all backward
 | 
					
						
							|  |  |  |  * edges, even if on the same line. | 
					
						
							|  |  |  |  * Handle that case here */ | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | static PyObject * | 
					
						
							|  |  |  | sys_trace_jump_func( | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |     PyObject *callable, PyObject *const *args, | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     size_t nargsf, PyObject *kwnames | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |     _PyLegacyEventHandler *self = _PyLegacyEventHandler_CAST(callable); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(kwnames == NULL); | 
					
						
							|  |  |  |     PyThreadState *tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     if (tstate->c_tracefunc == NULL) { | 
					
						
							|  |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     assert(PyVectorcall_NARGS(nargsf) == 3); | 
					
						
							| 
									
										
										
										
											2023-08-25 01:01:30 +02:00
										 |  |  |     int from = PyLong_AsInt(args[1])/sizeof(_Py_CODEUNIT); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(from >= 0); | 
					
						
							| 
									
										
										
										
											2023-08-25 01:01:30 +02:00
										 |  |  |     int to = PyLong_AsInt(args[2])/sizeof(_Py_CODEUNIT); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     assert(to >= 0); | 
					
						
							| 
									
										
										
										
											2023-05-12 12:21:20 +01:00
										 |  |  |     if (to > from) { | 
					
						
							|  |  |  |         /* Forward jump */ | 
					
						
							|  |  |  |         return &_PyInstrumentation_DISABLE; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     PyCodeObject *code = (PyCodeObject *)args[0]; | 
					
						
							|  |  |  |     assert(PyCode_Check(code)); | 
					
						
							|  |  |  |     /* We can call _Py_Instrumentation_GetLine because we always set
 | 
					
						
							|  |  |  |     * line events for tracing */ | 
					
						
							|  |  |  |     int to_line = _Py_Instrumentation_GetLine(code, to); | 
					
						
							| 
									
										
										
										
											2023-05-12 12:21:20 +01:00
										 |  |  |     int from_line = _Py_Instrumentation_GetLine(code, from); | 
					
						
							|  |  |  |     if (to_line != from_line) { | 
					
						
							|  |  |  |         /* Will be handled by target INSTRUMENTED_LINE */ | 
					
						
							|  |  |  |         return &_PyInstrumentation_DISABLE; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     PyFrameObject *frame = PyEval_GetFrame(); | 
					
						
							| 
									
										
										
										
											2023-05-12 12:21:20 +01:00
										 |  |  |     if (frame == NULL) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_SystemError, | 
					
						
							|  |  |  |                         "Missing frame when calling trace function."); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!frame->f_trace_lines) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         Py_RETURN_NONE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-12 12:21:20 +01:00
										 |  |  |     return trace_line(tstate, self, frame, to_line); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyTypeObject _PyLegacyEventHandler_Type = { | 
					
						
							| 
									
										
										
										
											2023-04-22 15:39:37 -04:00
										 |  |  |     PyVarObject_HEAD_INIT(&PyType_Type, 0) | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     "sys.legacy_event_handler", | 
					
						
							|  |  |  |     sizeof(_PyLegacyEventHandler), | 
					
						
							|  |  |  |     .tp_vectorcall_offset = offsetof(_PyLegacyEventHandler, vectorcall), | 
					
						
							|  |  |  |     .tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | | 
					
						
							|  |  |  |         Py_TPFLAGS_HAVE_VECTORCALL | Py_TPFLAGS_DISALLOW_INSTANTIATION, | 
					
						
							|  |  |  |     .tp_call = PyVectorcall_Call, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | set_callbacks(int tool, vectorcallfunc vectorcall, int legacy_event, int event1, int event2) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyLegacyEventHandler *callback = | 
					
						
							|  |  |  |         PyObject_NEW(_PyLegacyEventHandler, &_PyLegacyEventHandler_Type); | 
					
						
							|  |  |  |     if (callback == NULL) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     callback->vectorcall = vectorcall; | 
					
						
							|  |  |  |     callback->event = legacy_event; | 
					
						
							|  |  |  |     Py_XDECREF(_PyMonitoring_RegisterCallback(tool, event1, (PyObject *)callback)); | 
					
						
							|  |  |  |     if (event2 >= 0) { | 
					
						
							|  |  |  |         Py_XDECREF(_PyMonitoring_RegisterCallback(tool, event2, (PyObject *)callback)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_DECREF(callback); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef NDEBUG
 | 
					
						
							|  |  |  | /* Ensure that tstate is valid: sanity check for PyEval_AcquireThread() and
 | 
					
						
							|  |  |  |    PyEval_RestoreThread(). Detect if tstate memory was freed. It can happen | 
					
						
							|  |  |  |    when a thread continues to run after Python finalization, especially | 
					
						
							|  |  |  |    daemon threads. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | is_tstate_valid(PyThreadState *tstate) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(!_PyMem_IsPtrFreed(tstate)); | 
					
						
							|  |  |  |     assert(!_PyMem_IsPtrFreed(tstate->interp)); | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  | static Py_ssize_t | 
					
						
							|  |  |  | setup_profile(PyThreadState *tstate, Py_tracefunc func, PyObject *arg, PyObject **old_profileobj) | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     *old_profileobj = NULL; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     /* Setup PEP 669 monitoring callbacks and events. */ | 
					
						
							|  |  |  |     if (!tstate->interp->sys_profile_initialized) { | 
					
						
							|  |  |  |         tstate->interp->sys_profile_initialized = true; | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_start, PyTrace_CALL, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_START, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_RESUME)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-10-09 01:38:45 -07:00
										 |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_throw, PyTrace_CALL, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_THROW, -1)) { | 
					
						
							| 
									
										
										
										
											2023-10-09 01:38:45 -07:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_return, PyTrace_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_YIELD)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_unwind, PyTrace_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_UNWIND, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_call_or_return, PyTrace_C_CALL, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_CALL, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_call_or_return, PyTrace_C_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_C_RETURN, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_PROFILE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-24 11:00:32 +01:00
										 |  |  |                           sys_profile_call_or_return, PyTrace_C_EXCEPTION, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_C_RAISE, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int delta = (func != NULL) - (tstate->c_profilefunc != NULL); | 
					
						
							|  |  |  |     tstate->c_profilefunc = func; | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     *old_profileobj = tstate->c_profileobj; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     tstate->c_profileobj = Py_XNewRef(arg); | 
					
						
							|  |  |  |     tstate->interp->sys_profiling_threads += delta; | 
					
						
							|  |  |  |     assert(tstate->interp->sys_profiling_threads >= 0); | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     return tstate->interp->sys_profiling_threads; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  | _PyEval_SetProfile(PyThreadState *tstate, Py_tracefunc func, PyObject *arg) | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     assert(is_tstate_valid(tstate)); | 
					
						
							| 
									
										
										
										
											2025-01-20 06:34:35 -05:00
										 |  |  |     /* The caller must hold a thread state */ | 
					
						
							|  |  |  |     _Py_AssertHoldsTstate(); | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Call _PySys_Audit() in the context of the current thread state,
 | 
					
						
							|  |  |  |        even if tstate is not the current thread state. */ | 
					
						
							|  |  |  |     PyThreadState *current_tstate = _PyThreadState_GET(); | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     if (_PySys_Audit(current_tstate, "sys.setprofile", NULL) < 0) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     // needs to be decref'd outside of the lock
 | 
					
						
							|  |  |  |     PyObject *old_profileobj; | 
					
						
							|  |  |  |     LOCK_SETUP(); | 
					
						
							|  |  |  |     Py_ssize_t profiling_threads = setup_profile(tstate, func, arg, &old_profileobj); | 
					
						
							|  |  |  |     UNLOCK_SETUP(); | 
					
						
							|  |  |  |     Py_XDECREF(old_profileobj); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     uint32_t events = 0; | 
					
						
							|  |  |  |     if (profiling_threads) { | 
					
						
							|  |  |  |         events = | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_PY_START) | (1 << PY_MONITORING_EVENT_PY_RESUME) | | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_PY_RETURN) | (1 << PY_MONITORING_EVENT_PY_YIELD) | | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_CALL) | (1 << PY_MONITORING_EVENT_PY_UNWIND) | | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_PY_THROW); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return _PyMonitoring_SetEvents(PY_MONITORING_SYS_PROFILE_ID, events); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Py_ssize_t | 
					
						
							|  |  |  | setup_tracing(PyThreadState *tstate, Py_tracefunc func, PyObject *arg, PyObject **old_traceobj) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     *old_traceobj = NULL; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     /* Setup PEP 669 monitoring callbacks and events. */ | 
					
						
							|  |  |  |     if (!tstate->interp->sys_trace_initialized) { | 
					
						
							|  |  |  |         tstate->interp->sys_trace_initialized = true; | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_start, PyTrace_CALL, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_START, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_RESUME)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_throw, PyTrace_CALL, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_THROW, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_return, PyTrace_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_RETURN, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_yield, PyTrace_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_YIELD, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_exception_func, PyTrace_EXCEPTION, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_RAISE, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_STOP_ITERATION)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_line_func, PyTrace_LINE, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_LINE, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_unwind, PyTrace_RETURN, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_PY_UNWIND, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_jump_func, PyTrace_LINE, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_JUMP, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (set_callbacks(PY_MONITORING_SYS_TRACE_ID, | 
					
						
							| 
									
										
										
										
											2025-03-19 16:54:41 +01:00
										 |  |  |                           sys_trace_instruction_func, PyTrace_OPCODE, | 
					
						
							|  |  |  |                           PY_MONITORING_EVENT_INSTRUCTION, -1)) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int delta = (func != NULL) - (tstate->c_tracefunc != NULL); | 
					
						
							|  |  |  |     tstate->c_tracefunc = func; | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     *old_traceobj = tstate->c_traceobj; | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     tstate->c_traceobj = Py_XNewRef(arg); | 
					
						
							|  |  |  |     tstate->interp->sys_tracing_threads += delta; | 
					
						
							|  |  |  |     assert(tstate->interp->sys_tracing_threads >= 0); | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     return tstate->interp->sys_tracing_threads; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | _PyEval_SetTrace(PyThreadState *tstate, Py_tracefunc func, PyObject *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     assert(is_tstate_valid(tstate)); | 
					
						
							| 
									
										
										
										
											2025-01-20 06:34:35 -05:00
										 |  |  |     /* The caller must hold a thread state */ | 
					
						
							|  |  |  |     _Py_AssertHoldsTstate(); | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Call _PySys_Audit() in the context of the current thread state,
 | 
					
						
							|  |  |  |        even if tstate is not the current thread state. */ | 
					
						
							|  |  |  |     PyThreadState *current_tstate = _PyThreadState_GET(); | 
					
						
							|  |  |  |     if (_PySys_Audit(current_tstate, "sys.settrace", NULL) < 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // needs to be decref'd outside of the lock
 | 
					
						
							|  |  |  |     PyObject *old_traceobj; | 
					
						
							|  |  |  |     LOCK_SETUP(); | 
					
						
							| 
									
										
										
										
											2025-03-21 14:22:37 -04:00
										 |  |  |     assert(tstate->interp->sys_tracing_threads >= 0); | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     Py_ssize_t tracing_threads = setup_tracing(tstate, func, arg, &old_traceobj); | 
					
						
							|  |  |  |     UNLOCK_SETUP(); | 
					
						
							|  |  |  |     Py_XDECREF(old_traceobj); | 
					
						
							|  |  |  |     if (tracing_threads < 0) { | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     uint32_t events = 0; | 
					
						
							| 
									
										
										
										
											2024-04-19 14:47:42 -07:00
										 |  |  |     if (tracing_threads) { | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         events = | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_PY_START) | (1 << PY_MONITORING_EVENT_PY_RESUME) | | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_PY_RETURN) | (1 << PY_MONITORING_EVENT_PY_YIELD) | | 
					
						
							|  |  |  |             (1 << PY_MONITORING_EVENT_RAISE) | (1 << PY_MONITORING_EVENT_LINE) | | 
					
						
							| 
									
										
										
										
											2024-04-26 09:01:44 -07:00
										 |  |  |             (1 << PY_MONITORING_EVENT_JUMP) | | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |             (1 << PY_MONITORING_EVENT_PY_UNWIND) | (1 << PY_MONITORING_EVENT_PY_THROW) | | 
					
						
							| 
									
										
										
										
											2024-04-26 09:01:44 -07:00
										 |  |  |             (1 << PY_MONITORING_EVENT_STOP_ITERATION); | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         PyFrameObject* frame = PyEval_GetFrame(); | 
					
						
							| 
									
										
										
										
											2024-07-15 16:11:02 -07:00
										 |  |  |         if (frame && frame->f_trace_opcodes) { | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  |             int ret = _PyEval_SetOpcodeTrace(frame, true); | 
					
						
							|  |  |  |             if (ret != 0) { | 
					
						
							|  |  |  |                 return ret; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-03 09:39:50 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 12:04:55 +01:00
										 |  |  |     return _PyMonitoring_SetEvents(PY_MONITORING_SYS_TRACE_ID, events); | 
					
						
							|  |  |  | } |