Commit df22c03b authored by Victor Stinner's avatar Victor Stinner Committed by GitHub

bpo-36829: PyErr_WriteUnraisable() normalizes exception (GH-13507)

PyErr_WriteUnraisable() now creates a traceback object if there is no
current traceback. Moreover, call PyErr_NormalizeException() and
PyException_SetTraceback() to normalize the exception value. Ignore
silently any error.
parent 5edcf263
...@@ -86,6 +86,10 @@ PyAPI_FUNC(void) _Py_DumpHexadecimal( ...@@ -86,6 +86,10 @@ PyAPI_FUNC(void) _Py_DumpHexadecimal(
unsigned long value, unsigned long value,
Py_ssize_t width); Py_ssize_t width);
PyAPI_FUNC(PyObject*) _PyTraceBack_FromFrame(
PyObject *tb_next,
struct _frame *frame);
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif #endif
......
...@@ -882,19 +882,14 @@ class UnraisableHookTest(unittest.TestCase): ...@@ -882,19 +882,14 @@ class UnraisableHookTest(unittest.TestCase):
import _testcapi import _testcapi
import types import types
try: try:
# raise the exception to get a traceback in the except block _testcapi.write_unraisable_exc(exc, obj)
try: return types.SimpleNamespace(exc_type=type(exc),
raise exc exc_value=exc,
except Exception as exc2: exc_traceback=exc.__traceback__,
_testcapi.write_unraisable_exc(exc2, obj) object=obj)
return types.SimpleNamespace(exc_type=type(exc2),
exc_value=exc2,
exc_traceback=exc2.__traceback__,
object=obj)
finally: finally:
# Explicitly break any reference cycle # Explicitly break any reference cycle
exc = None exc = None
exc2 = None
def test_original_unraisablehook(self): def test_original_unraisablehook(self):
obj = "an object" obj = "an object"
......
:c:func:`PyErr_WriteUnraisable` now creates a traceback object if there is
no current traceback. Moreover, call :c:func:`PyErr_NormalizeException` and
:c:func:`PyException_SetTraceback` to normalize the exception value. Ignore any
error.
...@@ -4,6 +4,7 @@ ...@@ -4,6 +4,7 @@
#include "Python.h" #include "Python.h"
#include "pycore_coreconfig.h" #include "pycore_coreconfig.h"
#include "pycore_pystate.h" #include "pycore_pystate.h"
#include "pycore_traceback.h"
#ifndef __STDC__ #ifndef __STDC__
#ifndef MS_WINDOWS #ifndef MS_WINDOWS
...@@ -1048,7 +1049,7 @@ write_unraisable_exc_file(PyObject *exc_type, PyObject *exc_value, ...@@ -1048,7 +1049,7 @@ write_unraisable_exc_file(PyObject *exc_type, PyObject *exc_value,
} }
} }
if (!exc_type) { if (exc_type == NULL || exc_type == Py_None) {
return -1; return -1;
} }
...@@ -1106,6 +1107,7 @@ write_unraisable_exc_file(PyObject *exc_type, PyObject *exc_value, ...@@ -1106,6 +1107,7 @@ write_unraisable_exc_file(PyObject *exc_type, PyObject *exc_value,
} }
} }
} }
if (PyFile_WriteString("\n", file) < 0) { if (PyFile_WriteString("\n", file) < 0) {
return -1; return -1;
} }
...@@ -1177,6 +1179,24 @@ PyErr_WriteUnraisable(PyObject *obj) ...@@ -1177,6 +1179,24 @@ PyErr_WriteUnraisable(PyObject *obj)
goto default_hook; goto default_hook;
} }
if (exc_tb == NULL) {
struct _frame *frame = _PyThreadState_GET()->frame;
if (frame != NULL) {
exc_tb = _PyTraceBack_FromFrame(NULL, frame);
if (exc_tb == NULL) {
PyErr_Clear();
}
}
}
PyErr_NormalizeException(&exc_type, &exc_value, &exc_tb);
if (exc_tb != NULL && exc_tb != Py_None && PyTraceBack_Check(exc_tb)) {
if (PyException_SetTraceback(exc_value, exc_tb) < 0) {
PyErr_Clear();
}
}
_Py_IDENTIFIER(unraisablehook); _Py_IDENTIFIER(unraisablehook);
PyObject *hook = _PySys_GetObjectId(&PyId_unraisablehook); PyObject *hook = _PySys_GetObjectId(&PyId_unraisablehook);
if (hook != NULL && hook != Py_None) { if (hook != NULL && hook != Py_None) {
......
...@@ -227,13 +227,24 @@ PyTypeObject PyTraceBack_Type = { ...@@ -227,13 +227,24 @@ PyTypeObject PyTraceBack_Type = {
tb_new, /* tp_new */ tb_new, /* tp_new */
}; };
PyObject*
_PyTraceBack_FromFrame(PyObject *tb_next, PyFrameObject *frame)
{
assert(tb_next == NULL || PyTraceBack_Check(tb_next));
assert(frame != NULL);
return tb_create_raw((PyTracebackObject *)tb_next, frame, frame->f_lasti,
PyFrame_GetLineNumber(frame));
}
int int
PyTraceBack_Here(PyFrameObject *frame) PyTraceBack_Here(PyFrameObject *frame)
{ {
PyObject *exc, *val, *tb, *newtb; PyObject *exc, *val, *tb, *newtb;
PyErr_Fetch(&exc, &val, &tb); PyErr_Fetch(&exc, &val, &tb);
newtb = tb_create_raw((PyTracebackObject *)tb, frame, frame->f_lasti, newtb = _PyTraceBack_FromFrame(tb, frame);
PyFrame_GetLineNumber(frame));
if (newtb == NULL) { if (newtb == NULL) {
_PyErr_ChainExceptions(exc, val, tb); _PyErr_ChainExceptions(exc, val, tb);
return -1; return -1;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment