diff options
author | orivej <orivej@yandex-team.ru> | 2022-02-10 16:44:49 +0300 |
---|---|---|
committer | Daniil Cherednik <dcherednik@yandex-team.ru> | 2022-02-10 16:44:49 +0300 |
commit | 718c552901d703c502ccbefdfc3c9028d608b947 (patch) | |
tree | 46534a98bbefcd7b1f3faa5b52c138ab27db75b7 /contrib/tools/python3/src/Objects/exceptions.c | |
parent | e9656aae26e0358d5378e5b63dcac5c8dbe0e4d0 (diff) | |
download | ydb-718c552901d703c502ccbefdfc3c9028d608b947.tar.gz |
Restoring authorship annotation for <orivej@yandex-team.ru>. Commit 1 of 2.
Diffstat (limited to 'contrib/tools/python3/src/Objects/exceptions.c')
-rw-r--r-- | contrib/tools/python3/src/Objects/exceptions.c | 5528 |
1 files changed, 2764 insertions, 2764 deletions
diff --git a/contrib/tools/python3/src/Objects/exceptions.c b/contrib/tools/python3/src/Objects/exceptions.c index e67ecfab85..5742355712 100644 --- a/contrib/tools/python3/src/Objects/exceptions.c +++ b/contrib/tools/python3/src/Objects/exceptions.c @@ -1,309 +1,309 @@ -/* - * 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> +/* + * 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 * + + +/* 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}, -}; - - +{ + 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}, +}; + + static inline PyBaseExceptionObject* _PyBaseExceptionObject_cast(PyObject *exc) { @@ -312,55 +312,55 @@ _PyBaseExceptionObject_cast(PyObject *exc) } -PyObject * +PyObject * PyException_GetTraceback(PyObject *self) { PyBaseExceptionObject *base_self = _PyBaseExceptionObject_cast(self); - Py_XINCREF(base_self->traceback); - return base_self->traceback; -} - - -int + Py_XINCREF(base_self->traceback); + return base_self->traceback; +} + + +int PyException_SetTraceback(PyObject *self, PyObject *tb) { return BaseException_set_tb(_PyBaseExceptionObject_cast(self), tb, NULL); -} - -PyObject * +} + +PyObject * PyException_GetCause(PyObject *self) { PyObject *cause = _PyBaseExceptionObject_cast(self)->cause; - Py_XINCREF(cause); - return cause; -} - -/* Steals a reference to cause */ -void -PyException_SetCause(PyObject *self, PyObject *cause) -{ + Py_XINCREF(cause); + return cause; +} + +/* Steals a reference to cause */ +void +PyException_SetCause(PyObject *self, PyObject *cause) +{ PyBaseExceptionObject *base_self = _PyBaseExceptionObject_cast(self); base_self->suppress_context = 1; Py_XSETREF(base_self->cause, cause); -} - -PyObject * +} + +PyObject * PyException_GetContext(PyObject *self) { PyObject *context = _PyBaseExceptionObject_cast(self)->context; - Py_XINCREF(context); - return context; -} - -/* Steals a reference to context */ -void -PyException_SetContext(PyObject *self, PyObject *context) -{ + Py_XINCREF(context); + return context; +} + +/* Steals a reference to context */ +void +PyException_SetContext(PyObject *self, PyObject *context) +{ Py_XSETREF(_PyBaseExceptionObject_cast(self)->context, context); -} - +} + #undef PyExceptionClass_Name - + const char * PyExceptionClass_Name(PyObject *ob) { @@ -368,847 +368,847 @@ 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*/ +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_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 + (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 * +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) { +{ + 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; + 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 * + 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) -{ +{ + 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"); @@ -1217,1096 +1217,1096 @@ OSError_written_set(PyOSErrorObject *self, PyObject *arg, void *context) 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) - */ + 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; + 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_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; + 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 + 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; - +} + + +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 this is a subclass of MemoryError, don't use the freelist * and just return a fresh object */ 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) -{ + 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) +{ BaseException_clear(self); if (!Py_IS_TYPE(self, (PyTypeObject *) PyExc_MemoryError)) { @@ -2314,212 +2314,212 @@ MemoryError_dealloc(PyBaseExceptionObject *self) return; } - _PyObject_GC_UNTRACK(self); - - if (memerrors_numfree >= MEMERRORS_SAVE) - Py_TYPE(self)->tp_free((PyObject *)self); - else { - self->dict = (PyObject *) memerrors_freelist; - memerrors_freelist = self; - memerrors_numfree++; - } -} + _PyObject_GC_UNTRACK(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); +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]); - } + } + 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 */ - +} + +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) { \ @@ -2527,7 +2527,7 @@ _PyExc_Init(void) } \ Py_INCREF(PyExc_ ## TYPE); \ } - + #define ADD_ERRNO(TYPE, CODE) \ do { \ PyObject *_code = PyLong_FromLong(CODE); \ @@ -2538,7 +2538,7 @@ _PyExc_Init(void) } \ Py_DECREF(_code); \ } while (0) - + PRE_INIT(BaseException); PRE_INIT(Exception); PRE_INIT(TypeError); @@ -2589,9 +2589,9 @@ _PyExc_Init(void) PRE_INIT(BytesWarning); PRE_INIT(ResourceWarning); - /* OSError subclasses */ + /* OSError subclasses */ PRE_INIT(ConnectionError); - + PRE_INIT(BlockingIOError); PRE_INIT(BrokenPipeError); PRE_INIT(ChildProcessError); @@ -2606,27 +2606,27 @@ _PyExc_Init(void) 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) { + 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 +#ifdef ESHUTDOWN ADD_ERRNO(BrokenPipeError, ESHUTDOWN); -#endif +#endif ADD_ERRNO(ChildProcessError, ECHILD); ADD_ERRNO(ConnectionAbortedError, ECONNABORTED); ADD_ERRNO(ConnectionRefusedError, ECONNREFUSED); @@ -2640,13 +2640,13 @@ _PyExc_Init(void) 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 @@ -2753,270 +2753,270 @@ _PyBuiltins_AddExceptions(PyObject *bltinmod) #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; +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); + 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; - } - } + + /* 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; - } - } + 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; @@ -3028,51 +3028,51 @@ _check_for_legacy_statements(PySyntaxErrorObject *self, Py_ssize_t start) 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; -} + 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; +} |