Spaces:
Build error
Build error
| // Exception raising code | |
| // | |
| // Exceptions are raised by __Pyx_Raise() and stored as plain | |
| // type/value/tb in PyThreadState->curexc_*. When being caught by an | |
| // 'except' statement, curexc_* is moved over to exc_* by | |
| // __Pyx_GetException() | |
| /////////////// AssertionsEnabled.init /////////////// | |
| if (likely(__Pyx_init_assertions_enabled() == 0)); else | |
| // error propagation code is appended automatically | |
| /////////////// AssertionsEnabled.proto /////////////// | |
| // Py_OptimizeFlag is deprecated in Py3.12+ and not available in the Limited API. | |
| static int __pyx_assertions_enabled_flag; | |
| static int __Pyx_init_assertions_enabled(void) { | |
| PyObject *builtins, *debug, *debug_str; | |
| int flag; | |
| builtins = PyEval_GetBuiltins(); | |
| if (!builtins) goto bad; | |
| debug_str = PyUnicode_FromStringAndSize("__debug__", 9); | |
| if (!debug_str) goto bad; | |
| debug = PyObject_GetItem(builtins, debug_str); | |
| Py_DECREF(debug_str); | |
| if (!debug) goto bad; | |
| flag = PyObject_IsTrue(debug); | |
| Py_DECREF(debug); | |
| if (flag == -1) goto bad; | |
| __pyx_assertions_enabled_flag = flag; | |
| return 0; | |
| bad: | |
| __pyx_assertions_enabled_flag = 1; | |
| // We (rarely) may not have an exception set, but the calling code will call PyErr_Occurred() either way. | |
| return -1; | |
| } | |
| /////////////// ErrOccurredWithGIL.proto /////////////// | |
| static CYTHON_INLINE int __Pyx_ErrOccurredWithGIL(void); /* proto */ | |
| /////////////// ErrOccurredWithGIL /////////////// | |
| static CYTHON_INLINE int __Pyx_ErrOccurredWithGIL(void) { | |
| int err; | |
| PyGILState_STATE _save = PyGILState_Ensure(); | |
| err = !!PyErr_Occurred(); | |
| PyGILState_Release(_save); | |
| return err; | |
| } | |
| /////////////// PyThreadStateGet.proto /////////////// | |
| //@substitute: naming | |
| // !CYTHON_FAST_THREAD_STATE | |
| /////////////// PyErrExceptionMatches.proto /////////////// | |
| //@substitute: naming | |
| static CYTHON_INLINE int __Pyx_PyErr_ExceptionMatchesInState(PyThreadState* tstate, PyObject* err); | |
| /////////////// PyErrExceptionMatches /////////////// | |
| static int __Pyx_PyErr_ExceptionMatchesTuple(PyObject *exc_type, PyObject *tuple) { | |
| Py_ssize_t i, n; | |
| n = PyTuple_GET_SIZE(tuple); | |
| // the tighter subtype checking in Py3 allows faster out-of-order comparison | |
| for (i=0; i<n; i++) { | |
| if (exc_type == PyTuple_GET_ITEM(tuple, i)) return 1; | |
| } | |
| for (i=0; i<n; i++) { | |
| if (__Pyx_PyErr_GivenExceptionMatches(exc_type, PyTuple_GET_ITEM(tuple, i))) return 1; | |
| } | |
| return 0; | |
| } | |
| static CYTHON_INLINE int __Pyx_PyErr_ExceptionMatchesInState(PyThreadState* tstate, PyObject* err) { | |
| int result; | |
| PyObject *exc_type; | |
| PyObject *current_exception = tstate->current_exception; | |
| if (unlikely(!current_exception)) return 0; | |
| exc_type = (PyObject*) Py_TYPE(current_exception); | |
| if (exc_type == err) return 1; | |
| exc_type = tstate->curexc_type; | |
| if (exc_type == err) return 1; | |
| if (unlikely(!exc_type)) return 0; | |
| Py_INCREF(exc_type); | |
| if (unlikely(PyTuple_Check(err))) { | |
| result = __Pyx_PyErr_ExceptionMatchesTuple(exc_type, err); | |
| } else { | |
| result = __Pyx_PyErr_GivenExceptionMatches(exc_type, err); | |
| } | |
| Py_DECREF(exc_type); | |
| return result; | |
| } | |
| /////////////// PyErrFetchRestore.proto /////////////// | |
| //@substitute: naming | |
| //@requires: PyThreadStateGet | |
| static CYTHON_INLINE void __Pyx_ErrRestoreInState(PyThreadState *tstate, PyObject *type, PyObject *value, PyObject *tb); /*proto*/ | |
| static CYTHON_INLINE void __Pyx_ErrFetchInState(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb); /*proto*/ | |
| /////////////// PyErrFetchRestore /////////////// | |
| static CYTHON_INLINE void __Pyx_ErrRestoreInState(PyThreadState *tstate, PyObject *type, PyObject *value, PyObject *tb) { | |
| PyObject *tmp_value; | |
| assert(type == NULL || (value != NULL && type == (PyObject*) Py_TYPE(value))); | |
| if (value) { | |
| if (unlikely(((PyBaseExceptionObject*) value)->traceback != tb)) | |
| // If this fails, we may lose the traceback but still set the expected exception below. | |
| PyException_SetTraceback(value, tb); | |
| } | |
| tmp_value = tstate->current_exception; | |
| tstate->current_exception = value; | |
| Py_XDECREF(tmp_value); | |
| Py_XDECREF(type); | |
| Py_XDECREF(tb); | |
| PyObject *tmp_type, *tmp_value, *tmp_tb; | |
| tmp_type = tstate->curexc_type; | |
| tmp_value = tstate->curexc_value; | |
| tmp_tb = tstate->curexc_traceback; | |
| tstate->curexc_type = type; | |
| tstate->curexc_value = value; | |
| tstate->curexc_traceback = tb; | |
| Py_XDECREF(tmp_type); | |
| Py_XDECREF(tmp_value); | |
| Py_XDECREF(tmp_tb); | |
| } | |
| static CYTHON_INLINE void __Pyx_ErrFetchInState(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb) { | |
| PyObject* exc_value; | |
| exc_value = tstate->current_exception; | |
| tstate->current_exception = 0; | |
| *value = exc_value; | |
| *type = NULL; | |
| *tb = NULL; | |
| if (exc_value) { | |
| *type = (PyObject*) Py_TYPE(exc_value); | |
| Py_INCREF(*type); | |
| *tb = ((PyBaseExceptionObject*) exc_value)->traceback; | |
| Py_XINCREF(*tb); | |
| *tb = PyException_GetTraceback(exc_value); | |
| } | |
| *type = tstate->curexc_type; | |
| *value = tstate->curexc_value; | |
| *tb = tstate->curexc_traceback; | |
| tstate->curexc_type = 0; | |
| tstate->curexc_value = 0; | |
| tstate->curexc_traceback = 0; | |
| } | |
| /////////////// RaiseException.proto /////////////// | |
| static void __Pyx_Raise(PyObject *type, PyObject *value, PyObject *tb, PyObject *cause); /*proto*/ | |
| /////////////// RaiseException /////////////// | |
| //@requires: PyErrFetchRestore | |
| //@requires: PyThreadStateGet | |
| // The following function is based on do_raise() from ceval.c. There | |
| // are separate versions for Python2 and Python3 as exception handling | |
| // has changed quite a lot between the two versions. | |
| static void __Pyx_Raise(PyObject *type, PyObject *value, PyObject *tb, PyObject *cause) { | |
| __Pyx_PyThreadState_declare | |
| CYTHON_UNUSED_VAR(cause); | |
| /* 'cause' is only used in Py3 */ | |
| Py_XINCREF(type); | |
| if (!value || value == Py_None) | |
| value = NULL; | |
| else | |
| Py_INCREF(value); | |
| if (!tb || tb == Py_None) | |
| tb = NULL; | |
| else { | |
| Py_INCREF(tb); | |
| if (!PyTraceBack_Check(tb)) { | |
| PyErr_SetString(PyExc_TypeError, | |
| "raise: arg 3 must be a traceback or None"); | |
| goto raise_error; | |
| } | |
| } | |
| if (PyType_Check(type)) { | |
| /* instantiate the type now (we don't know when and how it will be caught) */ | |
| /* PyPy can't handle value == NULL */ | |
| if (!value) { | |
| Py_INCREF(Py_None); | |
| value = Py_None; | |
| } | |
| PyErr_NormalizeException(&type, &value, &tb); | |
| } else { | |
| /* Raising an instance. The value should be a dummy. */ | |
| if (value) { | |
| PyErr_SetString(PyExc_TypeError, | |
| "instance exception may not have a separate value"); | |
| goto raise_error; | |
| } | |
| /* Normalize to raise <class>, <instance> */ | |
| value = type; | |
| type = (PyObject*) Py_TYPE(type); | |
| Py_INCREF(type); | |
| if (!PyType_IsSubtype((PyTypeObject *)type, (PyTypeObject *)PyExc_BaseException)) { | |
| PyErr_SetString(PyExc_TypeError, | |
| "raise: exception class must be a subclass of BaseException"); | |
| goto raise_error; | |
| } | |
| } | |
| __Pyx_PyThreadState_assign | |
| __Pyx_ErrRestore(type, value, tb); | |
| return; | |
| raise_error: | |
| Py_XDECREF(value); | |
| Py_XDECREF(type); | |
| Py_XDECREF(tb); | |
| return; | |
| } | |
| static void __Pyx_Raise(PyObject *type, PyObject *value, PyObject *tb, PyObject *cause) { | |
| PyObject* owned_instance = NULL; | |
| if (tb == Py_None) { | |
| tb = 0; | |
| } else if (tb && !PyTraceBack_Check(tb)) { | |
| PyErr_SetString(PyExc_TypeError, | |
| "raise: arg 3 must be a traceback or None"); | |
| goto bad; | |
| } | |
| if (value == Py_None) | |
| value = 0; | |
| if (PyExceptionInstance_Check(type)) { | |
| if (value) { | |
| PyErr_SetString(PyExc_TypeError, | |
| "instance exception may not have a separate value"); | |
| goto bad; | |
| } | |
| value = type; | |
| type = (PyObject*) Py_TYPE(value); | |
| } else if (PyExceptionClass_Check(type)) { | |
| // make sure value is an exception instance of type | |
| PyObject *instance_class = NULL; | |
| if (value && PyExceptionInstance_Check(value)) { | |
| instance_class = (PyObject*) Py_TYPE(value); | |
| if (instance_class != type) { | |
| int is_subclass = PyObject_IsSubclass(instance_class, type); | |
| if (!is_subclass) { | |
| instance_class = NULL; | |
| } else if (unlikely(is_subclass == -1)) { | |
| // error on subclass test | |
| goto bad; | |
| } else { | |
| // believe the instance | |
| type = instance_class; | |
| } | |
| } | |
| } | |
| if (!instance_class) { | |
| // instantiate the type now (we don't know when and how it will be caught) | |
| // assuming that 'value' is an argument to the type's constructor | |
| // not using PyErr_NormalizeException() to avoid ref-counting problems | |
| PyObject *args; | |
| if (!value) | |
| args = PyTuple_New(0); | |
| else if (PyTuple_Check(value)) { | |
| Py_INCREF(value); | |
| args = value; | |
| } else | |
| args = PyTuple_Pack(1, value); | |
| if (!args) | |
| goto bad; | |
| owned_instance = PyObject_Call(type, args, NULL); | |
| Py_DECREF(args); | |
| if (!owned_instance) | |
| goto bad; | |
| value = owned_instance; | |
| if (!PyExceptionInstance_Check(value)) { | |
| PyErr_Format(PyExc_TypeError, | |
| "calling %R should have returned an instance of " | |
| "BaseException, not %R", | |
| type, Py_TYPE(value)); | |
| goto bad; | |
| } | |
| } | |
| } else { | |
| PyErr_SetString(PyExc_TypeError, | |
| "raise: exception class must be a subclass of BaseException"); | |
| goto bad; | |
| } | |
| if (cause) { | |
| PyObject *fixed_cause; | |
| if (cause == Py_None) { | |
| // raise ... from None | |
| fixed_cause = NULL; | |
| } else if (PyExceptionClass_Check(cause)) { | |
| fixed_cause = PyObject_CallObject(cause, NULL); | |
| if (fixed_cause == NULL) | |
| goto bad; | |
| } else if (PyExceptionInstance_Check(cause)) { | |
| fixed_cause = cause; | |
| Py_INCREF(fixed_cause); | |
| } else { | |
| PyErr_SetString(PyExc_TypeError, | |
| "exception causes must derive from " | |
| "BaseException"); | |
| goto bad; | |
| } | |
| PyException_SetCause(value, fixed_cause); | |
| } | |
| PyErr_SetObject(type, value); | |
| if (tb) { | |
| // If this fails, we just get a different exception, so ignore the return value. | |
| PyException_SetTraceback(value, tb); | |
| PyThreadState *tstate = __Pyx_PyThreadState_Current; | |
| PyObject* tmp_tb = tstate->curexc_traceback; | |
| if (tb != tmp_tb) { | |
| Py_INCREF(tb); | |
| tstate->curexc_traceback = tb; | |
| Py_XDECREF(tmp_tb); | |
| } | |
| PyObject *tmp_type, *tmp_value, *tmp_tb; | |
| PyErr_Fetch(&tmp_type, &tmp_value, &tmp_tb); | |
| Py_INCREF(tb); | |
| PyErr_Restore(tmp_type, tmp_value, tb); | |
| Py_XDECREF(tmp_tb); | |
| } | |
| bad: | |
| Py_XDECREF(owned_instance); | |
| return; | |
| } | |
| /////////////// GetTopmostException.proto /////////////// | |
| static _PyErr_StackItem * __Pyx_PyErr_GetTopmostException(PyThreadState *tstate); | |
| /////////////// GetTopmostException /////////////// | |
| // Copied from errors.c in CPython. | |
| static _PyErr_StackItem * | |
| __Pyx_PyErr_GetTopmostException(PyThreadState *tstate) | |
| { | |
| _PyErr_StackItem *exc_info = tstate->exc_info; | |
| while ((exc_info->exc_value == NULL || exc_info->exc_value == Py_None) && | |
| exc_info->previous_item != NULL) | |
| { | |
| exc_info = exc_info->previous_item; | |
| } | |
| return exc_info; | |
| } | |
| /////////////// GetException.proto /////////////// | |
| //@substitute: naming | |
| //@requires: PyThreadStateGet | |
| static int __Pyx__GetException(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb); /*proto*/ | |
| static int __Pyx_GetException(PyObject **type, PyObject **value, PyObject **tb); /*proto*/ | |
| /////////////// GetException /////////////// | |
| static int __Pyx__GetException(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb) | |
| static int __Pyx_GetException(PyObject **type, PyObject **value, PyObject **tb) | |
| { | |
| PyObject *local_type = NULL, *local_value, *local_tb = NULL; | |
| PyObject *tmp_type, *tmp_value, *tmp_tb; | |
| local_value = tstate->current_exception; | |
| tstate->current_exception = 0; | |
| if (likely(local_value)) { | |
| local_type = (PyObject*) Py_TYPE(local_value); | |
| Py_INCREF(local_type); | |
| local_tb = PyException_GetTraceback(local_value); | |
| } | |
| local_type = tstate->curexc_type; | |
| local_value = tstate->curexc_value; | |
| local_tb = tstate->curexc_traceback; | |
| tstate->curexc_type = 0; | |
| tstate->curexc_value = 0; | |
| tstate->curexc_traceback = 0; | |
| PyErr_Fetch(&local_type, &local_value, &local_tb); | |
| PyErr_NormalizeException(&local_type, &local_value, &local_tb); | |
| if (unlikely(tstate->current_exception)) | |
| if (unlikely(tstate->curexc_type)) | |
| if (unlikely(PyErr_Occurred())) | |
| goto bad; | |
| if (local_tb) { | |
| if (unlikely(PyException_SetTraceback(local_value, local_tb) < 0)) | |
| goto bad; | |
| } | |
| // traceback may be NULL for freshly raised exceptions | |
| Py_XINCREF(local_tb); | |
| // exception state may be temporarily empty in parallel loops (race condition) | |
| Py_XINCREF(local_type); | |
| Py_XINCREF(local_value); | |
| *type = local_type; | |
| *value = local_value; | |
| *tb = local_tb; | |
| { | |
| _PyErr_StackItem *exc_info = tstate->exc_info; | |
| tmp_value = exc_info->exc_value; | |
| exc_info->exc_value = local_value; | |
| tmp_type = NULL; | |
| tmp_tb = NULL; | |
| Py_XDECREF(local_type); | |
| Py_XDECREF(local_tb); | |
| tmp_type = exc_info->exc_type; | |
| tmp_value = exc_info->exc_value; | |
| tmp_tb = exc_info->exc_traceback; | |
| exc_info->exc_type = local_type; | |
| exc_info->exc_value = local_value; | |
| exc_info->exc_traceback = local_tb; | |
| } | |
| tmp_type = tstate->exc_type; | |
| tmp_value = tstate->exc_value; | |
| tmp_tb = tstate->exc_traceback; | |
| tstate->exc_type = local_type; | |
| tstate->exc_value = local_value; | |
| tstate->exc_traceback = local_tb; | |
| // Make sure tstate is in a consistent state when we XDECREF | |
| // these objects (DECREF may run arbitrary code). | |
| Py_XDECREF(tmp_type); | |
| Py_XDECREF(tmp_value); | |
| Py_XDECREF(tmp_tb); | |
| PyErr_SetExcInfo(local_type, local_value, local_tb); | |
| return 0; | |
| bad: | |
| *type = 0; | |
| *value = 0; | |
| *tb = 0; | |
| Py_XDECREF(local_type); | |
| Py_XDECREF(local_value); | |
| Py_XDECREF(local_tb); | |
| return -1; | |
| } | |
| /////////////// ReRaiseException.proto /////////////// | |
| static CYTHON_INLINE void __Pyx_ReraiseException(void); /*proto*/ | |
| /////////////// ReRaiseException /////////////// | |
| //@requires: GetTopmostException | |
| static CYTHON_INLINE void __Pyx_ReraiseException(void) { | |
| PyObject *type = NULL, *value = NULL, *tb = NULL; | |
| PyThreadState *tstate = PyThreadState_GET(); | |
| _PyErr_StackItem *exc_info = __Pyx_PyErr_GetTopmostException(tstate); | |
| value = exc_info->exc_value; | |
| if (unlikely(value == Py_None)) { | |
| value = NULL; | |
| } else if (value) { | |
| Py_INCREF(value); | |
| type = (PyObject*) Py_TYPE(value); | |
| Py_INCREF(type); | |
| tb = PyException_GetTraceback(value); | |
| } | |
| type = exc_info->exc_type; | |
| tb = exc_info->exc_traceback; | |
| Py_XINCREF(type); | |
| Py_XINCREF(value); | |
| Py_XINCREF(tb); | |
| type = tstate->exc_type; | |
| value = tstate->exc_value; | |
| tb = tstate->exc_traceback; | |
| Py_XINCREF(type); | |
| Py_XINCREF(value); | |
| Py_XINCREF(tb); | |
| PyErr_GetExcInfo(&type, &value, &tb); | |
| if (unlikely(!type || type == Py_None)) { | |
| Py_XDECREF(type); | |
| Py_XDECREF(value); | |
| Py_XDECREF(tb); | |
| // message copied from Py3 | |
| PyErr_SetString(PyExc_RuntimeError, | |
| "No active exception to reraise"); | |
| } else { | |
| PyErr_Restore(type, value, tb); | |
| } | |
| } | |
| /////////////// SaveResetException.proto /////////////// | |
| //@substitute: naming | |
| //@requires: PyThreadStateGet | |
| static CYTHON_INLINE void __Pyx__ExceptionSave(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb); /*proto*/ | |
| static CYTHON_INLINE void __Pyx__ExceptionReset(PyThreadState *tstate, PyObject *type, PyObject *value, PyObject *tb); /*proto*/ | |
| /////////////// SaveResetException /////////////// | |
| //@requires: GetTopmostException | |
| static CYTHON_INLINE void __Pyx__ExceptionSave(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb) { | |
| _PyErr_StackItem *exc_info = __Pyx_PyErr_GetTopmostException(tstate); | |
| PyObject *exc_value = exc_info->exc_value; | |
| if (exc_value == NULL || exc_value == Py_None) { | |
| *value = NULL; | |
| *type = NULL; | |
| *tb = NULL; | |
| } else { | |
| *value = exc_value; | |
| Py_INCREF(*value); | |
| *type = (PyObject*) Py_TYPE(exc_value); | |
| Py_INCREF(*type); | |
| *tb = PyException_GetTraceback(exc_value); | |
| } | |
| _PyErr_StackItem *exc_info = __Pyx_PyErr_GetTopmostException(tstate); | |
| *type = exc_info->exc_type; | |
| *value = exc_info->exc_value; | |
| *tb = exc_info->exc_traceback; | |
| Py_XINCREF(*type); | |
| Py_XINCREF(*value); | |
| Py_XINCREF(*tb); | |
| *type = tstate->exc_type; | |
| *value = tstate->exc_value; | |
| *tb = tstate->exc_traceback; | |
| Py_XINCREF(*type); | |
| Py_XINCREF(*value); | |
| Py_XINCREF(*tb); | |
| } | |
| static CYTHON_INLINE void __Pyx__ExceptionReset(PyThreadState *tstate, PyObject *type, PyObject *value, PyObject *tb) { | |
| _PyErr_StackItem *exc_info = tstate->exc_info; | |
| PyObject *tmp_value = exc_info->exc_value; | |
| exc_info->exc_value = value; | |
| Py_XDECREF(tmp_value); | |
| // TODO: avoid passing these at all | |
| Py_XDECREF(type); | |
| Py_XDECREF(tb); | |
| PyObject *tmp_type, *tmp_value, *tmp_tb; | |
| _PyErr_StackItem *exc_info = tstate->exc_info; | |
| tmp_type = exc_info->exc_type; | |
| tmp_value = exc_info->exc_value; | |
| tmp_tb = exc_info->exc_traceback; | |
| exc_info->exc_type = type; | |
| exc_info->exc_value = value; | |
| exc_info->exc_traceback = tb; | |
| tmp_type = tstate->exc_type; | |
| tmp_value = tstate->exc_value; | |
| tmp_tb = tstate->exc_traceback; | |
| tstate->exc_type = type; | |
| tstate->exc_value = value; | |
| tstate->exc_traceback = tb; | |
| Py_XDECREF(tmp_type); | |
| Py_XDECREF(tmp_value); | |
| Py_XDECREF(tmp_tb); | |
| } | |
| /////////////// SwapException.proto /////////////// | |
| //@substitute: naming | |
| //@requires: PyThreadStateGet | |
| static CYTHON_INLINE void __Pyx__ExceptionSwap(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb); /*proto*/ | |
| static CYTHON_INLINE void __Pyx_ExceptionSwap(PyObject **type, PyObject **value, PyObject **tb); /*proto*/ | |
| /////////////// SwapException /////////////// | |
| static CYTHON_INLINE void __Pyx__ExceptionSwap(PyThreadState *tstate, PyObject **type, PyObject **value, PyObject **tb) { | |
| PyObject *tmp_type, *tmp_value, *tmp_tb; | |
| _PyErr_StackItem *exc_info = tstate->exc_info; | |
| tmp_value = exc_info->exc_value; | |
| exc_info->exc_value = *value; | |
| if (tmp_value == NULL || tmp_value == Py_None) { | |
| Py_XDECREF(tmp_value); | |
| tmp_value = NULL; | |
| tmp_type = NULL; | |
| tmp_tb = NULL; | |
| } else { | |
| // TODO: avoid swapping these at all | |
| tmp_type = (PyObject*) Py_TYPE(tmp_value); | |
| Py_INCREF(tmp_type); | |
| tmp_tb = ((PyBaseExceptionObject*) tmp_value)->traceback; | |
| Py_XINCREF(tmp_tb); | |
| tmp_tb = PyException_GetTraceback(tmp_value); | |
| } | |
| _PyErr_StackItem *exc_info = tstate->exc_info; | |
| tmp_type = exc_info->exc_type; | |
| tmp_value = exc_info->exc_value; | |
| tmp_tb = exc_info->exc_traceback; | |
| exc_info->exc_type = *type; | |
| exc_info->exc_value = *value; | |
| exc_info->exc_traceback = *tb; | |
| tmp_type = tstate->exc_type; | |
| tmp_value = tstate->exc_value; | |
| tmp_tb = tstate->exc_traceback; | |
| tstate->exc_type = *type; | |
| tstate->exc_value = *value; | |
| tstate->exc_traceback = *tb; | |
| *type = tmp_type; | |
| *value = tmp_value; | |
| *tb = tmp_tb; | |
| } | |
| static CYTHON_INLINE void __Pyx_ExceptionSwap(PyObject **type, PyObject **value, PyObject **tb) { | |
| PyObject *tmp_type, *tmp_value, *tmp_tb; | |
| PyErr_GetExcInfo(&tmp_type, &tmp_value, &tmp_tb); | |
| PyErr_SetExcInfo(*type, *value, *tb); | |
| *type = tmp_type; | |
| *value = tmp_value; | |
| *tb = tmp_tb; | |
| } | |
| /////////////// WriteUnraisableException.proto /////////////// | |
| static void __Pyx_WriteUnraisable(const char *name, int clineno, | |
| int lineno, const char *filename, | |
| int full_traceback, int nogil); /*proto*/ | |
| /////////////// WriteUnraisableException /////////////// | |
| //@requires: PyErrFetchRestore | |
| //@requires: PyThreadStateGet | |
| static void __Pyx_WriteUnraisable(const char *name, int clineno, | |
| int lineno, const char *filename, | |
| int full_traceback, int nogil) { | |
| PyObject *old_exc, *old_val, *old_tb; | |
| PyObject *ctx; | |
| __Pyx_PyThreadState_declare | |
| PyGILState_STATE state; | |
| if (nogil) | |
| state = PyGILState_Ensure(); | |
| /* arbitrary, to suppress warning */ | |
| else state = (PyGILState_STATE)0; | |
| CYTHON_UNUSED_VAR(clineno); | |
| CYTHON_UNUSED_VAR(lineno); | |
| CYTHON_UNUSED_VAR(filename); | |
| CYTHON_MAYBE_UNUSED_VAR(nogil); | |
| __Pyx_PyThreadState_assign | |
| __Pyx_ErrFetch(&old_exc, &old_val, &old_tb); | |
| if (full_traceback) { | |
| Py_XINCREF(old_exc); | |
| Py_XINCREF(old_val); | |
| Py_XINCREF(old_tb); | |
| __Pyx_ErrRestore(old_exc, old_val, old_tb); | |
| PyErr_PrintEx(1); | |
| } | |
| ctx = PyString_FromString(name); | |
| ctx = PyUnicode_FromString(name); | |
| __Pyx_ErrRestore(old_exc, old_val, old_tb); | |
| if (!ctx) { | |
| PyErr_WriteUnraisable(Py_None); | |
| } else { | |
| PyErr_WriteUnraisable(ctx); | |
| Py_DECREF(ctx); | |
| } | |
| if (nogil) | |
| PyGILState_Release(state); | |
| } | |
| /////////////// CLineInTraceback.proto /////////////// | |
| static int __Pyx_CLineForTraceback(PyThreadState *tstate, int c_line);/*proto*/ | |
| /////////////// CLineInTraceback /////////////// | |
| //@requires: ObjectHandling.c::PyObjectGetAttrStrNoError | |
| //@requires: ObjectHandling.c::PyDictVersioning | |
| //@requires: PyErrFetchRestore | |
| //@substitute: naming | |
| static int __Pyx_CLineForTraceback(PyThreadState *tstate, int c_line) { | |
| PyObject *use_cline; | |
| PyObject *ptype, *pvalue, *ptraceback; | |
| PyObject **cython_runtime_dict; | |
| CYTHON_MAYBE_UNUSED_VAR(tstate); | |
| if (unlikely(!${cython_runtime_cname})) { | |
| // Very early error where the runtime module is not set up yet. | |
| return c_line; | |
| } | |
| __Pyx_ErrFetchInState(tstate, &ptype, &pvalue, &ptraceback); | |
| cython_runtime_dict = _PyObject_GetDictPtr(${cython_runtime_cname}); | |
| if (likely(cython_runtime_dict)) { | |
| __PYX_PY_DICT_LOOKUP_IF_MODIFIED( | |
| use_cline, *cython_runtime_dict, | |
| __Pyx_PyDict_GetItemStr(*cython_runtime_dict, PYIDENT("cline_in_traceback"))) | |
| } else | |
| { | |
| PyObject *use_cline_obj = __Pyx_PyObject_GetAttrStrNoError(${cython_runtime_cname}, PYIDENT("cline_in_traceback")); | |
| if (use_cline_obj) { | |
| use_cline = PyObject_Not(use_cline_obj) ? Py_False : Py_True; | |
| Py_DECREF(use_cline_obj); | |
| } else { | |
| PyErr_Clear(); | |
| use_cline = NULL; | |
| } | |
| } | |
| if (!use_cline) { | |
| c_line = 0; | |
| // No need to handle errors here when we reset the exception state just afterwards. | |
| (void) PyObject_SetAttr(${cython_runtime_cname}, PYIDENT("cline_in_traceback"), Py_False); | |
| } | |
| else if (use_cline == Py_False || (use_cline != Py_True && PyObject_Not(use_cline) != 0)) { | |
| c_line = 0; | |
| } | |
| __Pyx_ErrRestoreInState(tstate, ptype, pvalue, ptraceback); | |
| return c_line; | |
| } | |
| /////////////// AddTraceback.proto /////////////// | |
| static void __Pyx_AddTraceback(const char *funcname, int c_line, | |
| int py_line, const char *filename); /*proto*/ | |
| /////////////// AddTraceback /////////////// | |
| //@requires: ModuleSetupCode.c::CodeObjectCache | |
| //@requires: CLineInTraceback | |
| //@substitute: naming | |
| static PyObject *__Pyx_PyCode_Replace_For_AddTraceback(PyObject *code, PyObject *scratch_dict, | |
| PyObject *firstlineno, PyObject *name) { | |
| PyObject *replace = NULL; | |
| if (unlikely(PyDict_SetItemString(scratch_dict, "co_firstlineno", firstlineno))) return NULL; | |
| if (unlikely(PyDict_SetItemString(scratch_dict, "co_name", name))) return NULL; | |
| replace = PyObject_GetAttrString(code, "replace"); | |
| if (likely(replace)) { | |
| PyObject *result; | |
| result = PyObject_Call(replace, $empty_tuple, scratch_dict); | |
| Py_DECREF(replace); | |
| return result; | |
| } | |
| PyErr_Clear(); | |
| // If we're here, we're probably on Python <=3.7 which doesn't have code.replace. | |
| // In this we take a lazy interpreted route (without regard to performance | |
| // since it's fairly old and this is mostly just to get something working) | |
| { | |
| PyObject *compiled = NULL, *result = NULL; | |
| if (unlikely(PyDict_SetItemString(scratch_dict, "code", code))) return NULL; | |
| if (unlikely(PyDict_SetItemString(scratch_dict, "type", (PyObject*)(&PyType_Type)))) return NULL; | |
| compiled = Py_CompileString( | |
| "out = type(code)(\n" | |
| " code.co_argcount, code.co_kwonlyargcount, code.co_nlocals, code.co_stacksize,\n" | |
| " code.co_flags, code.co_code, code.co_consts, code.co_names,\n" | |
| " code.co_varnames, code.co_filename, co_name, co_firstlineno,\n" | |
| " code.co_lnotab)\n", "<dummy>", Py_file_input); | |
| if (!compiled) return NULL; | |
| result = PyEval_EvalCode(compiled, scratch_dict, scratch_dict); | |
| Py_DECREF(compiled); | |
| if (!result) PyErr_Print(); | |
| Py_DECREF(result); | |
| result = PyDict_GetItemString(scratch_dict, "out"); | |
| if (result) Py_INCREF(result); | |
| return result; | |
| } | |
| return NULL; | |
| } | |
| static void __Pyx_AddTraceback(const char *funcname, int c_line, | |
| int py_line, const char *filename) { | |
| PyObject *code_object = NULL, *py_py_line = NULL, *py_funcname = NULL, *dict = NULL; | |
| PyObject *replace = NULL, *getframe = NULL, *frame = NULL; | |
| PyObject *exc_type, *exc_value, *exc_traceback; | |
| int success = 0; | |
| if (c_line) { | |
| // Avoid "unused" warning as long as we don't use this. | |
| (void) $cfilenm_cname; | |
| (void) __Pyx_CLineForTraceback(__Pyx_PyThreadState_Current, c_line); | |
| } | |
| // DW - this is a horrendous hack, but I'm quite proud of it. Essentially | |
| // we need to generate a frame with the right line number/filename/funcname. | |
| // We do this by compiling a small bit of code that uses sys._getframe to get a | |
| // frame, and then customizing the details of the code to match. | |
| // We then run the code object and use the generated frame to set the traceback. | |
| PyErr_Fetch(&exc_type, &exc_value, &exc_traceback); | |
| code_object = Py_CompileString("_getframe()", filename, Py_eval_input); | |
| if (unlikely(!code_object)) goto bad; | |
| py_py_line = PyLong_FromLong(py_line); | |
| if (unlikely(!py_py_line)) goto bad; | |
| py_funcname = PyUnicode_FromString(funcname); | |
| if (unlikely(!py_funcname)) goto bad; | |
| dict = PyDict_New(); | |
| if (unlikely(!dict)) goto bad; | |
| { | |
| PyObject *old_code_object = code_object; | |
| code_object = __Pyx_PyCode_Replace_For_AddTraceback(code_object, dict, py_py_line, py_funcname); | |
| Py_DECREF(old_code_object); | |
| } | |
| if (unlikely(!code_object)) goto bad; | |
| // Note that getframe is borrowed | |
| getframe = PySys_GetObject("_getframe"); | |
| if (unlikely(!getframe)) goto bad; | |
| // reuse dict as globals (nothing conflicts, and it saves an allocation) | |
| if (unlikely(PyDict_SetItemString(dict, "_getframe", getframe))) goto bad; | |
| frame = PyEval_EvalCode(code_object, dict, dict); | |
| if (unlikely(!frame) || frame == Py_None) goto bad; | |
| success = 1; | |
| bad: | |
| PyErr_Restore(exc_type, exc_value, exc_traceback); | |
| Py_XDECREF(code_object); | |
| Py_XDECREF(py_py_line); | |
| Py_XDECREF(py_funcname); | |
| Py_XDECREF(dict); | |
| Py_XDECREF(replace); | |
| if (success) { | |
| // Unfortunately an easy way to check the type of frame isn't in the | |
| // limited API. The check against None should cover the most | |
| // likely wrong answer though. | |
| PyTraceBack_Here( | |
| // Python < 0x03090000 didn't expose PyFrameObject | |
| // but they all expose struct _frame as an opaque type | |
| (struct _frame*)frame); | |
| } | |
| Py_XDECREF(frame); | |
| } | |
| static PyCodeObject* __Pyx_CreateCodeObjectForTraceback( | |
| const char *funcname, int c_line, | |
| int py_line, const char *filename) { | |
| PyCodeObject *py_code = NULL; | |
| PyObject *py_funcname = NULL; | |
| PyObject *py_srcfile = NULL; | |
| py_srcfile = PyString_FromString(filename); | |
| if (!py_srcfile) goto bad; | |
| if (c_line) { | |
| py_funcname = PyString_FromFormat( "%s (%s:%d)", funcname, $cfilenm_cname, c_line); | |
| if (!py_funcname) goto bad; | |
| py_funcname = PyUnicode_FromFormat( "%s (%s:%d)", funcname, $cfilenm_cname, c_line); | |
| if (!py_funcname) goto bad; | |
| funcname = PyUnicode_AsUTF8(py_funcname); | |
| if (!funcname) goto bad; | |
| } | |
| else { | |
| py_funcname = PyString_FromString(funcname); | |
| if (!py_funcname) goto bad; | |
| } | |
| py_code = __Pyx_PyCode_New( | |
| 0, /*int argcount,*/ | |
| 0, /*int posonlyargcount,*/ | |
| 0, /*int kwonlyargcount,*/ | |
| 0, /*int nlocals,*/ | |
| 0, /*int stacksize,*/ | |
| 0, /*int flags,*/ | |
| $empty_bytes, /*PyObject *code,*/ | |
| $empty_tuple, /*PyObject *consts,*/ | |
| $empty_tuple, /*PyObject *names,*/ | |
| $empty_tuple, /*PyObject *varnames,*/ | |
| $empty_tuple, /*PyObject *freevars,*/ | |
| $empty_tuple, /*PyObject *cellvars,*/ | |
| py_srcfile, /*PyObject *filename,*/ | |
| py_funcname, /*PyObject *name,*/ | |
| py_line, /*int firstlineno,*/ | |
| $empty_bytes /*PyObject *lnotab*/ | |
| ); | |
| Py_DECREF(py_srcfile); | |
| py_code = PyCode_NewEmpty(filename, funcname, py_line); | |
| Py_XDECREF(py_funcname); /* XDECREF since it's only set on Py3 if cline */ | |
| return py_code; | |
| bad: | |
| Py_XDECREF(py_funcname); | |
| Py_XDECREF(py_srcfile); | |
| return NULL; | |
| } | |
| static void __Pyx_AddTraceback(const char *funcname, int c_line, | |
| int py_line, const char *filename) { | |
| PyCodeObject *py_code = 0; | |
| PyFrameObject *py_frame = 0; | |
| PyThreadState *tstate = __Pyx_PyThreadState_Current; | |
| PyObject *ptype, *pvalue, *ptraceback; | |
| if (c_line) { | |
| c_line = __Pyx_CLineForTraceback(tstate, c_line); | |
| } | |
| // Negate to avoid collisions between py and c lines. | |
| py_code = $global_code_object_cache_find(c_line ? -c_line : py_line); | |
| if (!py_code) { | |
| __Pyx_ErrFetchInState(tstate, &ptype, &pvalue, &ptraceback); | |
| py_code = __Pyx_CreateCodeObjectForTraceback( | |
| funcname, c_line, py_line, filename); | |
| if (!py_code) { | |
| /* If the code object creation fails, then we should clear the | |
| fetched exception references and propagate the new exception */ | |
| Py_XDECREF(ptype); | |
| Py_XDECREF(pvalue); | |
| Py_XDECREF(ptraceback); | |
| goto bad; | |
| } | |
| __Pyx_ErrRestoreInState(tstate, ptype, pvalue, ptraceback); | |
| $global_code_object_cache_insert(c_line ? -c_line : py_line, py_code); | |
| } | |
| py_frame = PyFrame_New( | |
| tstate, /*PyThreadState *tstate,*/ | |
| py_code, /*PyCodeObject *code,*/ | |
| $moddict_cname, /*PyObject *globals,*/ | |
| 0 /*PyObject *locals*/ | |
| ); | |
| if (!py_frame) goto bad; | |
| __Pyx_PyFrame_SetLineNumber(py_frame, py_line); | |
| PyTraceBack_Here(py_frame); | |
| bad: | |
| Py_XDECREF(py_code); | |
| Py_XDECREF(py_frame); | |
| } | |