mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 11:14:33 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			3051 lines
		
	
	
	
		
			87 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			3051 lines
		
	
	
	
		
			87 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * New exceptions.c written in Iceland by Richard Jones and Georg Brandl.
 | |
|  *
 | |
|  * Thanks go to Tim Peters and Michael Hudson for debugging.
 | |
|  */
 | |
| 
 | |
| #define PY_SSIZE_T_CLEAN
 | |
| #include <Python.h>
 | |
| #include "pycore_initconfig.h"
 | |
| #include "pycore_object.h"
 | |
| #include "structmember.h"         // PyMemberDef
 | |
| #include "osdefs.h"               // SEP
 | |
| 
 | |
| 
 | |
| /* Compatibility aliases */
 | |
| PyObject *PyExc_EnvironmentError = NULL;
 | |
| PyObject *PyExc_IOError = NULL;
 | |
| #ifdef MS_WINDOWS
 | |
| PyObject *PyExc_WindowsError = NULL;
 | |
| #endif
 | |
| 
 | |
| /* The dict map from errno codes to OSError subclasses */
 | |
| static PyObject *errnomap = NULL;
 | |
| 
 | |
| 
 | |
| /* NOTE: If the exception class hierarchy changes, don't forget to update
 | |
|  * Lib/test/exception_hierarchy.txt
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  *    BaseException
 | |
|  */
 | |
| static PyObject *
 | |
| BaseException_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyBaseExceptionObject *self;
 | |
| 
 | |
|     self = (PyBaseExceptionObject *)type->tp_alloc(type, 0);
 | |
|     if (!self)
 | |
|         return NULL;
 | |
|     /* the dict is created on the fly in PyObject_GenericSetAttr */
 | |
|     self->dict = NULL;
 | |
|     self->traceback = self->cause = self->context = NULL;
 | |
|     self->suppress_context = 0;
 | |
| 
 | |
|     if (args) {
 | |
|         self->args = args;
 | |
|         Py_INCREF(args);
 | |
|         return (PyObject *)self;
 | |
|     }
 | |
| 
 | |
|     self->args = PyTuple_New(0);
 | |
|     if (!self->args) {
 | |
|         Py_DECREF(self);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return (PyObject *)self;
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_init(PyBaseExceptionObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     if (!_PyArg_NoKeywords(Py_TYPE(self)->tp_name, kwds))
 | |
|         return -1;
 | |
| 
 | |
|     Py_INCREF(args);
 | |
|     Py_XSETREF(self->args, args);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_clear(PyBaseExceptionObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->dict);
 | |
|     Py_CLEAR(self->args);
 | |
|     Py_CLEAR(self->traceback);
 | |
|     Py_CLEAR(self->cause);
 | |
|     Py_CLEAR(self->context);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| BaseException_dealloc(PyBaseExceptionObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     BaseException_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_traverse(PyBaseExceptionObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(self->dict);
 | |
|     Py_VISIT(self->args);
 | |
|     Py_VISIT(self->traceback);
 | |
|     Py_VISIT(self->cause);
 | |
|     Py_VISIT(self->context);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_str(PyBaseExceptionObject *self)
 | |
| {
 | |
|     switch (PyTuple_GET_SIZE(self->args)) {
 | |
|     case 0:
 | |
|         return PyUnicode_FromString("");
 | |
|     case 1:
 | |
|         return PyObject_Str(PyTuple_GET_ITEM(self->args, 0));
 | |
|     default:
 | |
|         return PyObject_Str(self->args);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_repr(PyBaseExceptionObject *self)
 | |
| {
 | |
|     const char *name = _PyType_Name(Py_TYPE(self));
 | |
|     if (PyTuple_GET_SIZE(self->args) == 1)
 | |
|         return PyUnicode_FromFormat("%s(%R)", name,
 | |
|                                     PyTuple_GET_ITEM(self->args, 0));
 | |
|     else
 | |
|         return PyUnicode_FromFormat("%s%R", name, self->args);
 | |
| }
 | |
| 
 | |
| /* Pickling support */
 | |
| static PyObject *
 | |
| BaseException_reduce(PyBaseExceptionObject *self, PyObject *Py_UNUSED(ignored))
 | |
| {
 | |
|     if (self->args && self->dict)
 | |
|         return PyTuple_Pack(3, Py_TYPE(self), self->args, self->dict);
 | |
|     else
 | |
|         return PyTuple_Pack(2, Py_TYPE(self), self->args);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Needed for backward compatibility, since exceptions used to store
 | |
|  * all their attributes in the __dict__. Code is taken from cPickle's
 | |
|  * load_build function.
 | |
|  */
 | |
| static PyObject *
 | |
| BaseException_setstate(PyObject *self, PyObject *state)
 | |
| {
 | |
|     PyObject *d_key, *d_value;
 | |
|     Py_ssize_t i = 0;
 | |
| 
 | |
|     if (state != Py_None) {
 | |
|         if (!PyDict_Check(state)) {
 | |
|             PyErr_SetString(PyExc_TypeError, "state is not a dictionary");
 | |
|             return NULL;
 | |
|         }
 | |
|         while (PyDict_Next(state, &i, &d_key, &d_value)) {
 | |
|             if (PyObject_SetAttr(self, d_key, d_value) < 0)
 | |
|                 return NULL;
 | |
|         }
 | |
|     }
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_with_traceback(PyObject *self, PyObject *tb) {
 | |
|     if (PyException_SetTraceback(self, tb))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_INCREF(self);
 | |
|     return self;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(with_traceback_doc,
 | |
| "Exception.with_traceback(tb) --\n\
 | |
|     set self.__traceback__ to tb and return self.");
 | |
| 
 | |
| 
 | |
| static PyMethodDef BaseException_methods[] = {
 | |
|    {"__reduce__", (PyCFunction)BaseException_reduce, METH_NOARGS },
 | |
|    {"__setstate__", (PyCFunction)BaseException_setstate, METH_O },
 | |
|    {"with_traceback", (PyCFunction)BaseException_with_traceback, METH_O,
 | |
|     with_traceback_doc},
 | |
|    {NULL, NULL, 0, NULL},
 | |
| };
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_get_args(PyBaseExceptionObject *self, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     if (self->args == NULL) {
 | |
|         Py_RETURN_NONE;
 | |
|     }
 | |
|     Py_INCREF(self->args);
 | |
|     return self->args;
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_set_args(PyBaseExceptionObject *self, PyObject *val, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     PyObject *seq;
 | |
|     if (val == NULL) {
 | |
|         PyErr_SetString(PyExc_TypeError, "args may not be deleted");
 | |
|         return -1;
 | |
|     }
 | |
|     seq = PySequence_Tuple(val);
 | |
|     if (!seq)
 | |
|         return -1;
 | |
|     Py_XSETREF(self->args, seq);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_get_tb(PyBaseExceptionObject *self, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     if (self->traceback == NULL) {
 | |
|         Py_RETURN_NONE;
 | |
|     }
 | |
|     Py_INCREF(self->traceback);
 | |
|     return self->traceback;
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_set_tb(PyBaseExceptionObject *self, PyObject *tb, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     if (tb == NULL) {
 | |
|         PyErr_SetString(PyExc_TypeError, "__traceback__ may not be deleted");
 | |
|         return -1;
 | |
|     }
 | |
|     else if (!(tb == Py_None || PyTraceBack_Check(tb))) {
 | |
|         PyErr_SetString(PyExc_TypeError,
 | |
|                         "__traceback__ must be a traceback or None");
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     Py_INCREF(tb);
 | |
|     Py_XSETREF(self->traceback, tb);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_get_context(PyObject *self, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     PyObject *res = PyException_GetContext(self);
 | |
|     if (res)
 | |
|         return res;  /* new reference already returned above */
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_set_context(PyObject *self, PyObject *arg, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     if (arg == NULL) {
 | |
|         PyErr_SetString(PyExc_TypeError, "__context__ may not be deleted");
 | |
|         return -1;
 | |
|     } else if (arg == Py_None) {
 | |
|         arg = NULL;
 | |
|     } else if (!PyExceptionInstance_Check(arg)) {
 | |
|         PyErr_SetString(PyExc_TypeError, "exception context must be None "
 | |
|                         "or derive from BaseException");
 | |
|         return -1;
 | |
|     } else {
 | |
|         /* PyException_SetContext steals this reference */
 | |
|         Py_INCREF(arg);
 | |
|     }
 | |
|     PyException_SetContext(self, arg);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| BaseException_get_cause(PyObject *self, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     PyObject *res = PyException_GetCause(self);
 | |
|     if (res)
 | |
|         return res;  /* new reference already returned above */
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| BaseException_set_cause(PyObject *self, PyObject *arg, void *Py_UNUSED(ignored))
 | |
| {
 | |
|     if (arg == NULL) {
 | |
|         PyErr_SetString(PyExc_TypeError, "__cause__ may not be deleted");
 | |
|         return -1;
 | |
|     } else if (arg == Py_None) {
 | |
|         arg = NULL;
 | |
|     } else if (!PyExceptionInstance_Check(arg)) {
 | |
|         PyErr_SetString(PyExc_TypeError, "exception cause must be None "
 | |
|                         "or derive from BaseException");
 | |
|         return -1;
 | |
|     } else {
 | |
|         /* PyException_SetCause steals this reference */
 | |
|         Py_INCREF(arg);
 | |
|     }
 | |
|     PyException_SetCause(self, arg);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyGetSetDef BaseException_getset[] = {
 | |
|     {"__dict__", PyObject_GenericGetDict, PyObject_GenericSetDict},
 | |
|     {"args", (getter)BaseException_get_args, (setter)BaseException_set_args},
 | |
|     {"__traceback__", (getter)BaseException_get_tb, (setter)BaseException_set_tb},
 | |
|     {"__context__", BaseException_get_context,
 | |
|      BaseException_set_context, PyDoc_STR("exception context")},
 | |
|     {"__cause__", BaseException_get_cause,
 | |
|      BaseException_set_cause, PyDoc_STR("exception cause")},
 | |
|     {NULL},
 | |
| };
 | |
| 
 | |
| 
 | |
| PyObject *
 | |
| PyException_GetTraceback(PyObject *self) {
 | |
|     PyBaseExceptionObject *base_self = (PyBaseExceptionObject *)self;
 | |
|     Py_XINCREF(base_self->traceback);
 | |
|     return base_self->traceback;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyException_SetTraceback(PyObject *self, PyObject *tb) {
 | |
|     return BaseException_set_tb((PyBaseExceptionObject *)self, tb, NULL);
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyException_GetCause(PyObject *self) {
 | |
|     PyObject *cause = ((PyBaseExceptionObject *)self)->cause;
 | |
|     Py_XINCREF(cause);
 | |
|     return cause;
 | |
| }
 | |
| 
 | |
| /* Steals a reference to cause */
 | |
| void
 | |
| PyException_SetCause(PyObject *self, PyObject *cause)
 | |
| {
 | |
|     ((PyBaseExceptionObject *)self)->suppress_context = 1;
 | |
|     Py_XSETREF(((PyBaseExceptionObject *)self)->cause, cause);
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyException_GetContext(PyObject *self) {
 | |
|     PyObject *context = ((PyBaseExceptionObject *)self)->context;
 | |
|     Py_XINCREF(context);
 | |
|     return context;
 | |
| }
 | |
| 
 | |
| /* Steals a reference to context */
 | |
| void
 | |
| PyException_SetContext(PyObject *self, PyObject *context)
 | |
| {
 | |
|     Py_XSETREF(((PyBaseExceptionObject *)self)->context, context);
 | |
| }
 | |
| 
 | |
| #undef PyExceptionClass_Name
 | |
| 
 | |
| const char *
 | |
| PyExceptionClass_Name(PyObject *ob)
 | |
| {
 | |
|     return ((PyTypeObject*)ob)->tp_name;
 | |
| }
 | |
| 
 | |
| static struct PyMemberDef BaseException_members[] = {
 | |
|     {"__suppress_context__", T_BOOL,
 | |
|      offsetof(PyBaseExceptionObject, suppress_context)},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| 
 | |
| static PyTypeObject _PyExc_BaseException = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "BaseException", /*tp_name*/
 | |
|     sizeof(PyBaseExceptionObject), /*tp_basicsize*/
 | |
|     0,                          /*tp_itemsize*/
 | |
|     (destructor)BaseException_dealloc, /*tp_dealloc*/
 | |
|     0,                          /*tp_vectorcall_offset*/
 | |
|     0,                          /*tp_getattr*/
 | |
|     0,                          /*tp_setattr*/
 | |
|     0,                          /*tp_as_async*/
 | |
|     (reprfunc)BaseException_repr, /*tp_repr*/
 | |
|     0,                          /*tp_as_number*/
 | |
|     0,                          /*tp_as_sequence*/
 | |
|     0,                          /*tp_as_mapping*/
 | |
|     0,                          /*tp_hash */
 | |
|     0,                          /*tp_call*/
 | |
|     (reprfunc)BaseException_str,  /*tp_str*/
 | |
|     PyObject_GenericGetAttr,    /*tp_getattro*/
 | |
|     PyObject_GenericSetAttr,    /*tp_setattro*/
 | |
|     0,                          /*tp_as_buffer*/
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC |
 | |
|         Py_TPFLAGS_BASE_EXC_SUBCLASS,  /*tp_flags*/
 | |
|     PyDoc_STR("Common base class for all exceptions"), /* tp_doc */
 | |
|     (traverseproc)BaseException_traverse, /* tp_traverse */
 | |
|     (inquiry)BaseException_clear, /* tp_clear */
 | |
|     0,                          /* tp_richcompare */
 | |
|     0,                          /* tp_weaklistoffset */
 | |
|     0,                          /* tp_iter */
 | |
|     0,                          /* tp_iternext */
 | |
|     BaseException_methods,      /* tp_methods */
 | |
|     BaseException_members,      /* tp_members */
 | |
|     BaseException_getset,       /* tp_getset */
 | |
|     0,                          /* tp_base */
 | |
|     0,                          /* tp_dict */
 | |
|     0,                          /* tp_descr_get */
 | |
|     0,                          /* tp_descr_set */
 | |
|     offsetof(PyBaseExceptionObject, dict), /* tp_dictoffset */
 | |
|     (initproc)BaseException_init, /* tp_init */
 | |
|     0,                          /* tp_alloc */
 | |
|     BaseException_new,          /* tp_new */
 | |
| };
 | |
| /* the CPython API expects exceptions to be (PyObject *) - both a hold-over
 | |
| from the previous implementation and also allowing Python objects to be used
 | |
| in the API */
 | |
| PyObject *PyExc_BaseException = (PyObject *)&_PyExc_BaseException;
 | |
| 
 | |
| /* note these macros omit the last semicolon so the macro invocation may
 | |
|  * include it and not look strange.
 | |
|  */
 | |
| #define SimpleExtendsException(EXCBASE, EXCNAME, EXCDOC) \
 | |
| static PyTypeObject _PyExc_ ## EXCNAME = { \
 | |
|     PyVarObject_HEAD_INIT(NULL, 0) \
 | |
|     # EXCNAME, \
 | |
|     sizeof(PyBaseExceptionObject), \
 | |
|     0, (destructor)BaseException_dealloc, 0, 0, 0, 0, 0, 0, 0, \
 | |
|     0, 0, 0, 0, 0, 0, 0, \
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC, \
 | |
|     PyDoc_STR(EXCDOC), (traverseproc)BaseException_traverse, \
 | |
|     (inquiry)BaseException_clear, 0, 0, 0, 0, 0, 0, 0, &_ ## EXCBASE, \
 | |
|     0, 0, 0, offsetof(PyBaseExceptionObject, dict), \
 | |
|     (initproc)BaseException_init, 0, BaseException_new,\
 | |
| }; \
 | |
| PyObject *PyExc_ ## EXCNAME = (PyObject *)&_PyExc_ ## EXCNAME
 | |
| 
 | |
| #define MiddlingExtendsException(EXCBASE, EXCNAME, EXCSTORE, EXCDOC) \
 | |
| static PyTypeObject _PyExc_ ## EXCNAME = { \
 | |
|     PyVarObject_HEAD_INIT(NULL, 0) \
 | |
|     # EXCNAME, \
 | |
|     sizeof(Py ## EXCSTORE ## Object), \
 | |
|     0, (destructor)EXCSTORE ## _dealloc, 0, 0, 0, 0, 0, 0, 0, 0, 0, \
 | |
|     0, 0, 0, 0, 0, \
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC, \
 | |
|     PyDoc_STR(EXCDOC), (traverseproc)EXCSTORE ## _traverse, \
 | |
|     (inquiry)EXCSTORE ## _clear, 0, 0, 0, 0, 0, 0, 0, &_ ## EXCBASE, \
 | |
|     0, 0, 0, offsetof(Py ## EXCSTORE ## Object, dict), \
 | |
|     (initproc)EXCSTORE ## _init, 0, 0, \
 | |
| }; \
 | |
| PyObject *PyExc_ ## EXCNAME = (PyObject *)&_PyExc_ ## EXCNAME
 | |
| 
 | |
| #define ComplexExtendsException(EXCBASE, EXCNAME, EXCSTORE, EXCNEW, \
 | |
|                                 EXCMETHODS, EXCMEMBERS, EXCGETSET, \
 | |
|                                 EXCSTR, EXCDOC) \
 | |
| static PyTypeObject _PyExc_ ## EXCNAME = { \
 | |
|     PyVarObject_HEAD_INIT(NULL, 0) \
 | |
|     # EXCNAME, \
 | |
|     sizeof(Py ## EXCSTORE ## Object), 0, \
 | |
|     (destructor)EXCSTORE ## _dealloc, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, \
 | |
|     (reprfunc)EXCSTR, 0, 0, 0, \
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC, \
 | |
|     PyDoc_STR(EXCDOC), (traverseproc)EXCSTORE ## _traverse, \
 | |
|     (inquiry)EXCSTORE ## _clear, 0, 0, 0, 0, EXCMETHODS, \
 | |
|     EXCMEMBERS, EXCGETSET, &_ ## EXCBASE, \
 | |
|     0, 0, 0, offsetof(Py ## EXCSTORE ## Object, dict), \
 | |
|     (initproc)EXCSTORE ## _init, 0, EXCNEW,\
 | |
| }; \
 | |
| PyObject *PyExc_ ## EXCNAME = (PyObject *)&_PyExc_ ## EXCNAME
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    Exception extends BaseException
 | |
|  */
 | |
| SimpleExtendsException(PyExc_BaseException, Exception,
 | |
|                        "Common base class for all non-exit exceptions.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    TypeError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, TypeError,
 | |
|                        "Inappropriate argument type.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    StopAsyncIteration extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, StopAsyncIteration,
 | |
|                        "Signal the end from iterator.__anext__().");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    StopIteration extends Exception
 | |
|  */
 | |
| 
 | |
| static PyMemberDef StopIteration_members[] = {
 | |
|     {"value", T_OBJECT, offsetof(PyStopIterationObject, value), 0,
 | |
|         PyDoc_STR("generator return value")},
 | |
|     {NULL}  /* Sentinel */
 | |
| };
 | |
| 
 | |
| static int
 | |
| StopIteration_init(PyStopIterationObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     Py_ssize_t size = PyTuple_GET_SIZE(args);
 | |
|     PyObject *value;
 | |
| 
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, kwds) == -1)
 | |
|         return -1;
 | |
|     Py_CLEAR(self->value);
 | |
|     if (size > 0)
 | |
|         value = PyTuple_GET_ITEM(args, 0);
 | |
|     else
 | |
|         value = Py_None;
 | |
|     Py_INCREF(value);
 | |
|     self->value = value;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| StopIteration_clear(PyStopIterationObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->value);
 | |
|     return BaseException_clear((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| StopIteration_dealloc(PyStopIterationObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     StopIteration_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| StopIteration_traverse(PyStopIterationObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(self->value);
 | |
|     return BaseException_traverse((PyBaseExceptionObject *)self, visit, arg);
 | |
| }
 | |
| 
 | |
| ComplexExtendsException(
 | |
|     PyExc_Exception,       /* base */
 | |
|     StopIteration,         /* name */
 | |
|     StopIteration,         /* prefix for *_init, etc */
 | |
|     0,                     /* new */
 | |
|     0,                     /* methods */
 | |
|     StopIteration_members, /* members */
 | |
|     0,                     /* getset */
 | |
|     0,                     /* str */
 | |
|     "Signal the end from iterator.__next__()."
 | |
| );
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    GeneratorExit extends BaseException
 | |
|  */
 | |
| SimpleExtendsException(PyExc_BaseException, GeneratorExit,
 | |
|                        "Request that a generator exit.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    SystemExit extends BaseException
 | |
|  */
 | |
| 
 | |
| static int
 | |
| SystemExit_init(PySystemExitObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     Py_ssize_t size = PyTuple_GET_SIZE(args);
 | |
| 
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, kwds) == -1)
 | |
|         return -1;
 | |
| 
 | |
|     if (size == 0)
 | |
|         return 0;
 | |
|     if (size == 1) {
 | |
|         Py_INCREF(PyTuple_GET_ITEM(args, 0));
 | |
|         Py_XSETREF(self->code, PyTuple_GET_ITEM(args, 0));
 | |
|     }
 | |
|     else { /* size > 1 */
 | |
|         Py_INCREF(args);
 | |
|         Py_XSETREF(self->code, args);
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| SystemExit_clear(PySystemExitObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->code);
 | |
|     return BaseException_clear((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| SystemExit_dealloc(PySystemExitObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     SystemExit_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SystemExit_traverse(PySystemExitObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(self->code);
 | |
|     return BaseException_traverse((PyBaseExceptionObject *)self, visit, arg);
 | |
| }
 | |
| 
 | |
| static PyMemberDef SystemExit_members[] = {
 | |
|     {"code", T_OBJECT, offsetof(PySystemExitObject, code), 0,
 | |
|         PyDoc_STR("exception code")},
 | |
|     {NULL}  /* Sentinel */
 | |
| };
 | |
| 
 | |
| ComplexExtendsException(PyExc_BaseException, SystemExit, SystemExit,
 | |
|                         0, 0, SystemExit_members, 0, 0,
 | |
|                         "Request to exit from the interpreter.");
 | |
| 
 | |
| /*
 | |
|  *    KeyboardInterrupt extends BaseException
 | |
|  */
 | |
| SimpleExtendsException(PyExc_BaseException, KeyboardInterrupt,
 | |
|                        "Program interrupted by user.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ImportError extends Exception
 | |
|  */
 | |
| 
 | |
| static int
 | |
| ImportError_init(PyImportErrorObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     static char *kwlist[] = {"name", "path", 0};
 | |
|     PyObject *empty_tuple;
 | |
|     PyObject *msg = NULL;
 | |
|     PyObject *name = NULL;
 | |
|     PyObject *path = NULL;
 | |
| 
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, NULL) == -1)
 | |
|         return -1;
 | |
| 
 | |
|     empty_tuple = PyTuple_New(0);
 | |
|     if (!empty_tuple)
 | |
|         return -1;
 | |
|     if (!PyArg_ParseTupleAndKeywords(empty_tuple, kwds, "|$OO:ImportError", kwlist,
 | |
|                                      &name, &path)) {
 | |
|         Py_DECREF(empty_tuple);
 | |
|         return -1;
 | |
|     }
 | |
|     Py_DECREF(empty_tuple);
 | |
| 
 | |
|     Py_XINCREF(name);
 | |
|     Py_XSETREF(self->name, name);
 | |
| 
 | |
|     Py_XINCREF(path);
 | |
|     Py_XSETREF(self->path, path);
 | |
| 
 | |
|     if (PyTuple_GET_SIZE(args) == 1) {
 | |
|         msg = PyTuple_GET_ITEM(args, 0);
 | |
|         Py_INCREF(msg);
 | |
|     }
 | |
|     Py_XSETREF(self->msg, msg);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ImportError_clear(PyImportErrorObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->msg);
 | |
|     Py_CLEAR(self->name);
 | |
|     Py_CLEAR(self->path);
 | |
|     return BaseException_clear((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| ImportError_dealloc(PyImportErrorObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     ImportError_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| ImportError_traverse(PyImportErrorObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(self->msg);
 | |
|     Py_VISIT(self->name);
 | |
|     Py_VISIT(self->path);
 | |
|     return BaseException_traverse((PyBaseExceptionObject *)self, visit, arg);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| ImportError_str(PyImportErrorObject *self)
 | |
| {
 | |
|     if (self->msg && PyUnicode_CheckExact(self->msg)) {
 | |
|         Py_INCREF(self->msg);
 | |
|         return self->msg;
 | |
|     }
 | |
|     else {
 | |
|         return BaseException_str((PyBaseExceptionObject *)self);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| ImportError_getstate(PyImportErrorObject *self)
 | |
| {
 | |
|     PyObject *dict = ((PyBaseExceptionObject *)self)->dict;
 | |
|     if (self->name || self->path) {
 | |
|         _Py_IDENTIFIER(name);
 | |
|         _Py_IDENTIFIER(path);
 | |
|         dict = dict ? PyDict_Copy(dict) : PyDict_New();
 | |
|         if (dict == NULL)
 | |
|             return NULL;
 | |
|         if (self->name && _PyDict_SetItemId(dict, &PyId_name, self->name) < 0) {
 | |
|             Py_DECREF(dict);
 | |
|             return NULL;
 | |
|         }
 | |
|         if (self->path && _PyDict_SetItemId(dict, &PyId_path, self->path) < 0) {
 | |
|             Py_DECREF(dict);
 | |
|             return NULL;
 | |
|         }
 | |
|         return dict;
 | |
|     }
 | |
|     else if (dict) {
 | |
|         Py_INCREF(dict);
 | |
|         return dict;
 | |
|     }
 | |
|     else {
 | |
|         Py_RETURN_NONE;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Pickling support */
 | |
| static PyObject *
 | |
| ImportError_reduce(PyImportErrorObject *self, PyObject *Py_UNUSED(ignored))
 | |
| {
 | |
|     PyObject *res;
 | |
|     PyObject *args;
 | |
|     PyObject *state = ImportError_getstate(self);
 | |
|     if (state == NULL)
 | |
|         return NULL;
 | |
|     args = ((PyBaseExceptionObject *)self)->args;
 | |
|     if (state == Py_None)
 | |
|         res = PyTuple_Pack(2, Py_TYPE(self), args);
 | |
|     else
 | |
|         res = PyTuple_Pack(3, Py_TYPE(self), args, state);
 | |
|     Py_DECREF(state);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| static PyMemberDef ImportError_members[] = {
 | |
|     {"msg", T_OBJECT, offsetof(PyImportErrorObject, msg), 0,
 | |
|         PyDoc_STR("exception message")},
 | |
|     {"name", T_OBJECT, offsetof(PyImportErrorObject, name), 0,
 | |
|         PyDoc_STR("module name")},
 | |
|     {"path", T_OBJECT, offsetof(PyImportErrorObject, path), 0,
 | |
|         PyDoc_STR("module path")},
 | |
|     {NULL}  /* Sentinel */
 | |
| };
 | |
| 
 | |
| static PyMethodDef ImportError_methods[] = {
 | |
|     {"__reduce__", (PyCFunction)ImportError_reduce, METH_NOARGS},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| ComplexExtendsException(PyExc_Exception, ImportError,
 | |
|                         ImportError, 0 /* new */,
 | |
|                         ImportError_methods, ImportError_members,
 | |
|                         0 /* getset */, ImportError_str,
 | |
|                         "Import can't find module, or can't find name in "
 | |
|                         "module.");
 | |
| 
 | |
| /*
 | |
|  *    ModuleNotFoundError extends ImportError
 | |
|  */
 | |
| 
 | |
| MiddlingExtendsException(PyExc_ImportError, ModuleNotFoundError, ImportError,
 | |
|                          "Module not found.");
 | |
| 
 | |
| /*
 | |
|  *    OSError extends Exception
 | |
|  */
 | |
| 
 | |
| #ifdef MS_WINDOWS
 | |
| #include "errmap.h"
 | |
| #endif
 | |
| 
 | |
| /* Where a function has a single filename, such as open() or some
 | |
|  * of the os module functions, PyErr_SetFromErrnoWithFilename() is
 | |
|  * called, giving a third argument which is the filename.  But, so
 | |
|  * that old code using in-place unpacking doesn't break, e.g.:
 | |
|  *
 | |
|  * except OSError, (errno, strerror):
 | |
|  *
 | |
|  * we hack args so that it only contains two items.  This also
 | |
|  * means we need our own __str__() which prints out the filename
 | |
|  * when it was supplied.
 | |
|  *
 | |
|  * (If a function has two filenames, such as rename(), symlink(),
 | |
|  * or copy(), PyErr_SetFromErrnoWithFilenameObjects() is called,
 | |
|  * which allows passing in a second filename.)
 | |
|  */
 | |
| 
 | |
| /* This function doesn't cleanup on error, the caller should */
 | |
| static int
 | |
| oserror_parse_args(PyObject **p_args,
 | |
|                    PyObject **myerrno, PyObject **strerror,
 | |
|                    PyObject **filename, PyObject **filename2
 | |
| #ifdef MS_WINDOWS
 | |
|                    , PyObject **winerror
 | |
| #endif
 | |
|                   )
 | |
| {
 | |
|     Py_ssize_t nargs;
 | |
|     PyObject *args = *p_args;
 | |
| #ifndef MS_WINDOWS
 | |
|     /*
 | |
|      * ignored on non-Windows platforms,
 | |
|      * but parsed so OSError has a consistent signature
 | |
|      */
 | |
|     PyObject *_winerror = NULL;
 | |
|     PyObject **winerror = &_winerror;
 | |
| #endif /* MS_WINDOWS */
 | |
| 
 | |
|     nargs = PyTuple_GET_SIZE(args);
 | |
| 
 | |
|     if (nargs >= 2 && nargs <= 5) {
 | |
|         if (!PyArg_UnpackTuple(args, "OSError", 2, 5,
 | |
|                                myerrno, strerror,
 | |
|                                filename, winerror, filename2))
 | |
|             return -1;
 | |
| #ifdef MS_WINDOWS
 | |
|         if (*winerror && PyLong_Check(*winerror)) {
 | |
|             long errcode, winerrcode;
 | |
|             PyObject *newargs;
 | |
|             Py_ssize_t i;
 | |
| 
 | |
|             winerrcode = PyLong_AsLong(*winerror);
 | |
|             if (winerrcode == -1 && PyErr_Occurred())
 | |
|                 return -1;
 | |
|             /* Set errno to the corresponding POSIX errno (overriding
 | |
|                first argument).  Windows Socket error codes (>= 10000)
 | |
|                have the same value as their POSIX counterparts.
 | |
|             */
 | |
|             if (winerrcode < 10000)
 | |
|                 errcode = winerror_to_errno(winerrcode);
 | |
|             else
 | |
|                 errcode = winerrcode;
 | |
|             *myerrno = PyLong_FromLong(errcode);
 | |
|             if (!*myerrno)
 | |
|                 return -1;
 | |
|             newargs = PyTuple_New(nargs);
 | |
|             if (!newargs)
 | |
|                 return -1;
 | |
|             PyTuple_SET_ITEM(newargs, 0, *myerrno);
 | |
|             for (i = 1; i < nargs; i++) {
 | |
|                 PyObject *val = PyTuple_GET_ITEM(args, i);
 | |
|                 Py_INCREF(val);
 | |
|                 PyTuple_SET_ITEM(newargs, i, val);
 | |
|             }
 | |
|             Py_DECREF(args);
 | |
|             args = *p_args = newargs;
 | |
|         }
 | |
| #endif /* MS_WINDOWS */
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| oserror_init(PyOSErrorObject *self, PyObject **p_args,
 | |
|              PyObject *myerrno, PyObject *strerror,
 | |
|              PyObject *filename, PyObject *filename2
 | |
| #ifdef MS_WINDOWS
 | |
|              , PyObject *winerror
 | |
| #endif
 | |
|              )
 | |
| {
 | |
|     PyObject *args = *p_args;
 | |
|     Py_ssize_t nargs = PyTuple_GET_SIZE(args);
 | |
| 
 | |
|     /* self->filename will remain Py_None otherwise */
 | |
|     if (filename && filename != Py_None) {
 | |
|         if (Py_IS_TYPE(self, (PyTypeObject *) PyExc_BlockingIOError) &&
 | |
|             PyNumber_Check(filename)) {
 | |
|             /* BlockingIOError's 3rd argument can be the number of
 | |
|              * characters written.
 | |
|              */
 | |
|             self->written = PyNumber_AsSsize_t(filename, PyExc_ValueError);
 | |
|             if (self->written == -1 && PyErr_Occurred())
 | |
|                 return -1;
 | |
|         }
 | |
|         else {
 | |
|             Py_INCREF(filename);
 | |
|             self->filename = filename;
 | |
| 
 | |
|             if (filename2 && filename2 != Py_None) {
 | |
|                 Py_INCREF(filename2);
 | |
|                 self->filename2 = filename2;
 | |
|             }
 | |
| 
 | |
|             if (nargs >= 2 && nargs <= 5) {
 | |
|                 /* filename, filename2, and winerror are removed from the args tuple
 | |
|                    (for compatibility purposes, see test_exceptions.py) */
 | |
|                 PyObject *subslice = PyTuple_GetSlice(args, 0, 2);
 | |
|                 if (!subslice)
 | |
|                     return -1;
 | |
| 
 | |
|                 Py_DECREF(args);  /* replacing args */
 | |
|                 *p_args = args = subslice;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     Py_XINCREF(myerrno);
 | |
|     self->myerrno = myerrno;
 | |
| 
 | |
|     Py_XINCREF(strerror);
 | |
|     self->strerror = strerror;
 | |
| 
 | |
| #ifdef MS_WINDOWS
 | |
|     Py_XINCREF(winerror);
 | |
|     self->winerror = winerror;
 | |
| #endif
 | |
| 
 | |
|     /* Steals the reference to args */
 | |
|     Py_XSETREF(self->args, args);
 | |
|     *p_args = args = NULL;
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| OSError_new(PyTypeObject *type, PyObject *args, PyObject *kwds);
 | |
| static int
 | |
| OSError_init(PyOSErrorObject *self, PyObject *args, PyObject *kwds);
 | |
| 
 | |
| static int
 | |
| oserror_use_init(PyTypeObject *type)
 | |
| {
 | |
|     /* When __init__ is defined in an OSError subclass, we want any
 | |
|        extraneous argument to __new__ to be ignored.  The only reasonable
 | |
|        solution, given __new__ takes a variable number of arguments,
 | |
|        is to defer arg parsing and initialization to __init__.
 | |
| 
 | |
|        But when __new__ is overridden as well, it should call our __new__
 | |
|        with the right arguments.
 | |
| 
 | |
|        (see http://bugs.python.org/issue12555#msg148829 )
 | |
|     */
 | |
|     if (type->tp_init != (initproc) OSError_init &&
 | |
|         type->tp_new == (newfunc) OSError_new) {
 | |
|         assert((PyObject *) type != PyExc_OSError);
 | |
|         return 1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| OSError_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyOSErrorObject *self = NULL;
 | |
|     PyObject *myerrno = NULL, *strerror = NULL;
 | |
|     PyObject *filename = NULL, *filename2 = NULL;
 | |
| #ifdef MS_WINDOWS
 | |
|     PyObject *winerror = NULL;
 | |
| #endif
 | |
| 
 | |
|     Py_INCREF(args);
 | |
| 
 | |
|     if (!oserror_use_init(type)) {
 | |
|         if (!_PyArg_NoKeywords(type->tp_name, kwds))
 | |
|             goto error;
 | |
| 
 | |
|         if (oserror_parse_args(&args, &myerrno, &strerror,
 | |
|                                &filename, &filename2
 | |
| #ifdef MS_WINDOWS
 | |
|                                , &winerror
 | |
| #endif
 | |
|             ))
 | |
|             goto error;
 | |
| 
 | |
|         if (myerrno && PyLong_Check(myerrno) &&
 | |
|             errnomap && (PyObject *) type == PyExc_OSError) {
 | |
|             PyObject *newtype;
 | |
|             newtype = PyDict_GetItemWithError(errnomap, myerrno);
 | |
|             if (newtype) {
 | |
|                 assert(PyType_Check(newtype));
 | |
|                 type = (PyTypeObject *) newtype;
 | |
|             }
 | |
|             else if (PyErr_Occurred())
 | |
|                 goto error;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     self = (PyOSErrorObject *) type->tp_alloc(type, 0);
 | |
|     if (!self)
 | |
|         goto error;
 | |
| 
 | |
|     self->dict = NULL;
 | |
|     self->traceback = self->cause = self->context = NULL;
 | |
|     self->written = -1;
 | |
| 
 | |
|     if (!oserror_use_init(type)) {
 | |
|         if (oserror_init(self, &args, myerrno, strerror, filename, filename2
 | |
| #ifdef MS_WINDOWS
 | |
|                          , winerror
 | |
| #endif
 | |
|             ))
 | |
|             goto error;
 | |
|     }
 | |
|     else {
 | |
|         self->args = PyTuple_New(0);
 | |
|         if (self->args == NULL)
 | |
|             goto error;
 | |
|     }
 | |
| 
 | |
|     Py_XDECREF(args);
 | |
|     return (PyObject *) self;
 | |
| 
 | |
| error:
 | |
|     Py_XDECREF(args);
 | |
|     Py_XDECREF(self);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| static int
 | |
| OSError_init(PyOSErrorObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyObject *myerrno = NULL, *strerror = NULL;
 | |
|     PyObject *filename = NULL, *filename2 = NULL;
 | |
| #ifdef MS_WINDOWS
 | |
|     PyObject *winerror = NULL;
 | |
| #endif
 | |
| 
 | |
|     if (!oserror_use_init(Py_TYPE(self)))
 | |
|         /* Everything already done in OSError_new */
 | |
|         return 0;
 | |
| 
 | |
|     if (!_PyArg_NoKeywords(Py_TYPE(self)->tp_name, kwds))
 | |
|         return -1;
 | |
| 
 | |
|     Py_INCREF(args);
 | |
|     if (oserror_parse_args(&args, &myerrno, &strerror, &filename, &filename2
 | |
| #ifdef MS_WINDOWS
 | |
|                            , &winerror
 | |
| #endif
 | |
|         ))
 | |
|         goto error;
 | |
| 
 | |
|     if (oserror_init(self, &args, myerrno, strerror, filename, filename2
 | |
| #ifdef MS_WINDOWS
 | |
|                      , winerror
 | |
| #endif
 | |
|         ))
 | |
|         goto error;
 | |
| 
 | |
|     return 0;
 | |
| 
 | |
| error:
 | |
|     Py_DECREF(args);
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| static int
 | |
| OSError_clear(PyOSErrorObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->myerrno);
 | |
|     Py_CLEAR(self->strerror);
 | |
|     Py_CLEAR(self->filename);
 | |
|     Py_CLEAR(self->filename2);
 | |
| #ifdef MS_WINDOWS
 | |
|     Py_CLEAR(self->winerror);
 | |
| #endif
 | |
|     return BaseException_clear((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| OSError_dealloc(PyOSErrorObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     OSError_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| OSError_traverse(PyOSErrorObject *self, visitproc visit,
 | |
|         void *arg)
 | |
| {
 | |
|     Py_VISIT(self->myerrno);
 | |
|     Py_VISIT(self->strerror);
 | |
|     Py_VISIT(self->filename);
 | |
|     Py_VISIT(self->filename2);
 | |
| #ifdef MS_WINDOWS
 | |
|     Py_VISIT(self->winerror);
 | |
| #endif
 | |
|     return BaseException_traverse((PyBaseExceptionObject *)self, visit, arg);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| OSError_str(PyOSErrorObject *self)
 | |
| {
 | |
| #define OR_NONE(x) ((x)?(x):Py_None)
 | |
| #ifdef MS_WINDOWS
 | |
|     /* If available, winerror has the priority over myerrno */
 | |
|     if (self->winerror && self->filename) {
 | |
|         if (self->filename2) {
 | |
|             return PyUnicode_FromFormat("[WinError %S] %S: %R -> %R",
 | |
|                                         OR_NONE(self->winerror),
 | |
|                                         OR_NONE(self->strerror),
 | |
|                                         self->filename,
 | |
|                                         self->filename2);
 | |
|         } else {
 | |
|             return PyUnicode_FromFormat("[WinError %S] %S: %R",
 | |
|                                         OR_NONE(self->winerror),
 | |
|                                         OR_NONE(self->strerror),
 | |
|                                         self->filename);
 | |
|         }
 | |
|     }
 | |
|     if (self->winerror && self->strerror)
 | |
|         return PyUnicode_FromFormat("[WinError %S] %S",
 | |
|                                     self->winerror ? self->winerror: Py_None,
 | |
|                                     self->strerror ? self->strerror: Py_None);
 | |
| #endif
 | |
|     if (self->filename) {
 | |
|         if (self->filename2) {
 | |
|             return PyUnicode_FromFormat("[Errno %S] %S: %R -> %R",
 | |
|                                         OR_NONE(self->myerrno),
 | |
|                                         OR_NONE(self->strerror),
 | |
|                                         self->filename,
 | |
|                                         self->filename2);
 | |
|         } else {
 | |
|             return PyUnicode_FromFormat("[Errno %S] %S: %R",
 | |
|                                         OR_NONE(self->myerrno),
 | |
|                                         OR_NONE(self->strerror),
 | |
|                                         self->filename);
 | |
|         }
 | |
|     }
 | |
|     if (self->myerrno && self->strerror)
 | |
|         return PyUnicode_FromFormat("[Errno %S] %S",
 | |
|                                     self->myerrno, self->strerror);
 | |
|     return BaseException_str((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| OSError_reduce(PyOSErrorObject *self, PyObject *Py_UNUSED(ignored))
 | |
| {
 | |
|     PyObject *args = self->args;
 | |
|     PyObject *res = NULL, *tmp;
 | |
| 
 | |
|     /* self->args is only the first two real arguments if there was a
 | |
|      * file name given to OSError. */
 | |
|     if (PyTuple_GET_SIZE(args) == 2 && self->filename) {
 | |
|         Py_ssize_t size = self->filename2 ? 5 : 3;
 | |
|         args = PyTuple_New(size);
 | |
|         if (!args)
 | |
|             return NULL;
 | |
| 
 | |
|         tmp = PyTuple_GET_ITEM(self->args, 0);
 | |
|         Py_INCREF(tmp);
 | |
|         PyTuple_SET_ITEM(args, 0, tmp);
 | |
| 
 | |
|         tmp = PyTuple_GET_ITEM(self->args, 1);
 | |
|         Py_INCREF(tmp);
 | |
|         PyTuple_SET_ITEM(args, 1, tmp);
 | |
| 
 | |
|         Py_INCREF(self->filename);
 | |
|         PyTuple_SET_ITEM(args, 2, self->filename);
 | |
| 
 | |
|         if (self->filename2) {
 | |
|             /*
 | |
|              * This tuple is essentially used as OSError(*args).
 | |
|              * So, to recreate filename2, we need to pass in
 | |
|              * winerror as well.
 | |
|              */
 | |
|             Py_INCREF(Py_None);
 | |
|             PyTuple_SET_ITEM(args, 3, Py_None);
 | |
| 
 | |
|             /* filename2 */
 | |
|             Py_INCREF(self->filename2);
 | |
|             PyTuple_SET_ITEM(args, 4, self->filename2);
 | |
|         }
 | |
|     } else
 | |
|         Py_INCREF(args);
 | |
| 
 | |
|     if (self->dict)
 | |
|         res = PyTuple_Pack(3, Py_TYPE(self), args, self->dict);
 | |
|     else
 | |
|         res = PyTuple_Pack(2, Py_TYPE(self), args);
 | |
|     Py_DECREF(args);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| OSError_written_get(PyOSErrorObject *self, void *context)
 | |
| {
 | |
|     if (self->written == -1) {
 | |
|         PyErr_SetString(PyExc_AttributeError, "characters_written");
 | |
|         return NULL;
 | |
|     }
 | |
|     return PyLong_FromSsize_t(self->written);
 | |
| }
 | |
| 
 | |
| static int
 | |
| OSError_written_set(PyOSErrorObject *self, PyObject *arg, void *context)
 | |
| {
 | |
|     if (arg == NULL) {
 | |
|         if (self->written == -1) {
 | |
|             PyErr_SetString(PyExc_AttributeError, "characters_written");
 | |
|             return -1;
 | |
|         }
 | |
|         self->written = -1;
 | |
|         return 0;
 | |
|     }
 | |
|     Py_ssize_t n;
 | |
|     n = PyNumber_AsSsize_t(arg, PyExc_ValueError);
 | |
|     if (n == -1 && PyErr_Occurred())
 | |
|         return -1;
 | |
|     self->written = n;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyMemberDef OSError_members[] = {
 | |
|     {"errno", T_OBJECT, offsetof(PyOSErrorObject, myerrno), 0,
 | |
|         PyDoc_STR("POSIX exception code")},
 | |
|     {"strerror", T_OBJECT, offsetof(PyOSErrorObject, strerror), 0,
 | |
|         PyDoc_STR("exception strerror")},
 | |
|     {"filename", T_OBJECT, offsetof(PyOSErrorObject, filename), 0,
 | |
|         PyDoc_STR("exception filename")},
 | |
|     {"filename2", T_OBJECT, offsetof(PyOSErrorObject, filename2), 0,
 | |
|         PyDoc_STR("second exception filename")},
 | |
| #ifdef MS_WINDOWS
 | |
|     {"winerror", T_OBJECT, offsetof(PyOSErrorObject, winerror), 0,
 | |
|         PyDoc_STR("Win32 exception code")},
 | |
| #endif
 | |
|     {NULL}  /* Sentinel */
 | |
| };
 | |
| 
 | |
| static PyMethodDef OSError_methods[] = {
 | |
|     {"__reduce__", (PyCFunction)OSError_reduce, METH_NOARGS},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| static PyGetSetDef OSError_getset[] = {
 | |
|     {"characters_written", (getter) OSError_written_get,
 | |
|                            (setter) OSError_written_set, NULL},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| 
 | |
| ComplexExtendsException(PyExc_Exception, OSError,
 | |
|                         OSError, OSError_new,
 | |
|                         OSError_methods, OSError_members, OSError_getset,
 | |
|                         OSError_str,
 | |
|                         "Base class for I/O related errors.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    Various OSError subclasses
 | |
|  */
 | |
| MiddlingExtendsException(PyExc_OSError, BlockingIOError, OSError,
 | |
|                          "I/O operation would block.");
 | |
| MiddlingExtendsException(PyExc_OSError, ConnectionError, OSError,
 | |
|                          "Connection error.");
 | |
| MiddlingExtendsException(PyExc_OSError, ChildProcessError, OSError,
 | |
|                          "Child process error.");
 | |
| MiddlingExtendsException(PyExc_ConnectionError, BrokenPipeError, OSError,
 | |
|                          "Broken pipe.");
 | |
| MiddlingExtendsException(PyExc_ConnectionError, ConnectionAbortedError, OSError,
 | |
|                          "Connection aborted.");
 | |
| MiddlingExtendsException(PyExc_ConnectionError, ConnectionRefusedError, OSError,
 | |
|                          "Connection refused.");
 | |
| MiddlingExtendsException(PyExc_ConnectionError, ConnectionResetError, OSError,
 | |
|                          "Connection reset.");
 | |
| MiddlingExtendsException(PyExc_OSError, FileExistsError, OSError,
 | |
|                          "File already exists.");
 | |
| MiddlingExtendsException(PyExc_OSError, FileNotFoundError, OSError,
 | |
|                          "File not found.");
 | |
| MiddlingExtendsException(PyExc_OSError, IsADirectoryError, OSError,
 | |
|                          "Operation doesn't work on directories.");
 | |
| MiddlingExtendsException(PyExc_OSError, NotADirectoryError, OSError,
 | |
|                          "Operation only works on directories.");
 | |
| MiddlingExtendsException(PyExc_OSError, InterruptedError, OSError,
 | |
|                          "Interrupted by signal.");
 | |
| MiddlingExtendsException(PyExc_OSError, PermissionError, OSError,
 | |
|                          "Not enough permissions.");
 | |
| MiddlingExtendsException(PyExc_OSError, ProcessLookupError, OSError,
 | |
|                          "Process not found.");
 | |
| MiddlingExtendsException(PyExc_OSError, TimeoutError, OSError,
 | |
|                          "Timeout expired.");
 | |
| 
 | |
| /*
 | |
|  *    EOFError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, EOFError,
 | |
|                        "Read beyond end of file.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    RuntimeError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, RuntimeError,
 | |
|                        "Unspecified run-time error.");
 | |
| 
 | |
| /*
 | |
|  *    RecursionError extends RuntimeError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_RuntimeError, RecursionError,
 | |
|                        "Recursion limit exceeded.");
 | |
| 
 | |
| /*
 | |
|  *    NotImplementedError extends RuntimeError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_RuntimeError, NotImplementedError,
 | |
|                        "Method or function hasn't been implemented yet.");
 | |
| 
 | |
| /*
 | |
|  *    NameError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, NameError,
 | |
|                        "Name not found globally.");
 | |
| 
 | |
| /*
 | |
|  *    UnboundLocalError extends NameError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_NameError, UnboundLocalError,
 | |
|                        "Local name referenced but not bound to a value.");
 | |
| 
 | |
| /*
 | |
|  *    AttributeError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, AttributeError,
 | |
|                        "Attribute not found.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    SyntaxError extends Exception
 | |
|  */
 | |
| 
 | |
| /* Helper function to customize error message for some syntax errors */
 | |
| static int _report_missing_parentheses(PySyntaxErrorObject *self);
 | |
| 
 | |
| static int
 | |
| SyntaxError_init(PySyntaxErrorObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyObject *info = NULL;
 | |
|     Py_ssize_t lenargs = PyTuple_GET_SIZE(args);
 | |
| 
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, kwds) == -1)
 | |
|         return -1;
 | |
| 
 | |
|     if (lenargs >= 1) {
 | |
|         Py_INCREF(PyTuple_GET_ITEM(args, 0));
 | |
|         Py_XSETREF(self->msg, PyTuple_GET_ITEM(args, 0));
 | |
|     }
 | |
|     if (lenargs == 2) {
 | |
|         info = PyTuple_GET_ITEM(args, 1);
 | |
|         info = PySequence_Tuple(info);
 | |
|         if (!info)
 | |
|             return -1;
 | |
| 
 | |
|         if (PyTuple_GET_SIZE(info) != 4) {
 | |
|             /* not a very good error message, but it's what Python 2.4 gives */
 | |
|             PyErr_SetString(PyExc_IndexError, "tuple index out of range");
 | |
|             Py_DECREF(info);
 | |
|             return -1;
 | |
|         }
 | |
| 
 | |
|         Py_INCREF(PyTuple_GET_ITEM(info, 0));
 | |
|         Py_XSETREF(self->filename, PyTuple_GET_ITEM(info, 0));
 | |
| 
 | |
|         Py_INCREF(PyTuple_GET_ITEM(info, 1));
 | |
|         Py_XSETREF(self->lineno, PyTuple_GET_ITEM(info, 1));
 | |
| 
 | |
|         Py_INCREF(PyTuple_GET_ITEM(info, 2));
 | |
|         Py_XSETREF(self->offset, PyTuple_GET_ITEM(info, 2));
 | |
| 
 | |
|         Py_INCREF(PyTuple_GET_ITEM(info, 3));
 | |
|         Py_XSETREF(self->text, PyTuple_GET_ITEM(info, 3));
 | |
| 
 | |
|         Py_DECREF(info);
 | |
| 
 | |
|         /*
 | |
|          * Issue #21669: Custom error for 'print' & 'exec' as statements
 | |
|          *
 | |
|          * Only applies to SyntaxError instances, not to subclasses such
 | |
|          * as TabError or IndentationError (see issue #31161)
 | |
|          */
 | |
|         if (Py_IS_TYPE(self, (PyTypeObject *)PyExc_SyntaxError) &&
 | |
|                 self->text && PyUnicode_Check(self->text) &&
 | |
|                 _report_missing_parentheses(self) < 0) {
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| SyntaxError_clear(PySyntaxErrorObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->msg);
 | |
|     Py_CLEAR(self->filename);
 | |
|     Py_CLEAR(self->lineno);
 | |
|     Py_CLEAR(self->offset);
 | |
|     Py_CLEAR(self->text);
 | |
|     Py_CLEAR(self->print_file_and_line);
 | |
|     return BaseException_clear((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| SyntaxError_dealloc(PySyntaxErrorObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     SyntaxError_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SyntaxError_traverse(PySyntaxErrorObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(self->msg);
 | |
|     Py_VISIT(self->filename);
 | |
|     Py_VISIT(self->lineno);
 | |
|     Py_VISIT(self->offset);
 | |
|     Py_VISIT(self->text);
 | |
|     Py_VISIT(self->print_file_and_line);
 | |
|     return BaseException_traverse((PyBaseExceptionObject *)self, visit, arg);
 | |
| }
 | |
| 
 | |
| /* This is called "my_basename" instead of just "basename" to avoid name
 | |
|    conflicts with glibc; basename is already prototyped if _GNU_SOURCE is
 | |
|    defined, and Python does define that. */
 | |
| static PyObject*
 | |
| my_basename(PyObject *name)
 | |
| {
 | |
|     Py_ssize_t i, size, offset;
 | |
|     int kind;
 | |
|     const void *data;
 | |
| 
 | |
|     if (PyUnicode_READY(name))
 | |
|         return NULL;
 | |
|     kind = PyUnicode_KIND(name);
 | |
|     data = PyUnicode_DATA(name);
 | |
|     size = PyUnicode_GET_LENGTH(name);
 | |
|     offset = 0;
 | |
|     for(i=0; i < size; i++) {
 | |
|         if (PyUnicode_READ(kind, data, i) == SEP) {
 | |
|             offset = i + 1;
 | |
|         }
 | |
|     }
 | |
|     if (offset != 0) {
 | |
|         return PyUnicode_Substring(name, offset, size);
 | |
|     }
 | |
|     else {
 | |
|         Py_INCREF(name);
 | |
|         return name;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| SyntaxError_str(PySyntaxErrorObject *self)
 | |
| {
 | |
|     int have_lineno = 0;
 | |
|     PyObject *filename;
 | |
|     PyObject *result;
 | |
|     /* Below, we always ignore overflow errors, just printing -1.
 | |
|        Still, we cannot allow an OverflowError to be raised, so
 | |
|        we need to call PyLong_AsLongAndOverflow. */
 | |
|     int overflow;
 | |
| 
 | |
|     /* XXX -- do all the additional formatting with filename and
 | |
|        lineno here */
 | |
| 
 | |
|     if (self->filename && PyUnicode_Check(self->filename)) {
 | |
|         filename = my_basename(self->filename);
 | |
|         if (filename == NULL)
 | |
|             return NULL;
 | |
|     } else {
 | |
|         filename = NULL;
 | |
|     }
 | |
|     have_lineno = (self->lineno != NULL) && PyLong_CheckExact(self->lineno);
 | |
| 
 | |
|     if (!filename && !have_lineno)
 | |
|         return PyObject_Str(self->msg ? self->msg : Py_None);
 | |
| 
 | |
|     if (filename && have_lineno)
 | |
|         result = PyUnicode_FromFormat("%S (%U, line %ld)",
 | |
|                    self->msg ? self->msg : Py_None,
 | |
|                    filename,
 | |
|                    PyLong_AsLongAndOverflow(self->lineno, &overflow));
 | |
|     else if (filename)
 | |
|         result = PyUnicode_FromFormat("%S (%U)",
 | |
|                    self->msg ? self->msg : Py_None,
 | |
|                    filename);
 | |
|     else /* only have_lineno */
 | |
|         result = PyUnicode_FromFormat("%S (line %ld)",
 | |
|                    self->msg ? self->msg : Py_None,
 | |
|                    PyLong_AsLongAndOverflow(self->lineno, &overflow));
 | |
|     Py_XDECREF(filename);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static PyMemberDef SyntaxError_members[] = {
 | |
|     {"msg", T_OBJECT, offsetof(PySyntaxErrorObject, msg), 0,
 | |
|         PyDoc_STR("exception msg")},
 | |
|     {"filename", T_OBJECT, offsetof(PySyntaxErrorObject, filename), 0,
 | |
|         PyDoc_STR("exception filename")},
 | |
|     {"lineno", T_OBJECT, offsetof(PySyntaxErrorObject, lineno), 0,
 | |
|         PyDoc_STR("exception lineno")},
 | |
|     {"offset", T_OBJECT, offsetof(PySyntaxErrorObject, offset), 0,
 | |
|         PyDoc_STR("exception offset")},
 | |
|     {"text", T_OBJECT, offsetof(PySyntaxErrorObject, text), 0,
 | |
|         PyDoc_STR("exception text")},
 | |
|     {"print_file_and_line", T_OBJECT,
 | |
|         offsetof(PySyntaxErrorObject, print_file_and_line), 0,
 | |
|         PyDoc_STR("exception print_file_and_line")},
 | |
|     {NULL}  /* Sentinel */
 | |
| };
 | |
| 
 | |
| ComplexExtendsException(PyExc_Exception, SyntaxError, SyntaxError,
 | |
|                         0, 0, SyntaxError_members, 0,
 | |
|                         SyntaxError_str, "Invalid syntax.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    IndentationError extends SyntaxError
 | |
|  */
 | |
| MiddlingExtendsException(PyExc_SyntaxError, IndentationError, SyntaxError,
 | |
|                          "Improper indentation.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    TabError extends IndentationError
 | |
|  */
 | |
| MiddlingExtendsException(PyExc_IndentationError, TabError, SyntaxError,
 | |
|                          "Improper mixture of spaces and tabs.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    LookupError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, LookupError,
 | |
|                        "Base class for lookup errors.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    IndexError extends LookupError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_LookupError, IndexError,
 | |
|                        "Sequence index out of range.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    KeyError extends LookupError
 | |
|  */
 | |
| static PyObject *
 | |
| KeyError_str(PyBaseExceptionObject *self)
 | |
| {
 | |
|     /* If args is a tuple of exactly one item, apply repr to args[0].
 | |
|        This is done so that e.g. the exception raised by {}[''] prints
 | |
|          KeyError: ''
 | |
|        rather than the confusing
 | |
|          KeyError
 | |
|        alone.  The downside is that if KeyError is raised with an explanatory
 | |
|        string, that string will be displayed in quotes.  Too bad.
 | |
|        If args is anything else, use the default BaseException__str__().
 | |
|     */
 | |
|     if (PyTuple_GET_SIZE(self->args) == 1) {
 | |
|         return PyObject_Repr(PyTuple_GET_ITEM(self->args, 0));
 | |
|     }
 | |
|     return BaseException_str(self);
 | |
| }
 | |
| 
 | |
| ComplexExtendsException(PyExc_LookupError, KeyError, BaseException,
 | |
|                         0, 0, 0, 0, KeyError_str, "Mapping key not found.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ValueError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, ValueError,
 | |
|                        "Inappropriate argument value (of correct type).");
 | |
| 
 | |
| /*
 | |
|  *    UnicodeError extends ValueError
 | |
|  */
 | |
| 
 | |
| SimpleExtendsException(PyExc_ValueError, UnicodeError,
 | |
|                        "Unicode related error.");
 | |
| 
 | |
| static PyObject *
 | |
| get_string(PyObject *attr, const char *name)
 | |
| {
 | |
|     if (!attr) {
 | |
|         PyErr_Format(PyExc_TypeError, "%.200s attribute not set", name);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!PyBytes_Check(attr)) {
 | |
|         PyErr_Format(PyExc_TypeError, "%.200s attribute must be bytes", name);
 | |
|         return NULL;
 | |
|     }
 | |
|     Py_INCREF(attr);
 | |
|     return attr;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| get_unicode(PyObject *attr, const char *name)
 | |
| {
 | |
|     if (!attr) {
 | |
|         PyErr_Format(PyExc_TypeError, "%.200s attribute not set", name);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!PyUnicode_Check(attr)) {
 | |
|         PyErr_Format(PyExc_TypeError,
 | |
|                      "%.200s attribute must be unicode", name);
 | |
|         return NULL;
 | |
|     }
 | |
|     Py_INCREF(attr);
 | |
|     return attr;
 | |
| }
 | |
| 
 | |
| static int
 | |
| set_unicodefromstring(PyObject **attr, const char *value)
 | |
| {
 | |
|     PyObject *obj = PyUnicode_FromString(value);
 | |
|     if (!obj)
 | |
|         return -1;
 | |
|     Py_XSETREF(*attr, obj);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeEncodeError_GetEncoding(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->encoding, "encoding");
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeDecodeError_GetEncoding(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->encoding, "encoding");
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeEncodeError_GetObject(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->object, "object");
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeDecodeError_GetObject(PyObject *exc)
 | |
| {
 | |
|     return get_string(((PyUnicodeErrorObject *)exc)->object, "object");
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeTranslateError_GetObject(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->object, "object");
 | |
| }
 | |
| 
 | |
| int
 | |
| PyUnicodeEncodeError_GetStart(PyObject *exc, Py_ssize_t *start)
 | |
| {
 | |
|     Py_ssize_t size;
 | |
|     PyObject *obj = get_unicode(((PyUnicodeErrorObject *)exc)->object,
 | |
|                                 "object");
 | |
|     if (!obj)
 | |
|         return -1;
 | |
|     *start = ((PyUnicodeErrorObject *)exc)->start;
 | |
|     size = PyUnicode_GET_LENGTH(obj);
 | |
|     if (*start<0)
 | |
|         *start = 0; /*XXX check for values <0*/
 | |
|     if (*start>=size)
 | |
|         *start = size-1;
 | |
|     Py_DECREF(obj);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeDecodeError_GetStart(PyObject *exc, Py_ssize_t *start)
 | |
| {
 | |
|     Py_ssize_t size;
 | |
|     PyObject *obj = get_string(((PyUnicodeErrorObject *)exc)->object, "object");
 | |
|     if (!obj)
 | |
|         return -1;
 | |
|     size = PyBytes_GET_SIZE(obj);
 | |
|     *start = ((PyUnicodeErrorObject *)exc)->start;
 | |
|     if (*start<0)
 | |
|         *start = 0;
 | |
|     if (*start>=size)
 | |
|         *start = size-1;
 | |
|     Py_DECREF(obj);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeTranslateError_GetStart(PyObject *exc, Py_ssize_t *start)
 | |
| {
 | |
|     return PyUnicodeEncodeError_GetStart(exc, start);
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeEncodeError_SetStart(PyObject *exc, Py_ssize_t start)
 | |
| {
 | |
|     ((PyUnicodeErrorObject *)exc)->start = start;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeDecodeError_SetStart(PyObject *exc, Py_ssize_t start)
 | |
| {
 | |
|     ((PyUnicodeErrorObject *)exc)->start = start;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeTranslateError_SetStart(PyObject *exc, Py_ssize_t start)
 | |
| {
 | |
|     ((PyUnicodeErrorObject *)exc)->start = start;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeEncodeError_GetEnd(PyObject *exc, Py_ssize_t *end)
 | |
| {
 | |
|     Py_ssize_t size;
 | |
|     PyObject *obj = get_unicode(((PyUnicodeErrorObject *)exc)->object,
 | |
|                                 "object");
 | |
|     if (!obj)
 | |
|         return -1;
 | |
|     *end = ((PyUnicodeErrorObject *)exc)->end;
 | |
|     size = PyUnicode_GET_LENGTH(obj);
 | |
|     if (*end<1)
 | |
|         *end = 1;
 | |
|     if (*end>size)
 | |
|         *end = size;
 | |
|     Py_DECREF(obj);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeDecodeError_GetEnd(PyObject *exc, Py_ssize_t *end)
 | |
| {
 | |
|     Py_ssize_t size;
 | |
|     PyObject *obj = get_string(((PyUnicodeErrorObject *)exc)->object, "object");
 | |
|     if (!obj)
 | |
|         return -1;
 | |
|     size = PyBytes_GET_SIZE(obj);
 | |
|     *end = ((PyUnicodeErrorObject *)exc)->end;
 | |
|     if (*end<1)
 | |
|         *end = 1;
 | |
|     if (*end>size)
 | |
|         *end = size;
 | |
|     Py_DECREF(obj);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeTranslateError_GetEnd(PyObject *exc, Py_ssize_t *end)
 | |
| {
 | |
|     return PyUnicodeEncodeError_GetEnd(exc, end);
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeEncodeError_SetEnd(PyObject *exc, Py_ssize_t end)
 | |
| {
 | |
|     ((PyUnicodeErrorObject *)exc)->end = end;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeDecodeError_SetEnd(PyObject *exc, Py_ssize_t end)
 | |
| {
 | |
|     ((PyUnicodeErrorObject *)exc)->end = end;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeTranslateError_SetEnd(PyObject *exc, Py_ssize_t end)
 | |
| {
 | |
|     ((PyUnicodeErrorObject *)exc)->end = end;
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeEncodeError_GetReason(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->reason, "reason");
 | |
| }
 | |
| 
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeDecodeError_GetReason(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->reason, "reason");
 | |
| }
 | |
| 
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeTranslateError_GetReason(PyObject *exc)
 | |
| {
 | |
|     return get_unicode(((PyUnicodeErrorObject *)exc)->reason, "reason");
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeEncodeError_SetReason(PyObject *exc, const char *reason)
 | |
| {
 | |
|     return set_unicodefromstring(&((PyUnicodeErrorObject *)exc)->reason,
 | |
|                                  reason);
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeDecodeError_SetReason(PyObject *exc, const char *reason)
 | |
| {
 | |
|     return set_unicodefromstring(&((PyUnicodeErrorObject *)exc)->reason,
 | |
|                                  reason);
 | |
| }
 | |
| 
 | |
| 
 | |
| int
 | |
| PyUnicodeTranslateError_SetReason(PyObject *exc, const char *reason)
 | |
| {
 | |
|     return set_unicodefromstring(&((PyUnicodeErrorObject *)exc)->reason,
 | |
|                                  reason);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| UnicodeError_clear(PyUnicodeErrorObject *self)
 | |
| {
 | |
|     Py_CLEAR(self->encoding);
 | |
|     Py_CLEAR(self->object);
 | |
|     Py_CLEAR(self->reason);
 | |
|     return BaseException_clear((PyBaseExceptionObject *)self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| UnicodeError_dealloc(PyUnicodeErrorObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     UnicodeError_clear(self);
 | |
|     Py_TYPE(self)->tp_free((PyObject *)self);
 | |
| }
 | |
| 
 | |
| static int
 | |
| UnicodeError_traverse(PyUnicodeErrorObject *self, visitproc visit, void *arg)
 | |
| {
 | |
|     Py_VISIT(self->encoding);
 | |
|     Py_VISIT(self->object);
 | |
|     Py_VISIT(self->reason);
 | |
|     return BaseException_traverse((PyBaseExceptionObject *)self, visit, arg);
 | |
| }
 | |
| 
 | |
| static PyMemberDef UnicodeError_members[] = {
 | |
|     {"encoding", T_OBJECT, offsetof(PyUnicodeErrorObject, encoding), 0,
 | |
|         PyDoc_STR("exception encoding")},
 | |
|     {"object", T_OBJECT, offsetof(PyUnicodeErrorObject, object), 0,
 | |
|         PyDoc_STR("exception object")},
 | |
|     {"start", T_PYSSIZET, offsetof(PyUnicodeErrorObject, start), 0,
 | |
|         PyDoc_STR("exception start")},
 | |
|     {"end", T_PYSSIZET, offsetof(PyUnicodeErrorObject, end), 0,
 | |
|         PyDoc_STR("exception end")},
 | |
|     {"reason", T_OBJECT, offsetof(PyUnicodeErrorObject, reason), 0,
 | |
|         PyDoc_STR("exception reason")},
 | |
|     {NULL}  /* Sentinel */
 | |
| };
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    UnicodeEncodeError extends UnicodeError
 | |
|  */
 | |
| 
 | |
| static int
 | |
| UnicodeEncodeError_init(PyObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyUnicodeErrorObject *err;
 | |
| 
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, kwds) == -1)
 | |
|         return -1;
 | |
| 
 | |
|     err = (PyUnicodeErrorObject *)self;
 | |
| 
 | |
|     Py_CLEAR(err->encoding);
 | |
|     Py_CLEAR(err->object);
 | |
|     Py_CLEAR(err->reason);
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "UUnnU",
 | |
|                           &err->encoding, &err->object,
 | |
|                           &err->start, &err->end, &err->reason)) {
 | |
|         err->encoding = err->object = err->reason = NULL;
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     Py_INCREF(err->encoding);
 | |
|     Py_INCREF(err->object);
 | |
|     Py_INCREF(err->reason);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| UnicodeEncodeError_str(PyObject *self)
 | |
| {
 | |
|     PyUnicodeErrorObject *uself = (PyUnicodeErrorObject *)self;
 | |
|     PyObject *result = NULL;
 | |
|     PyObject *reason_str = NULL;
 | |
|     PyObject *encoding_str = NULL;
 | |
| 
 | |
|     if (!uself->object)
 | |
|         /* Not properly initialized. */
 | |
|         return PyUnicode_FromString("");
 | |
| 
 | |
|     /* Get reason and encoding as strings, which they might not be if
 | |
|        they've been modified after we were constructed. */
 | |
|     reason_str = PyObject_Str(uself->reason);
 | |
|     if (reason_str == NULL)
 | |
|         goto done;
 | |
|     encoding_str = PyObject_Str(uself->encoding);
 | |
|     if (encoding_str == NULL)
 | |
|         goto done;
 | |
| 
 | |
|     if (uself->start < PyUnicode_GET_LENGTH(uself->object) && uself->end == uself->start+1) {
 | |
|         Py_UCS4 badchar = PyUnicode_ReadChar(uself->object, uself->start);
 | |
|         const char *fmt;
 | |
|         if (badchar <= 0xff)
 | |
|             fmt = "'%U' codec can't encode character '\\x%02x' in position %zd: %U";
 | |
|         else if (badchar <= 0xffff)
 | |
|             fmt = "'%U' codec can't encode character '\\u%04x' in position %zd: %U";
 | |
|         else
 | |
|             fmt = "'%U' codec can't encode character '\\U%08x' in position %zd: %U";
 | |
|         result = PyUnicode_FromFormat(
 | |
|             fmt,
 | |
|             encoding_str,
 | |
|             (int)badchar,
 | |
|             uself->start,
 | |
|             reason_str);
 | |
|     }
 | |
|     else {
 | |
|         result = PyUnicode_FromFormat(
 | |
|             "'%U' codec can't encode characters in position %zd-%zd: %U",
 | |
|             encoding_str,
 | |
|             uself->start,
 | |
|             uself->end-1,
 | |
|             reason_str);
 | |
|     }
 | |
| done:
 | |
|     Py_XDECREF(reason_str);
 | |
|     Py_XDECREF(encoding_str);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static PyTypeObject _PyExc_UnicodeEncodeError = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "UnicodeEncodeError",
 | |
|     sizeof(PyUnicodeErrorObject), 0,
 | |
|     (destructor)UnicodeError_dealloc, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
 | |
|     (reprfunc)UnicodeEncodeError_str, 0, 0, 0,
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC,
 | |
|     PyDoc_STR("Unicode encoding error."), (traverseproc)UnicodeError_traverse,
 | |
|     (inquiry)UnicodeError_clear, 0, 0, 0, 0, 0, UnicodeError_members,
 | |
|     0, &_PyExc_UnicodeError, 0, 0, 0, offsetof(PyUnicodeErrorObject, dict),
 | |
|     (initproc)UnicodeEncodeError_init, 0, BaseException_new,
 | |
| };
 | |
| PyObject *PyExc_UnicodeEncodeError = (PyObject *)&_PyExc_UnicodeEncodeError;
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeEncodeError_Create(
 | |
|     const char *encoding, const Py_UNICODE *object, Py_ssize_t length,
 | |
|     Py_ssize_t start, Py_ssize_t end, const char *reason)
 | |
| {
 | |
|     return PyObject_CallFunction(PyExc_UnicodeEncodeError, "su#nns",
 | |
|                                  encoding, object, length, start, end, reason);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    UnicodeDecodeError extends UnicodeError
 | |
|  */
 | |
| 
 | |
| static int
 | |
| UnicodeDecodeError_init(PyObject *self, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyUnicodeErrorObject *ude;
 | |
| 
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, kwds) == -1)
 | |
|         return -1;
 | |
| 
 | |
|     ude = (PyUnicodeErrorObject *)self;
 | |
| 
 | |
|     Py_CLEAR(ude->encoding);
 | |
|     Py_CLEAR(ude->object);
 | |
|     Py_CLEAR(ude->reason);
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "UOnnU",
 | |
|                           &ude->encoding, &ude->object,
 | |
|                           &ude->start, &ude->end, &ude->reason)) {
 | |
|              ude->encoding = ude->object = ude->reason = NULL;
 | |
|              return -1;
 | |
|     }
 | |
| 
 | |
|     Py_INCREF(ude->encoding);
 | |
|     Py_INCREF(ude->object);
 | |
|     Py_INCREF(ude->reason);
 | |
| 
 | |
|     if (!PyBytes_Check(ude->object)) {
 | |
|         Py_buffer view;
 | |
|         if (PyObject_GetBuffer(ude->object, &view, PyBUF_SIMPLE) != 0)
 | |
|             goto error;
 | |
|         Py_XSETREF(ude->object, PyBytes_FromStringAndSize(view.buf, view.len));
 | |
|         PyBuffer_Release(&view);
 | |
|         if (!ude->object)
 | |
|             goto error;
 | |
|     }
 | |
|     return 0;
 | |
| 
 | |
| error:
 | |
|     Py_CLEAR(ude->encoding);
 | |
|     Py_CLEAR(ude->object);
 | |
|     Py_CLEAR(ude->reason);
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| static PyObject *
 | |
| UnicodeDecodeError_str(PyObject *self)
 | |
| {
 | |
|     PyUnicodeErrorObject *uself = (PyUnicodeErrorObject *)self;
 | |
|     PyObject *result = NULL;
 | |
|     PyObject *reason_str = NULL;
 | |
|     PyObject *encoding_str = NULL;
 | |
| 
 | |
|     if (!uself->object)
 | |
|         /* Not properly initialized. */
 | |
|         return PyUnicode_FromString("");
 | |
| 
 | |
|     /* Get reason and encoding as strings, which they might not be if
 | |
|        they've been modified after we were constructed. */
 | |
|     reason_str = PyObject_Str(uself->reason);
 | |
|     if (reason_str == NULL)
 | |
|         goto done;
 | |
|     encoding_str = PyObject_Str(uself->encoding);
 | |
|     if (encoding_str == NULL)
 | |
|         goto done;
 | |
| 
 | |
|     if (uself->start < PyBytes_GET_SIZE(uself->object) && uself->end == uself->start+1) {
 | |
|         int byte = (int)(PyBytes_AS_STRING(((PyUnicodeErrorObject *)self)->object)[uself->start]&0xff);
 | |
|         result = PyUnicode_FromFormat(
 | |
|             "'%U' codec can't decode byte 0x%02x in position %zd: %U",
 | |
|             encoding_str,
 | |
|             byte,
 | |
|             uself->start,
 | |
|             reason_str);
 | |
|     }
 | |
|     else {
 | |
|         result = PyUnicode_FromFormat(
 | |
|             "'%U' codec can't decode bytes in position %zd-%zd: %U",
 | |
|             encoding_str,
 | |
|             uself->start,
 | |
|             uself->end-1,
 | |
|             reason_str
 | |
|             );
 | |
|     }
 | |
| done:
 | |
|     Py_XDECREF(reason_str);
 | |
|     Py_XDECREF(encoding_str);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static PyTypeObject _PyExc_UnicodeDecodeError = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "UnicodeDecodeError",
 | |
|     sizeof(PyUnicodeErrorObject), 0,
 | |
|     (destructor)UnicodeError_dealloc, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
 | |
|     (reprfunc)UnicodeDecodeError_str, 0, 0, 0,
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC,
 | |
|     PyDoc_STR("Unicode decoding error."), (traverseproc)UnicodeError_traverse,
 | |
|     (inquiry)UnicodeError_clear, 0, 0, 0, 0, 0, UnicodeError_members,
 | |
|     0, &_PyExc_UnicodeError, 0, 0, 0, offsetof(PyUnicodeErrorObject, dict),
 | |
|     (initproc)UnicodeDecodeError_init, 0, BaseException_new,
 | |
| };
 | |
| PyObject *PyExc_UnicodeDecodeError = (PyObject *)&_PyExc_UnicodeDecodeError;
 | |
| 
 | |
| PyObject *
 | |
| PyUnicodeDecodeError_Create(
 | |
|     const char *encoding, const char *object, Py_ssize_t length,
 | |
|     Py_ssize_t start, Py_ssize_t end, const char *reason)
 | |
| {
 | |
|     return PyObject_CallFunction(PyExc_UnicodeDecodeError, "sy#nns",
 | |
|                                  encoding, object, length, start, end, reason);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    UnicodeTranslateError extends UnicodeError
 | |
|  */
 | |
| 
 | |
| static int
 | |
| UnicodeTranslateError_init(PyUnicodeErrorObject *self, PyObject *args,
 | |
|                            PyObject *kwds)
 | |
| {
 | |
|     if (BaseException_init((PyBaseExceptionObject *)self, args, kwds) == -1)
 | |
|         return -1;
 | |
| 
 | |
|     Py_CLEAR(self->object);
 | |
|     Py_CLEAR(self->reason);
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "UnnU",
 | |
|                           &self->object,
 | |
|                           &self->start, &self->end, &self->reason)) {
 | |
|         self->object = self->reason = NULL;
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     Py_INCREF(self->object);
 | |
|     Py_INCREF(self->reason);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyObject *
 | |
| UnicodeTranslateError_str(PyObject *self)
 | |
| {
 | |
|     PyUnicodeErrorObject *uself = (PyUnicodeErrorObject *)self;
 | |
|     PyObject *result = NULL;
 | |
|     PyObject *reason_str = NULL;
 | |
| 
 | |
|     if (!uself->object)
 | |
|         /* Not properly initialized. */
 | |
|         return PyUnicode_FromString("");
 | |
| 
 | |
|     /* Get reason as a string, which it might not be if it's been
 | |
|        modified after we were constructed. */
 | |
|     reason_str = PyObject_Str(uself->reason);
 | |
|     if (reason_str == NULL)
 | |
|         goto done;
 | |
| 
 | |
|     if (uself->start < PyUnicode_GET_LENGTH(uself->object) && uself->end == uself->start+1) {
 | |
|         Py_UCS4 badchar = PyUnicode_ReadChar(uself->object, uself->start);
 | |
|         const char *fmt;
 | |
|         if (badchar <= 0xff)
 | |
|             fmt = "can't translate character '\\x%02x' in position %zd: %U";
 | |
|         else if (badchar <= 0xffff)
 | |
|             fmt = "can't translate character '\\u%04x' in position %zd: %U";
 | |
|         else
 | |
|             fmt = "can't translate character '\\U%08x' in position %zd: %U";
 | |
|         result = PyUnicode_FromFormat(
 | |
|             fmt,
 | |
|             (int)badchar,
 | |
|             uself->start,
 | |
|             reason_str
 | |
|         );
 | |
|     } else {
 | |
|         result = PyUnicode_FromFormat(
 | |
|             "can't translate characters in position %zd-%zd: %U",
 | |
|             uself->start,
 | |
|             uself->end-1,
 | |
|             reason_str
 | |
|             );
 | |
|     }
 | |
| done:
 | |
|     Py_XDECREF(reason_str);
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static PyTypeObject _PyExc_UnicodeTranslateError = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "UnicodeTranslateError",
 | |
|     sizeof(PyUnicodeErrorObject), 0,
 | |
|     (destructor)UnicodeError_dealloc, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
 | |
|     (reprfunc)UnicodeTranslateError_str, 0, 0, 0,
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC,
 | |
|     PyDoc_STR("Unicode translation error."), (traverseproc)UnicodeError_traverse,
 | |
|     (inquiry)UnicodeError_clear, 0, 0, 0, 0, 0, UnicodeError_members,
 | |
|     0, &_PyExc_UnicodeError, 0, 0, 0, offsetof(PyUnicodeErrorObject, dict),
 | |
|     (initproc)UnicodeTranslateError_init, 0, BaseException_new,
 | |
| };
 | |
| PyObject *PyExc_UnicodeTranslateError = (PyObject *)&_PyExc_UnicodeTranslateError;
 | |
| 
 | |
| /* Deprecated. */
 | |
| PyObject *
 | |
| PyUnicodeTranslateError_Create(
 | |
|     const Py_UNICODE *object, Py_ssize_t length,
 | |
|     Py_ssize_t start, Py_ssize_t end, const char *reason)
 | |
| {
 | |
|     return PyObject_CallFunction(PyExc_UnicodeTranslateError, "u#nns",
 | |
|                                  object, length, start, end, reason);
 | |
| }
 | |
| 
 | |
| PyObject *
 | |
| _PyUnicodeTranslateError_Create(
 | |
|     PyObject *object,
 | |
|     Py_ssize_t start, Py_ssize_t end, const char *reason)
 | |
| {
 | |
|     return PyObject_CallFunction(PyExc_UnicodeTranslateError, "Onns",
 | |
|                                  object, start, end, reason);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  *    AssertionError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, AssertionError,
 | |
|                        "Assertion failed.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ArithmeticError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, ArithmeticError,
 | |
|                        "Base class for arithmetic errors.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    FloatingPointError extends ArithmeticError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_ArithmeticError, FloatingPointError,
 | |
|                        "Floating point operation failed.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    OverflowError extends ArithmeticError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_ArithmeticError, OverflowError,
 | |
|                        "Result too large to be represented.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ZeroDivisionError extends ArithmeticError
 | |
|  */
 | |
| SimpleExtendsException(PyExc_ArithmeticError, ZeroDivisionError,
 | |
|           "Second argument to a division or modulo operation was zero.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    SystemError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, SystemError,
 | |
|     "Internal error in the Python interpreter.\n"
 | |
|     "\n"
 | |
|     "Please report this to the Python maintainer, along with the traceback,\n"
 | |
|     "the Python version, and the hardware/OS platform and version.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ReferenceError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, ReferenceError,
 | |
|                        "Weak ref proxy used after referent went away.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    MemoryError extends Exception
 | |
|  */
 | |
| 
 | |
| #define MEMERRORS_SAVE 16
 | |
| static PyBaseExceptionObject *memerrors_freelist = NULL;
 | |
| static int memerrors_numfree = 0;
 | |
| 
 | |
| static PyObject *
 | |
| MemoryError_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     PyBaseExceptionObject *self;
 | |
| 
 | |
|     if (type != (PyTypeObject *) PyExc_MemoryError)
 | |
|         return BaseException_new(type, args, kwds);
 | |
|     if (memerrors_freelist == NULL)
 | |
|         return BaseException_new(type, args, kwds);
 | |
|     /* Fetch object from freelist and revive it */
 | |
|     self = memerrors_freelist;
 | |
|     self->args = PyTuple_New(0);
 | |
|     /* This shouldn't happen since the empty tuple is persistent */
 | |
|     if (self->args == NULL)
 | |
|         return NULL;
 | |
|     memerrors_freelist = (PyBaseExceptionObject *) self->dict;
 | |
|     memerrors_numfree--;
 | |
|     self->dict = NULL;
 | |
|     _Py_NewReference((PyObject *)self);
 | |
|     _PyObject_GC_TRACK(self);
 | |
|     return (PyObject *)self;
 | |
| }
 | |
| 
 | |
| static void
 | |
| MemoryError_dealloc(PyBaseExceptionObject *self)
 | |
| {
 | |
|     _PyObject_GC_UNTRACK(self);
 | |
|     BaseException_clear(self);
 | |
|     if (memerrors_numfree >= MEMERRORS_SAVE)
 | |
|         Py_TYPE(self)->tp_free((PyObject *)self);
 | |
|     else {
 | |
|         self->dict = (PyObject *) memerrors_freelist;
 | |
|         memerrors_freelist = self;
 | |
|         memerrors_numfree++;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static int
 | |
| preallocate_memerrors(void)
 | |
| {
 | |
|     /* We create enough MemoryErrors and then decref them, which will fill
 | |
|        up the freelist. */
 | |
|     int i;
 | |
|     PyObject *errors[MEMERRORS_SAVE];
 | |
|     for (i = 0; i < MEMERRORS_SAVE; i++) {
 | |
|         errors[i] = MemoryError_new((PyTypeObject *) PyExc_MemoryError,
 | |
|                                     NULL, NULL);
 | |
|         if (!errors[i]) {
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     for (i = 0; i < MEMERRORS_SAVE; i++) {
 | |
|         Py_DECREF(errors[i]);
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| free_preallocated_memerrors(void)
 | |
| {
 | |
|     while (memerrors_freelist != NULL) {
 | |
|         PyObject *self = (PyObject *) memerrors_freelist;
 | |
|         memerrors_freelist = (PyBaseExceptionObject *) memerrors_freelist->dict;
 | |
|         Py_TYPE(self)->tp_free((PyObject *)self);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static PyTypeObject _PyExc_MemoryError = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     "MemoryError",
 | |
|     sizeof(PyBaseExceptionObject),
 | |
|     0, (destructor)MemoryError_dealloc, 0, 0, 0, 0, 0, 0, 0,
 | |
|     0, 0, 0, 0, 0, 0, 0,
 | |
|     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | Py_TPFLAGS_HAVE_GC,
 | |
|     PyDoc_STR("Out of memory."), (traverseproc)BaseException_traverse,
 | |
|     (inquiry)BaseException_clear, 0, 0, 0, 0, 0, 0, 0, &_PyExc_Exception,
 | |
|     0, 0, 0, offsetof(PyBaseExceptionObject, dict),
 | |
|     (initproc)BaseException_init, 0, MemoryError_new
 | |
| };
 | |
| PyObject *PyExc_MemoryError = (PyObject *) &_PyExc_MemoryError;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    BufferError extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, BufferError, "Buffer error.");
 | |
| 
 | |
| 
 | |
| /* Warning category docstrings */
 | |
| 
 | |
| /*
 | |
|  *    Warning extends Exception
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Exception, Warning,
 | |
|                        "Base class for warning categories.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    UserWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, UserWarning,
 | |
|                        "Base class for warnings generated by user code.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    DeprecationWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, DeprecationWarning,
 | |
|                        "Base class for warnings about deprecated features.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    PendingDeprecationWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, PendingDeprecationWarning,
 | |
|     "Base class for warnings about features which will be deprecated\n"
 | |
|     "in the future.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    SyntaxWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, SyntaxWarning,
 | |
|                        "Base class for warnings about dubious syntax.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    RuntimeWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, RuntimeWarning,
 | |
|                  "Base class for warnings about dubious runtime behavior.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    FutureWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, FutureWarning,
 | |
|     "Base class for warnings about constructs that will change semantically\n"
 | |
|     "in the future.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ImportWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, ImportWarning,
 | |
|           "Base class for warnings about probable mistakes in module imports");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    UnicodeWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, UnicodeWarning,
 | |
|     "Base class for warnings about Unicode related problems, mostly\n"
 | |
|     "related to conversion problems.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    BytesWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, BytesWarning,
 | |
|     "Base class for warnings about bytes and buffer related problems, mostly\n"
 | |
|     "related to conversion from str or comparing to str.");
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *    ResourceWarning extends Warning
 | |
|  */
 | |
| SimpleExtendsException(PyExc_Warning, ResourceWarning,
 | |
|     "Base class for warnings about resource usage.");
 | |
| 
 | |
| 
 | |
| 
 | |
| #ifdef MS_WINDOWS
 | |
| #include <winsock2.h>
 | |
| /* The following constants were added to errno.h in VS2010 but have
 | |
|    preferred WSA equivalents. */
 | |
| #undef EADDRINUSE
 | |
| #undef EADDRNOTAVAIL
 | |
| #undef EAFNOSUPPORT
 | |
| #undef EALREADY
 | |
| #undef ECONNABORTED
 | |
| #undef ECONNREFUSED
 | |
| #undef ECONNRESET
 | |
| #undef EDESTADDRREQ
 | |
| #undef EHOSTUNREACH
 | |
| #undef EINPROGRESS
 | |
| #undef EISCONN
 | |
| #undef ELOOP
 | |
| #undef EMSGSIZE
 | |
| #undef ENETDOWN
 | |
| #undef ENETRESET
 | |
| #undef ENETUNREACH
 | |
| #undef ENOBUFS
 | |
| #undef ENOPROTOOPT
 | |
| #undef ENOTCONN
 | |
| #undef ENOTSOCK
 | |
| #undef EOPNOTSUPP
 | |
| #undef EPROTONOSUPPORT
 | |
| #undef EPROTOTYPE
 | |
| #undef ETIMEDOUT
 | |
| #undef EWOULDBLOCK
 | |
| 
 | |
| #if defined(WSAEALREADY) && !defined(EALREADY)
 | |
| #define EALREADY WSAEALREADY
 | |
| #endif
 | |
| #if defined(WSAECONNABORTED) && !defined(ECONNABORTED)
 | |
| #define ECONNABORTED WSAECONNABORTED
 | |
| #endif
 | |
| #if defined(WSAECONNREFUSED) && !defined(ECONNREFUSED)
 | |
| #define ECONNREFUSED WSAECONNREFUSED
 | |
| #endif
 | |
| #if defined(WSAECONNRESET) && !defined(ECONNRESET)
 | |
| #define ECONNRESET WSAECONNRESET
 | |
| #endif
 | |
| #if defined(WSAEINPROGRESS) && !defined(EINPROGRESS)
 | |
| #define EINPROGRESS WSAEINPROGRESS
 | |
| #endif
 | |
| #if defined(WSAESHUTDOWN) && !defined(ESHUTDOWN)
 | |
| #define ESHUTDOWN WSAESHUTDOWN
 | |
| #endif
 | |
| #if defined(WSAETIMEDOUT) && !defined(ETIMEDOUT)
 | |
| #define ETIMEDOUT WSAETIMEDOUT
 | |
| #endif
 | |
| #if defined(WSAEWOULDBLOCK) && !defined(EWOULDBLOCK)
 | |
| #define EWOULDBLOCK WSAEWOULDBLOCK
 | |
| #endif
 | |
| #endif /* MS_WINDOWS */
 | |
| 
 | |
| PyStatus
 | |
| _PyExc_Init(void)
 | |
| {
 | |
| #define PRE_INIT(TYPE) \
 | |
|     if (!(_PyExc_ ## TYPE.tp_flags & Py_TPFLAGS_READY)) { \
 | |
|         if (PyType_Ready(&_PyExc_ ## TYPE) < 0) { \
 | |
|             return _PyStatus_ERR("exceptions bootstrapping error."); \
 | |
|         } \
 | |
|         Py_INCREF(PyExc_ ## TYPE); \
 | |
|     }
 | |
| 
 | |
| #define ADD_ERRNO(TYPE, CODE) \
 | |
|     do { \
 | |
|         PyObject *_code = PyLong_FromLong(CODE); \
 | |
|         assert(_PyObject_RealIsSubclass(PyExc_ ## TYPE, PyExc_OSError)); \
 | |
|         if (!_code || PyDict_SetItem(errnomap, _code, PyExc_ ## TYPE)) \
 | |
|             return _PyStatus_ERR("errmap insertion problem."); \
 | |
|         Py_DECREF(_code); \
 | |
|     } while (0)
 | |
| 
 | |
|     PRE_INIT(BaseException);
 | |
|     PRE_INIT(Exception);
 | |
|     PRE_INIT(TypeError);
 | |
|     PRE_INIT(StopAsyncIteration);
 | |
|     PRE_INIT(StopIteration);
 | |
|     PRE_INIT(GeneratorExit);
 | |
|     PRE_INIT(SystemExit);
 | |
|     PRE_INIT(KeyboardInterrupt);
 | |
|     PRE_INIT(ImportError);
 | |
|     PRE_INIT(ModuleNotFoundError);
 | |
|     PRE_INIT(OSError);
 | |
|     PRE_INIT(EOFError);
 | |
|     PRE_INIT(RuntimeError);
 | |
|     PRE_INIT(RecursionError);
 | |
|     PRE_INIT(NotImplementedError);
 | |
|     PRE_INIT(NameError);
 | |
|     PRE_INIT(UnboundLocalError);
 | |
|     PRE_INIT(AttributeError);
 | |
|     PRE_INIT(SyntaxError);
 | |
|     PRE_INIT(IndentationError);
 | |
|     PRE_INIT(TabError);
 | |
|     PRE_INIT(LookupError);
 | |
|     PRE_INIT(IndexError);
 | |
|     PRE_INIT(KeyError);
 | |
|     PRE_INIT(ValueError);
 | |
|     PRE_INIT(UnicodeError);
 | |
|     PRE_INIT(UnicodeEncodeError);
 | |
|     PRE_INIT(UnicodeDecodeError);
 | |
|     PRE_INIT(UnicodeTranslateError);
 | |
|     PRE_INIT(AssertionError);
 | |
|     PRE_INIT(ArithmeticError);
 | |
|     PRE_INIT(FloatingPointError);
 | |
|     PRE_INIT(OverflowError);
 | |
|     PRE_INIT(ZeroDivisionError);
 | |
|     PRE_INIT(SystemError);
 | |
|     PRE_INIT(ReferenceError);
 | |
|     PRE_INIT(MemoryError);
 | |
|     PRE_INIT(BufferError);
 | |
|     PRE_INIT(Warning);
 | |
|     PRE_INIT(UserWarning);
 | |
|     PRE_INIT(DeprecationWarning);
 | |
|     PRE_INIT(PendingDeprecationWarning);
 | |
|     PRE_INIT(SyntaxWarning);
 | |
|     PRE_INIT(RuntimeWarning);
 | |
|     PRE_INIT(FutureWarning);
 | |
|     PRE_INIT(ImportWarning);
 | |
|     PRE_INIT(UnicodeWarning);
 | |
|     PRE_INIT(BytesWarning);
 | |
|     PRE_INIT(ResourceWarning);
 | |
| 
 | |
|     /* OSError subclasses */
 | |
|     PRE_INIT(ConnectionError);
 | |
| 
 | |
|     PRE_INIT(BlockingIOError);
 | |
|     PRE_INIT(BrokenPipeError);
 | |
|     PRE_INIT(ChildProcessError);
 | |
|     PRE_INIT(ConnectionAbortedError);
 | |
|     PRE_INIT(ConnectionRefusedError);
 | |
|     PRE_INIT(ConnectionResetError);
 | |
|     PRE_INIT(FileExistsError);
 | |
|     PRE_INIT(FileNotFoundError);
 | |
|     PRE_INIT(IsADirectoryError);
 | |
|     PRE_INIT(NotADirectoryError);
 | |
|     PRE_INIT(InterruptedError);
 | |
|     PRE_INIT(PermissionError);
 | |
|     PRE_INIT(ProcessLookupError);
 | |
|     PRE_INIT(TimeoutError);
 | |
| 
 | |
|     if (preallocate_memerrors() < 0) {
 | |
|         return _PyStatus_ERR("Could not preallocate MemoryError object");
 | |
|     }
 | |
| 
 | |
|     /* Add exceptions to errnomap */
 | |
|     if (!errnomap) {
 | |
|         errnomap = PyDict_New();
 | |
|         if (!errnomap) {
 | |
|             return _PyStatus_ERR("Cannot allocate map from errnos to OSError subclasses");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     ADD_ERRNO(BlockingIOError, EAGAIN);
 | |
|     ADD_ERRNO(BlockingIOError, EALREADY);
 | |
|     ADD_ERRNO(BlockingIOError, EINPROGRESS);
 | |
|     ADD_ERRNO(BlockingIOError, EWOULDBLOCK);
 | |
|     ADD_ERRNO(BrokenPipeError, EPIPE);
 | |
| #ifdef ESHUTDOWN
 | |
|     ADD_ERRNO(BrokenPipeError, ESHUTDOWN);
 | |
| #endif
 | |
|     ADD_ERRNO(ChildProcessError, ECHILD);
 | |
|     ADD_ERRNO(ConnectionAbortedError, ECONNABORTED);
 | |
|     ADD_ERRNO(ConnectionRefusedError, ECONNREFUSED);
 | |
|     ADD_ERRNO(ConnectionResetError, ECONNRESET);
 | |
|     ADD_ERRNO(FileExistsError, EEXIST);
 | |
|     ADD_ERRNO(FileNotFoundError, ENOENT);
 | |
|     ADD_ERRNO(IsADirectoryError, EISDIR);
 | |
|     ADD_ERRNO(NotADirectoryError, ENOTDIR);
 | |
|     ADD_ERRNO(InterruptedError, EINTR);
 | |
|     ADD_ERRNO(PermissionError, EACCES);
 | |
|     ADD_ERRNO(PermissionError, EPERM);
 | |
|     ADD_ERRNO(ProcessLookupError, ESRCH);
 | |
|     ADD_ERRNO(TimeoutError, ETIMEDOUT);
 | |
| 
 | |
|     return _PyStatus_OK();
 | |
| 
 | |
| #undef PRE_INIT
 | |
| #undef ADD_ERRNO
 | |
| }
 | |
| 
 | |
| 
 | |
| /* Add exception types to the builtins module */
 | |
| PyStatus
 | |
| _PyBuiltins_AddExceptions(PyObject *bltinmod)
 | |
| {
 | |
| #define POST_INIT(TYPE) \
 | |
|     if (PyDict_SetItemString(bdict, # TYPE, PyExc_ ## TYPE)) { \
 | |
|         return _PyStatus_ERR("Module dictionary insertion problem."); \
 | |
|     }
 | |
| 
 | |
| #define INIT_ALIAS(NAME, TYPE) \
 | |
|     do { \
 | |
|         Py_INCREF(PyExc_ ## TYPE); \
 | |
|         Py_XDECREF(PyExc_ ## NAME); \
 | |
|         PyExc_ ## NAME = PyExc_ ## TYPE; \
 | |
|         if (PyDict_SetItemString(bdict, # NAME, PyExc_ ## NAME)) { \
 | |
|             return _PyStatus_ERR("Module dictionary insertion problem."); \
 | |
|         } \
 | |
|     } while (0)
 | |
| 
 | |
|     PyObject *bdict;
 | |
| 
 | |
|     bdict = PyModule_GetDict(bltinmod);
 | |
|     if (bdict == NULL) {
 | |
|         return _PyStatus_ERR("exceptions bootstrapping error.");
 | |
|     }
 | |
| 
 | |
|     POST_INIT(BaseException);
 | |
|     POST_INIT(Exception);
 | |
|     POST_INIT(TypeError);
 | |
|     POST_INIT(StopAsyncIteration);
 | |
|     POST_INIT(StopIteration);
 | |
|     POST_INIT(GeneratorExit);
 | |
|     POST_INIT(SystemExit);
 | |
|     POST_INIT(KeyboardInterrupt);
 | |
|     POST_INIT(ImportError);
 | |
|     POST_INIT(ModuleNotFoundError);
 | |
|     POST_INIT(OSError);
 | |
|     INIT_ALIAS(EnvironmentError, OSError);
 | |
|     INIT_ALIAS(IOError, OSError);
 | |
| #ifdef MS_WINDOWS
 | |
|     INIT_ALIAS(WindowsError, OSError);
 | |
| #endif
 | |
|     POST_INIT(EOFError);
 | |
|     POST_INIT(RuntimeError);
 | |
|     POST_INIT(RecursionError);
 | |
|     POST_INIT(NotImplementedError);
 | |
|     POST_INIT(NameError);
 | |
|     POST_INIT(UnboundLocalError);
 | |
|     POST_INIT(AttributeError);
 | |
|     POST_INIT(SyntaxError);
 | |
|     POST_INIT(IndentationError);
 | |
|     POST_INIT(TabError);
 | |
|     POST_INIT(LookupError);
 | |
|     POST_INIT(IndexError);
 | |
|     POST_INIT(KeyError);
 | |
|     POST_INIT(ValueError);
 | |
|     POST_INIT(UnicodeError);
 | |
|     POST_INIT(UnicodeEncodeError);
 | |
|     POST_INIT(UnicodeDecodeError);
 | |
|     POST_INIT(UnicodeTranslateError);
 | |
|     POST_INIT(AssertionError);
 | |
|     POST_INIT(ArithmeticError);
 | |
|     POST_INIT(FloatingPointError);
 | |
|     POST_INIT(OverflowError);
 | |
|     POST_INIT(ZeroDivisionError);
 | |
|     POST_INIT(SystemError);
 | |
|     POST_INIT(ReferenceError);
 | |
|     POST_INIT(MemoryError);
 | |
|     POST_INIT(BufferError);
 | |
|     POST_INIT(Warning);
 | |
|     POST_INIT(UserWarning);
 | |
|     POST_INIT(DeprecationWarning);
 | |
|     POST_INIT(PendingDeprecationWarning);
 | |
|     POST_INIT(SyntaxWarning);
 | |
|     POST_INIT(RuntimeWarning);
 | |
|     POST_INIT(FutureWarning);
 | |
|     POST_INIT(ImportWarning);
 | |
|     POST_INIT(UnicodeWarning);
 | |
|     POST_INIT(BytesWarning);
 | |
|     POST_INIT(ResourceWarning);
 | |
| 
 | |
|     /* OSError subclasses */
 | |
|     POST_INIT(ConnectionError);
 | |
| 
 | |
|     POST_INIT(BlockingIOError);
 | |
|     POST_INIT(BrokenPipeError);
 | |
|     POST_INIT(ChildProcessError);
 | |
|     POST_INIT(ConnectionAbortedError);
 | |
|     POST_INIT(ConnectionRefusedError);
 | |
|     POST_INIT(ConnectionResetError);
 | |
|     POST_INIT(FileExistsError);
 | |
|     POST_INIT(FileNotFoundError);
 | |
|     POST_INIT(IsADirectoryError);
 | |
|     POST_INIT(NotADirectoryError);
 | |
|     POST_INIT(InterruptedError);
 | |
|     POST_INIT(PermissionError);
 | |
|     POST_INIT(ProcessLookupError);
 | |
|     POST_INIT(TimeoutError);
 | |
| 
 | |
|     return _PyStatus_OK();
 | |
| 
 | |
| #undef POST_INIT
 | |
| #undef INIT_ALIAS
 | |
| }
 | |
| 
 | |
| void
 | |
| _PyExc_Fini(void)
 | |
| {
 | |
|     free_preallocated_memerrors();
 | |
|     Py_CLEAR(errnomap);
 | |
| }
 | |
| 
 | |
| /* Helper to do the equivalent of "raise X from Y" in C, but always using
 | |
|  * the current exception rather than passing one in.
 | |
|  *
 | |
|  * We currently limit this to *only* exceptions that use the BaseException
 | |
|  * tp_init and tp_new methods, since we can be reasonably sure we can wrap
 | |
|  * those correctly without losing data and without losing backwards
 | |
|  * compatibility.
 | |
|  *
 | |
|  * We also aim to rule out *all* exceptions that might be storing additional
 | |
|  * state, whether by having a size difference relative to BaseException,
 | |
|  * additional arguments passed in during construction or by having a
 | |
|  * non-empty instance dict.
 | |
|  *
 | |
|  * We need to be very careful with what we wrap, since changing types to
 | |
|  * a broader exception type would be backwards incompatible for
 | |
|  * existing codecs, and with different init or new method implementations
 | |
|  * may either not support instantiation with PyErr_Format or lose
 | |
|  * information when instantiated that way.
 | |
|  *
 | |
|  * XXX (ncoghlan): This could be made more comprehensive by exploiting the
 | |
|  * fact that exceptions are expected to support pickling. If more builtin
 | |
|  * exceptions (e.g. AttributeError) start to be converted to rich
 | |
|  * exceptions with additional attributes, that's probably a better approach
 | |
|  * to pursue over adding special cases for particular stateful subclasses.
 | |
|  *
 | |
|  * Returns a borrowed reference to the new exception (if any), NULL if the
 | |
|  * existing exception was left in place.
 | |
|  */
 | |
| PyObject *
 | |
| _PyErr_TrySetFromCause(const char *format, ...)
 | |
| {
 | |
|     PyObject* msg_prefix;
 | |
|     PyObject *exc, *val, *tb;
 | |
|     PyTypeObject *caught_type;
 | |
|     PyObject **dictptr;
 | |
|     PyObject *instance_args;
 | |
|     Py_ssize_t num_args, caught_type_size, base_exc_size;
 | |
|     PyObject *new_exc, *new_val, *new_tb;
 | |
|     va_list vargs;
 | |
|     int same_basic_size;
 | |
| 
 | |
|     PyErr_Fetch(&exc, &val, &tb);
 | |
|     caught_type = (PyTypeObject *)exc;
 | |
|     /* Ensure type info indicates no extra state is stored at the C level
 | |
|      * and that the type can be reinstantiated using PyErr_Format
 | |
|      */
 | |
|     caught_type_size = caught_type->tp_basicsize;
 | |
|     base_exc_size = _PyExc_BaseException.tp_basicsize;
 | |
|     same_basic_size = (
 | |
|         caught_type_size == base_exc_size ||
 | |
|         (PyType_SUPPORTS_WEAKREFS(caught_type) &&
 | |
|             (caught_type_size == base_exc_size + (Py_ssize_t)sizeof(PyObject *))
 | |
|         )
 | |
|     );
 | |
|     if (caught_type->tp_init != (initproc)BaseException_init ||
 | |
|         caught_type->tp_new != BaseException_new ||
 | |
|         !same_basic_size ||
 | |
|         caught_type->tp_itemsize != _PyExc_BaseException.tp_itemsize) {
 | |
|         /* We can't be sure we can wrap this safely, since it may contain
 | |
|          * more state than just the exception type. Accordingly, we just
 | |
|          * leave it alone.
 | |
|          */
 | |
|         PyErr_Restore(exc, val, tb);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     /* Check the args are empty or contain a single string */
 | |
|     PyErr_NormalizeException(&exc, &val, &tb);
 | |
|     instance_args = ((PyBaseExceptionObject *)val)->args;
 | |
|     num_args = PyTuple_GET_SIZE(instance_args);
 | |
|     if (num_args > 1 ||
 | |
|         (num_args == 1 &&
 | |
|          !PyUnicode_CheckExact(PyTuple_GET_ITEM(instance_args, 0)))) {
 | |
|         /* More than 1 arg, or the one arg we do have isn't a string
 | |
|          */
 | |
|         PyErr_Restore(exc, val, tb);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     /* Ensure the instance dict is also empty */
 | |
|     dictptr = _PyObject_GetDictPtr(val);
 | |
|     if (dictptr != NULL && *dictptr != NULL &&
 | |
|         PyDict_GET_SIZE(*dictptr) > 0) {
 | |
|         /* While we could potentially copy a non-empty instance dictionary
 | |
|          * to the replacement exception, for now we take the more
 | |
|          * conservative path of leaving exceptions with attributes set
 | |
|          * alone.
 | |
|          */
 | |
|         PyErr_Restore(exc, val, tb);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     /* For exceptions that we can wrap safely, we chain the original
 | |
|      * exception to a new one of the exact same type with an
 | |
|      * error message that mentions the additional details and the
 | |
|      * original exception.
 | |
|      *
 | |
|      * It would be nice to wrap OSError and various other exception
 | |
|      * types as well, but that's quite a bit trickier due to the extra
 | |
|      * state potentially stored on OSError instances.
 | |
|      */
 | |
|     /* Ensure the traceback is set correctly on the existing exception */
 | |
|     if (tb != NULL) {
 | |
|         PyException_SetTraceback(val, tb);
 | |
|         Py_DECREF(tb);
 | |
|     }
 | |
| 
 | |
| #ifdef HAVE_STDARG_PROTOTYPES
 | |
|     va_start(vargs, format);
 | |
| #else
 | |
|     va_start(vargs);
 | |
| #endif
 | |
|     msg_prefix = PyUnicode_FromFormatV(format, vargs);
 | |
|     va_end(vargs);
 | |
|     if (msg_prefix == NULL) {
 | |
|         Py_DECREF(exc);
 | |
|         Py_DECREF(val);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     PyErr_Format(exc, "%U (%s: %S)",
 | |
|                  msg_prefix, Py_TYPE(val)->tp_name, val);
 | |
|     Py_DECREF(exc);
 | |
|     Py_DECREF(msg_prefix);
 | |
|     PyErr_Fetch(&new_exc, &new_val, &new_tb);
 | |
|     PyErr_NormalizeException(&new_exc, &new_val, &new_tb);
 | |
|     PyException_SetCause(new_val, val);
 | |
|     PyErr_Restore(new_exc, new_val, new_tb);
 | |
|     return new_val;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* To help with migration from Python 2, SyntaxError.__init__ applies some
 | |
|  * heuristics to try to report a more meaningful exception when print and
 | |
|  * exec are used like statements.
 | |
|  *
 | |
|  * The heuristics are currently expected to detect the following cases:
 | |
|  *   - top level statement
 | |
|  *   - statement in a nested suite
 | |
|  *   - trailing section of a one line complex statement
 | |
|  *
 | |
|  * They're currently known not to trigger:
 | |
|  *   - after a semi-colon
 | |
|  *
 | |
|  * The error message can be a bit odd in cases where the "arguments" are
 | |
|  * completely illegal syntactically, but that isn't worth the hassle of
 | |
|  * fixing.
 | |
|  *
 | |
|  * We also can't do anything about cases that are legal Python 3 syntax
 | |
|  * but mean something entirely different from what they did in Python 2
 | |
|  * (omitting the arguments entirely, printing items preceded by a unary plus
 | |
|  * or minus, using the stream redirection syntax).
 | |
|  */
 | |
| 
 | |
| 
 | |
| // Static helper for setting legacy print error message
 | |
| static int
 | |
| _set_legacy_print_statement_msg(PySyntaxErrorObject *self, Py_ssize_t start)
 | |
| {
 | |
|     // PRINT_OFFSET is to remove the `print ` prefix from the data.
 | |
|     const int PRINT_OFFSET = 6;
 | |
|     const int STRIP_BOTH = 2;
 | |
|     Py_ssize_t start_pos = start + PRINT_OFFSET;
 | |
|     Py_ssize_t text_len = PyUnicode_GET_LENGTH(self->text);
 | |
|     Py_UCS4 semicolon = ';';
 | |
|     Py_ssize_t end_pos = PyUnicode_FindChar(self->text, semicolon,
 | |
|                                             start_pos, text_len, 1);
 | |
|     if (end_pos < -1) {
 | |
|       return -1;
 | |
|     } else if (end_pos == -1) {
 | |
|       end_pos = text_len;
 | |
|     }
 | |
| 
 | |
|     PyObject *data = PyUnicode_Substring(self->text, start_pos, end_pos);
 | |
|     if (data == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     PyObject *strip_sep_obj = PyUnicode_FromString(" \t\r\n");
 | |
|     if (strip_sep_obj == NULL) {
 | |
|         Py_DECREF(data);
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     PyObject *new_data = _PyUnicode_XStrip(data, STRIP_BOTH, strip_sep_obj);
 | |
|     Py_DECREF(data);
 | |
|     Py_DECREF(strip_sep_obj);
 | |
|     if (new_data == NULL) {
 | |
|         return -1;
 | |
|     }
 | |
|     // gets the modified text_len after stripping `print `
 | |
|     text_len = PyUnicode_GET_LENGTH(new_data);
 | |
|     const char *maybe_end_arg = "";
 | |
|     if (text_len > 0 && PyUnicode_READ_CHAR(new_data, text_len-1) == ',') {
 | |
|         maybe_end_arg = " end=\" \"";
 | |
|     }
 | |
|     PyObject *error_msg = PyUnicode_FromFormat(
 | |
|         "Missing parentheses in call to 'print'. Did you mean print(%U%s)?",
 | |
|         new_data, maybe_end_arg
 | |
|     );
 | |
|     Py_DECREF(new_data);
 | |
|     if (error_msg == NULL)
 | |
|         return -1;
 | |
| 
 | |
|     Py_XSETREF(self->msg, error_msg);
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _check_for_legacy_statements(PySyntaxErrorObject *self, Py_ssize_t start)
 | |
| {
 | |
|     /* Return values:
 | |
|      *   -1: an error occurred
 | |
|      *    0: nothing happened
 | |
|      *    1: the check triggered & the error message was changed
 | |
|      */
 | |
|     static PyObject *print_prefix = NULL;
 | |
|     static PyObject *exec_prefix = NULL;
 | |
|     Py_ssize_t text_len = PyUnicode_GET_LENGTH(self->text), match;
 | |
|     int kind = PyUnicode_KIND(self->text);
 | |
|     const void *data = PyUnicode_DATA(self->text);
 | |
| 
 | |
|     /* Ignore leading whitespace */
 | |
|     while (start < text_len) {
 | |
|         Py_UCS4 ch = PyUnicode_READ(kind, data, start);
 | |
|         if (!Py_UNICODE_ISSPACE(ch))
 | |
|             break;
 | |
|         start++;
 | |
|     }
 | |
|     /* Checking against an empty or whitespace-only part of the string */
 | |
|     if (start == text_len) {
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     /* Check for legacy print statements */
 | |
|     if (print_prefix == NULL) {
 | |
|         print_prefix = PyUnicode_InternFromString("print ");
 | |
|         if (print_prefix == NULL) {
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     match = PyUnicode_Tailmatch(self->text, print_prefix,
 | |
|                                 start, text_len, -1);
 | |
|     if (match == -1) {
 | |
|         return -1;
 | |
|     }
 | |
|     if (match) {
 | |
|         return _set_legacy_print_statement_msg(self, start);
 | |
|     }
 | |
| 
 | |
|     /* Check for legacy exec statements */
 | |
|     if (exec_prefix == NULL) {
 | |
|         exec_prefix = PyUnicode_InternFromString("exec ");
 | |
|         if (exec_prefix == NULL) {
 | |
|             return -1;
 | |
|         }
 | |
|     }
 | |
|     match = PyUnicode_Tailmatch(self->text, exec_prefix, start, text_len, -1);
 | |
|     if (match == -1) {
 | |
|         return -1;
 | |
|     }
 | |
|     if (match) {
 | |
|         PyObject *msg = PyUnicode_FromString("Missing parentheses in call "
 | |
|                                              "to 'exec'");
 | |
|         if (msg == NULL) {
 | |
|             return -1;
 | |
|         }
 | |
|         Py_XSETREF(self->msg, msg);
 | |
|         return 1;
 | |
|     }
 | |
|     /* Fall back to the default error message */
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| _report_missing_parentheses(PySyntaxErrorObject *self)
 | |
| {
 | |
|     Py_UCS4 left_paren = 40;
 | |
|     Py_ssize_t left_paren_index;
 | |
|     Py_ssize_t text_len = PyUnicode_GET_LENGTH(self->text);
 | |
|     int legacy_check_result = 0;
 | |
| 
 | |
|     /* Skip entirely if there is an opening parenthesis */
 | |
|     left_paren_index = PyUnicode_FindChar(self->text, left_paren,
 | |
|                                           0, text_len, 1);
 | |
|     if (left_paren_index < -1) {
 | |
|         return -1;
 | |
|     }
 | |
|     if (left_paren_index != -1) {
 | |
|         /* Use default error message for any line with an opening paren */
 | |
|         return 0;
 | |
|     }
 | |
|     /* Handle the simple statement case */
 | |
|     legacy_check_result = _check_for_legacy_statements(self, 0);
 | |
|     if (legacy_check_result < 0) {
 | |
|         return -1;
 | |
| 
 | |
|     }
 | |
|     if (legacy_check_result == 0) {
 | |
|         /* Handle the one-line complex statement case */
 | |
|         Py_UCS4 colon = 58;
 | |
|         Py_ssize_t colon_index;
 | |
|         colon_index = PyUnicode_FindChar(self->text, colon,
 | |
|                                          0, text_len, 1);
 | |
|         if (colon_index < -1) {
 | |
|             return -1;
 | |
|         }
 | |
|         if (colon_index >= 0 && colon_index < text_len) {
 | |
|             /* Check again, starting from just after the colon */
 | |
|             if (_check_for_legacy_statements(self, colon_index+1) < 0) {
 | |
|                 return -1;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     return 0;
 | |
| }
 | 
