| /* Descriptors -- a new, flexible way to describe attributes */ |
| |
| #include "Python.h" |
| #include "pycore_abstract.h" // _PyObject_RealIsSubclass() |
| #include "pycore_call.h" // _PyStack_AsDict() |
| #include "pycore_ceval.h" // _Py_EnterRecursiveCallTstate() |
| #include "pycore_emscripten_trampoline.h" // descr_set_trampoline_call(), descr_get_trampoline_call() |
| #include "pycore_descrobject.h" // _PyMethodWrapper_Type |
| #include "pycore_modsupport.h" // _PyArg_UnpackStack() |
| #include "pycore_object.h" // _PyObject_GC_UNTRACK() |
| #include "pycore_pystate.h" // _PyThreadState_GET() |
| #include "pycore_tuple.h" // _PyTuple_ITEMS() |
| |
| |
| /*[clinic input] |
| class mappingproxy "mappingproxyobject *" "&PyDictProxy_Type" |
| class property "propertyobject *" "&PyProperty_Type" |
| [clinic start generated code]*/ |
| /*[clinic end generated code: output=da39a3ee5e6b4b0d input=556352653fd4c02e]*/ |
| |
| static void |
| descr_dealloc(PyObject *self) |
| { |
| PyDescrObject *descr = (PyDescrObject *)self; |
| _PyObject_GC_UNTRACK(descr); |
| Py_XDECREF(descr->d_type); |
| Py_XDECREF(descr->d_name); |
| Py_XDECREF(descr->d_qualname); |
| PyObject_GC_Del(descr); |
| } |
| |
| static PyObject * |
| descr_name(PyDescrObject *descr) |
| { |
| if (descr->d_name != NULL && PyUnicode_Check(descr->d_name)) |
| return descr->d_name; |
| return NULL; |
| } |
| |
| static PyObject * |
| descr_repr(PyDescrObject *descr, const char *format) |
| { |
| PyObject *name = NULL; |
| if (descr->d_name != NULL && PyUnicode_Check(descr->d_name)) |
| name = descr->d_name; |
| |
| return PyUnicode_FromFormat(format, name, "?", descr->d_type->tp_name); |
| } |
| |
| static PyObject * |
| method_repr(PyObject *descr) |
| { |
| return descr_repr((PyDescrObject *)descr, |
| "<method '%V' of '%s' objects>"); |
| } |
| |
| static PyObject * |
| member_repr(PyObject *descr) |
| { |
| return descr_repr((PyDescrObject *)descr, |
| "<member '%V' of '%s' objects>"); |
| } |
| |
| static PyObject * |
| getset_repr(PyObject *descr) |
| { |
| return descr_repr((PyDescrObject *)descr, |
| "<attribute '%V' of '%s' objects>"); |
| } |
| |
| static PyObject * |
| wrapperdescr_repr(PyObject *descr) |
| { |
| return descr_repr((PyDescrObject *)descr, |
| "<slot wrapper '%V' of '%s' objects>"); |
| } |
| |
| static int |
| descr_check(PyDescrObject *descr, PyObject *obj) |
| { |
| if (!PyObject_TypeCheck(obj, descr->d_type)) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' for '%.100s' objects " |
| "doesn't apply to a '%.100s' object", |
| descr_name((PyDescrObject *)descr), "?", |
| descr->d_type->tp_name, |
| Py_TYPE(obj)->tp_name); |
| return -1; |
| } |
| return 0; |
| } |
| |
| static PyObject * |
| classmethod_get(PyObject *self, PyObject *obj, PyObject *type) |
| { |
| PyMethodDescrObject *descr = (PyMethodDescrObject *)self; |
| /* Ensure a valid type. Class methods ignore obj. */ |
| if (type == NULL) { |
| if (obj != NULL) |
| type = (PyObject *)Py_TYPE(obj); |
| else { |
| /* Wot - no type?! */ |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' for type '%.100s' " |
| "needs either an object or a type", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name); |
| return NULL; |
| } |
| } |
| if (!PyType_Check(type)) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' for type '%.100s' " |
| "needs a type, not a '%.100s' as arg 2", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name, |
| Py_TYPE(type)->tp_name); |
| return NULL; |
| } |
| if (!PyType_IsSubtype((PyTypeObject *)type, PyDescr_TYPE(descr))) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' requires a subtype of '%.100s' " |
| "but received '%.100s'", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name, |
| ((PyTypeObject *)type)->tp_name); |
| return NULL; |
| } |
| PyTypeObject *cls = NULL; |
| if (descr->d_method->ml_flags & METH_METHOD) { |
| cls = descr->d_common.d_type; |
| } |
| return PyCMethod_New(descr->d_method, type, NULL, cls); |
| } |
| |
| static PyObject * |
| method_get(PyObject *self, PyObject *obj, PyObject *type) |
| { |
| PyMethodDescrObject *descr = (PyMethodDescrObject *)self; |
| if (obj == NULL) { |
| return Py_NewRef(descr); |
| } |
| if (descr_check((PyDescrObject *)descr, obj) < 0) { |
| return NULL; |
| } |
| if (descr->d_method->ml_flags & METH_METHOD) { |
| if (PyType_Check(type)) { |
| return PyCMethod_New(descr->d_method, obj, NULL, descr->d_common.d_type); |
| } else { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' needs a type, not '%s', as arg 2", |
| descr_name((PyDescrObject *)descr), |
| Py_TYPE(type)->tp_name); |
| return NULL; |
| } |
| } else { |
| return PyCFunction_NewEx(descr->d_method, obj, NULL); |
| } |
| } |
| |
| static PyObject * |
| member_get(PyObject *self, PyObject *obj, PyObject *type) |
| { |
| PyMemberDescrObject *descr = (PyMemberDescrObject *)self; |
| if (obj == NULL) { |
| return Py_NewRef(descr); |
| } |
| if (descr_check((PyDescrObject *)descr, obj) < 0) { |
| return NULL; |
| } |
| |
| if (descr->d_member->flags & Py_AUDIT_READ) { |
| if (PySys_Audit("object.__getattr__", "Os", |
| obj ? obj : Py_None, descr->d_member->name) < 0) { |
| return NULL; |
| } |
| } |
| |
| return PyMember_GetOne((char *)obj, descr->d_member); |
| } |
| |
| static PyObject * |
| getset_get(PyObject *self, PyObject *obj, PyObject *type) |
| { |
| PyGetSetDescrObject *descr = (PyGetSetDescrObject *)self; |
| if (obj == NULL) { |
| return Py_NewRef(descr); |
| } |
| if (descr_check((PyDescrObject *)descr, obj) < 0) { |
| return NULL; |
| } |
| if (descr->d_getset->get != NULL) |
| return descr_get_trampoline_call( |
| descr->d_getset->get, obj, descr->d_getset->closure); |
| PyErr_Format(PyExc_AttributeError, |
| "attribute '%V' of '%.100s' objects is not readable", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name); |
| return NULL; |
| } |
| |
| static PyObject * |
| wrapperdescr_get(PyObject *self, PyObject *obj, PyObject *type) |
| { |
| PyWrapperDescrObject *descr = (PyWrapperDescrObject *)self; |
| if (obj == NULL) { |
| return Py_NewRef(descr); |
| } |
| if (descr_check((PyDescrObject *)descr, obj) < 0) { |
| return NULL; |
| } |
| return PyWrapper_New((PyObject *)descr, obj); |
| } |
| |
| static int |
| descr_setcheck(PyDescrObject *descr, PyObject *obj, PyObject *value) |
| { |
| assert(obj != NULL); |
| if (!PyObject_TypeCheck(obj, descr->d_type)) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' for '%.100s' objects " |
| "doesn't apply to a '%.100s' object", |
| descr_name(descr), "?", |
| descr->d_type->tp_name, |
| Py_TYPE(obj)->tp_name); |
| return -1; |
| } |
| return 0; |
| } |
| |
| static int |
| member_set(PyObject *self, PyObject *obj, PyObject *value) |
| { |
| PyMemberDescrObject *descr = (PyMemberDescrObject *)self; |
| if (descr_setcheck((PyDescrObject *)descr, obj, value) < 0) { |
| return -1; |
| } |
| return PyMember_SetOne((char *)obj, descr->d_member, value); |
| } |
| |
| static int |
| getset_set(PyObject *self, PyObject *obj, PyObject *value) |
| { |
| PyGetSetDescrObject *descr = (PyGetSetDescrObject *)self; |
| if (descr_setcheck((PyDescrObject *)descr, obj, value) < 0) { |
| return -1; |
| } |
| if (descr->d_getset->set != NULL) { |
| return descr_set_trampoline_call( |
| descr->d_getset->set, obj, value, |
| descr->d_getset->closure); |
| } |
| PyErr_Format(PyExc_AttributeError, |
| "attribute '%V' of '%.100s' objects is not writable", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name); |
| return -1; |
| } |
| |
| |
| /* Vectorcall functions for each of the PyMethodDescr calling conventions. |
| * |
| * First, common helpers |
| */ |
| static inline int |
| method_check_args(PyObject *func, PyObject *const *args, Py_ssize_t nargs, PyObject *kwnames) |
| { |
| assert(!PyErr_Occurred()); |
| if (nargs < 1) { |
| PyObject *funcstr = _PyObject_FunctionStr(func); |
| if (funcstr != NULL) { |
| PyErr_Format(PyExc_TypeError, |
| "unbound method %U needs an argument", funcstr); |
| Py_DECREF(funcstr); |
| } |
| return -1; |
| } |
| PyObject *self = args[0]; |
| if (descr_check((PyDescrObject *)func, self) < 0) { |
| return -1; |
| } |
| if (kwnames && PyTuple_GET_SIZE(kwnames)) { |
| PyObject *funcstr = _PyObject_FunctionStr(func); |
| if (funcstr != NULL) { |
| PyErr_Format(PyExc_TypeError, |
| "%U takes no keyword arguments", funcstr); |
| Py_DECREF(funcstr); |
| } |
| return -1; |
| } |
| return 0; |
| } |
| |
| typedef void (*funcptr)(void); |
| |
| static inline funcptr |
| method_enter_call(PyThreadState *tstate, PyObject *func) |
| { |
| if (_Py_EnterRecursiveCallTstate(tstate, " while calling a Python object")) { |
| return NULL; |
| } |
| return (funcptr)((PyMethodDescrObject *)func)->d_method->ml_meth; |
| } |
| |
| /* Now the actual vectorcall functions */ |
| static PyObject * |
| method_vectorcall_VARARGS( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, kwnames)) { |
| return NULL; |
| } |
| PyObject *argstuple = _PyTuple_FromArray(args+1, nargs-1); |
| if (argstuple == NULL) { |
| return NULL; |
| } |
| PyCFunction meth = (PyCFunction)method_enter_call(tstate, func); |
| if (meth == NULL) { |
| Py_DECREF(argstuple); |
| return NULL; |
| } |
| PyObject *result = _PyCFunction_TrampolineCall( |
| meth, args[0], argstuple); |
| Py_DECREF(argstuple); |
| _Py_LeaveRecursiveCallTstate(tstate); |
| return result; |
| } |
| |
| static PyObject * |
| method_vectorcall_VARARGS_KEYWORDS( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, NULL)) { |
| return NULL; |
| } |
| PyObject *argstuple = _PyTuple_FromArray(args+1, nargs-1); |
| if (argstuple == NULL) { |
| return NULL; |
| } |
| PyObject *result = NULL; |
| /* Create a temporary dict for keyword arguments */ |
| PyObject *kwdict = NULL; |
| if (kwnames != NULL && PyTuple_GET_SIZE(kwnames) > 0) { |
| kwdict = _PyStack_AsDict(args + nargs, kwnames); |
| if (kwdict == NULL) { |
| goto exit; |
| } |
| } |
| PyCFunctionWithKeywords meth = (PyCFunctionWithKeywords) |
| method_enter_call(tstate, func); |
| if (meth == NULL) { |
| goto exit; |
| } |
| result = _PyCFunctionWithKeywords_TrampolineCall( |
| meth, args[0], argstuple, kwdict); |
| _Py_LeaveRecursiveCallTstate(tstate); |
| exit: |
| Py_DECREF(argstuple); |
| Py_XDECREF(kwdict); |
| return result; |
| } |
| |
| static PyObject * |
| method_vectorcall_FASTCALL_KEYWORDS_METHOD( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, NULL)) { |
| return NULL; |
| } |
| PyCMethod meth = (PyCMethod) method_enter_call(tstate, func); |
| if (meth == NULL) { |
| return NULL; |
| } |
| PyObject *result = meth(args[0], |
| ((PyMethodDescrObject *)func)->d_common.d_type, |
| args+1, nargs-1, kwnames); |
| _Py_LeaveRecursiveCall(); |
| return result; |
| } |
| |
| static PyObject * |
| method_vectorcall_FASTCALL( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, kwnames)) { |
| return NULL; |
| } |
| PyCFunctionFast meth = (PyCFunctionFast) |
| method_enter_call(tstate, func); |
| if (meth == NULL) { |
| return NULL; |
| } |
| PyObject *result = meth(args[0], args+1, nargs-1); |
| _Py_LeaveRecursiveCallTstate(tstate); |
| return result; |
| } |
| |
| static PyObject * |
| method_vectorcall_FASTCALL_KEYWORDS( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, NULL)) { |
| return NULL; |
| } |
| PyCFunctionFastWithKeywords meth = (PyCFunctionFastWithKeywords) |
| method_enter_call(tstate, func); |
| if (meth == NULL) { |
| return NULL; |
| } |
| PyObject *result = meth(args[0], args+1, nargs-1, kwnames); |
| _Py_LeaveRecursiveCallTstate(tstate); |
| return result; |
| } |
| |
| static PyObject * |
| method_vectorcall_NOARGS( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, kwnames)) { |
| return NULL; |
| } |
| if (nargs != 1) { |
| PyObject *funcstr = _PyObject_FunctionStr(func); |
| if (funcstr != NULL) { |
| PyErr_Format(PyExc_TypeError, |
| "%U takes no arguments (%zd given)", funcstr, nargs-1); |
| Py_DECREF(funcstr); |
| } |
| return NULL; |
| } |
| PyCFunction meth = (PyCFunction)method_enter_call(tstate, func); |
| if (meth == NULL) { |
| return NULL; |
| } |
| PyObject *result = _PyCFunction_TrampolineCall(meth, args[0], NULL); |
| _Py_LeaveRecursiveCallTstate(tstate); |
| return result; |
| } |
| |
| static PyObject * |
| method_vectorcall_O( |
| PyObject *func, PyObject *const *args, size_t nargsf, PyObject *kwnames) |
| { |
| PyThreadState *tstate = _PyThreadState_GET(); |
| Py_ssize_t nargs = PyVectorcall_NARGS(nargsf); |
| if (method_check_args(func, args, nargs, kwnames)) { |
| return NULL; |
| } |
| if (nargs != 2) { |
| PyObject *funcstr = _PyObject_FunctionStr(func); |
| if (funcstr != NULL) { |
| PyErr_Format(PyExc_TypeError, |
| "%U takes exactly one argument (%zd given)", |
| funcstr, nargs-1); |
| Py_DECREF(funcstr); |
| } |
| return NULL; |
| } |
| PyCFunction meth = (PyCFunction)method_enter_call(tstate, func); |
| if (meth == NULL) { |
| return NULL; |
| } |
| PyObject *result = _PyCFunction_TrampolineCall(meth, args[0], args[1]); |
| _Py_LeaveRecursiveCallTstate(tstate); |
| return result; |
| } |
| |
| |
| /* Instances of classmethod_descriptor are unlikely to be called directly. |
| For one, the analogous class "classmethod" (for Python classes) is not |
| callable. Second, users are not likely to access a classmethod_descriptor |
| directly, since it means pulling it from the class __dict__. |
| |
| This is just an excuse to say that this doesn't need to be optimized: |
| we implement this simply by calling __get__ and then calling the result. |
| */ |
| static PyObject * |
| classmethoddescr_call(PyObject *_descr, PyObject *args, |
| PyObject *kwds) |
| { |
| PyMethodDescrObject *descr = (PyMethodDescrObject *)_descr; |
| Py_ssize_t argc = PyTuple_GET_SIZE(args); |
| if (argc < 1) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' of '%.100s' " |
| "object needs an argument", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name); |
| return NULL; |
| } |
| PyObject *self = PyTuple_GET_ITEM(args, 0); |
| PyObject *bound = classmethod_get((PyObject *)descr, NULL, self); |
| if (bound == NULL) { |
| return NULL; |
| } |
| PyObject *res = PyObject_VectorcallDict(bound, _PyTuple_ITEMS(args)+1, |
| argc-1, kwds); |
| Py_DECREF(bound); |
| return res; |
| } |
| |
| Py_LOCAL_INLINE(PyObject *) |
| wrapperdescr_raw_call(PyWrapperDescrObject *descr, PyObject *self, |
| PyObject *args, PyObject *kwds) |
| { |
| wrapperfunc wrapper = descr->d_base->wrapper; |
| |
| if (descr->d_base->flags & PyWrapperFlag_KEYWORDS) { |
| wrapperfunc_kwds wk = (wrapperfunc_kwds)(void(*)(void))wrapper; |
| return (*wk)(self, args, descr->d_wrapped, kwds); |
| } |
| |
| if (kwds != NULL && (!PyDict_Check(kwds) || PyDict_GET_SIZE(kwds) != 0)) { |
| PyErr_Format(PyExc_TypeError, |
| "wrapper %s() takes no keyword arguments", |
| descr->d_base->name); |
| return NULL; |
| } |
| return (*wrapper)(self, args, descr->d_wrapped); |
| } |
| |
| static PyObject * |
| wrapperdescr_call(PyObject *_descr, PyObject *args, PyObject *kwds) |
| { |
| PyWrapperDescrObject *descr = (PyWrapperDescrObject *)_descr; |
| Py_ssize_t argc; |
| PyObject *self, *result; |
| |
| /* Make sure that the first argument is acceptable as 'self' */ |
| assert(PyTuple_Check(args)); |
| argc = PyTuple_GET_SIZE(args); |
| if (argc < 1) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' of '%.100s' " |
| "object needs an argument", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name); |
| return NULL; |
| } |
| self = PyTuple_GET_ITEM(args, 0); |
| if (!_PyObject_RealIsSubclass((PyObject *)Py_TYPE(self), |
| (PyObject *)PyDescr_TYPE(descr))) { |
| PyErr_Format(PyExc_TypeError, |
| "descriptor '%V' " |
| "requires a '%.100s' object " |
| "but received a '%.100s'", |
| descr_name((PyDescrObject *)descr), "?", |
| PyDescr_TYPE(descr)->tp_name, |
| Py_TYPE(self)->tp_name); |
| return NULL; |
| } |
| |
| args = PyTuple_GetSlice(args, 1, argc); |
| if (args == NULL) { |
| return NULL; |
| } |
| result = wrapperdescr_raw_call(descr, self, args, kwds); |
| Py_DECREF(args); |
| return result; |
| } |
| |
| |
| static PyObject * |
| method_get_doc(PyObject *_descr, void *closure) |
| { |
| PyMethodDescrObject *descr = (PyMethodDescrObject *)_descr; |
| return _PyType_GetDocFromInternalDoc(descr->d_method->ml_name, descr->d_method->ml_doc); |
| } |
| |
| static PyObject * |
| method_get_text_signature(PyObject *_descr, void *closure) |
| { |
| PyMethodDescrObject *descr = (PyMethodDescrObject *)_descr; |
| return _PyType_GetTextSignatureFromInternalDoc(descr->d_method->ml_name, |
| descr->d_method->ml_doc, |
| descr->d_method->ml_flags); |
| } |
| |
| static PyObject * |
| calculate_qualname(PyDescrObject *descr) |
| { |
| PyObject *type_qualname, *res; |
| |
| if (descr->d_name == NULL || !PyUnicode_Check(descr->d_name)) { |
| PyErr_SetString(PyExc_TypeError, |
| "<descriptor>.__name__ is not a unicode object"); |
| return NULL; |
| } |
| |
| type_qualname = PyObject_GetAttr( |
| (PyObject *)descr->d_type, &_Py_ID(__qualname__)); |
| if (type_qualname == NULL) |
| return NULL; |
| |
| if (!PyUnicode_Check(type_qualname)) { |
| PyErr_SetString(PyExc_TypeError, "<descriptor>.__objclass__." |
| "__qualname__ is not a unicode object"); |
| Py_XDECREF(type_qualname); |
| return NULL; |
| } |
| |
| res = PyUnicode_FromFormat("%S.%S", type_qualname, descr->d_name); |
| Py_DECREF(type_qualname); |
| return res; |
| } |
| |
| static PyObject * |
| descr_get_qualname(PyObject *self, void *Py_UNUSED(ignored)) |
| { |
| PyDescrObject *descr = (PyDescrObject *)self; |
| if (descr->d_qualname == NULL) |
| descr->d_qualname = calculate_qualname(descr); |
| return Py_XNewRef(descr->d_qualname); |
| } |
| |
| static PyObject * |
| descr_reduce(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| PyDescrObject *descr = (PyDescrObject *)self; |
| return Py_BuildValue("N(OO)", _PyEval_GetBuiltin(&_Py_ID(getattr)), |
| PyDescr_TYPE(descr), PyDescr_NAME(descr)); |
| } |
| |
| static PyMethodDef descr_methods[] = { |
| {"__reduce__", descr_reduce, METH_NOARGS, NULL}, |
| {NULL, NULL} |
| }; |
| |
| static PyMemberDef descr_members[] = { |
| {"__objclass__", _Py_T_OBJECT, offsetof(PyDescrObject, d_type), Py_READONLY}, |
| {"__name__", _Py_T_OBJECT, offsetof(PyDescrObject, d_name), Py_READONLY}, |
| {0} |
| }; |
| |
| static PyGetSetDef method_getset[] = { |
| {"__doc__", method_get_doc}, |
| {"__qualname__", descr_get_qualname}, |
| {"__text_signature__", method_get_text_signature}, |
| {0} |
| }; |
| |
| static PyObject * |
| member_get_doc(PyObject *_descr, void *closure) |
| { |
| PyMemberDescrObject *descr = (PyMemberDescrObject *)_descr; |
| if (descr->d_member->doc == NULL) { |
| Py_RETURN_NONE; |
| } |
| return PyUnicode_FromString(descr->d_member->doc); |
| } |
| |
| static PyGetSetDef member_getset[] = { |
| {"__doc__", member_get_doc}, |
| {"__qualname__", descr_get_qualname}, |
| {0} |
| }; |
| |
| static PyObject * |
| getset_get_doc(PyObject *self, void *closure) |
| { |
| PyGetSetDescrObject *descr = (PyGetSetDescrObject *)self; |
| if (descr->d_getset->doc == NULL) { |
| Py_RETURN_NONE; |
| } |
| return PyUnicode_FromString(descr->d_getset->doc); |
| } |
| |
| static PyGetSetDef getset_getset[] = { |
| {"__doc__", getset_get_doc}, |
| {"__qualname__", descr_get_qualname}, |
| {0} |
| }; |
| |
| static PyObject * |
| wrapperdescr_get_doc(PyObject *self, void *closure) |
| { |
| PyWrapperDescrObject *descr = (PyWrapperDescrObject *)self; |
| return _PyType_GetDocFromInternalDoc(descr->d_base->name, descr->d_base->doc); |
| } |
| |
| static PyObject * |
| wrapperdescr_get_text_signature(PyObject *self, void *closure) |
| { |
| PyWrapperDescrObject *descr = (PyWrapperDescrObject *)self; |
| return _PyType_GetTextSignatureFromInternalDoc(descr->d_base->name, |
| descr->d_base->doc, 0); |
| } |
| |
| static PyGetSetDef wrapperdescr_getset[] = { |
| {"__doc__", wrapperdescr_get_doc}, |
| {"__qualname__", descr_get_qualname}, |
| {"__text_signature__", wrapperdescr_get_text_signature}, |
| {0} |
| }; |
| |
| static int |
| descr_traverse(PyObject *self, visitproc visit, void *arg) |
| { |
| PyDescrObject *descr = (PyDescrObject *)self; |
| Py_VISIT(descr->d_type); |
| return 0; |
| } |
| |
| PyTypeObject PyMethodDescr_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "method_descriptor", |
| sizeof(PyMethodDescrObject), |
| 0, |
| descr_dealloc, /* tp_dealloc */ |
| offsetof(PyMethodDescrObject, vectorcall), /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| method_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| PyVectorcall_Call, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | |
| Py_TPFLAGS_HAVE_VECTORCALL | |
| Py_TPFLAGS_METHOD_DESCRIPTOR, /* tp_flags */ |
| 0, /* tp_doc */ |
| descr_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| descr_methods, /* tp_methods */ |
| descr_members, /* tp_members */ |
| method_getset, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| method_get, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| }; |
| |
| /* This is for METH_CLASS in C, not for "f = classmethod(f)" in Python! */ |
| PyTypeObject PyClassMethodDescr_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "classmethod_descriptor", |
| sizeof(PyMethodDescrObject), |
| 0, |
| descr_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| method_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| classmethoddescr_call, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags */ |
| 0, /* tp_doc */ |
| descr_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* tp_methods */ |
| descr_members, /* tp_members */ |
| method_getset, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| classmethod_get, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| }; |
| |
| PyTypeObject PyMemberDescr_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "member_descriptor", |
| sizeof(PyMemberDescrObject), |
| 0, |
| descr_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| member_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags */ |
| 0, /* tp_doc */ |
| descr_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| descr_methods, /* tp_methods */ |
| descr_members, /* tp_members */ |
| member_getset, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| member_get, /* tp_descr_get */ |
| member_set, /* tp_descr_set */ |
| }; |
| |
| PyTypeObject PyGetSetDescr_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "getset_descriptor", |
| sizeof(PyGetSetDescrObject), |
| 0, |
| descr_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| getset_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags */ |
| 0, /* tp_doc */ |
| descr_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| 0, /* tp_methods */ |
| descr_members, /* tp_members */ |
| getset_getset, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| getset_get, /* tp_descr_get */ |
| getset_set, /* tp_descr_set */ |
| }; |
| |
| PyTypeObject PyWrapperDescr_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "wrapper_descriptor", |
| sizeof(PyWrapperDescrObject), |
| 0, |
| descr_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| wrapperdescr_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| wrapperdescr_call, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | |
| Py_TPFLAGS_METHOD_DESCRIPTOR, /* tp_flags */ |
| 0, /* tp_doc */ |
| descr_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| descr_methods, /* tp_methods */ |
| descr_members, /* tp_members */ |
| wrapperdescr_getset, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| wrapperdescr_get, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| }; |
| |
| static PyDescrObject * |
| descr_new(PyTypeObject *descrtype, PyTypeObject *type, const char *name) |
| { |
| PyDescrObject *descr; |
| |
| descr = (PyDescrObject *)PyType_GenericAlloc(descrtype, 0); |
| if (descr != NULL) { |
| _PyObject_SetDeferredRefcount((PyObject *)descr); |
| descr->d_type = (PyTypeObject*)Py_XNewRef(type); |
| descr->d_name = PyUnicode_InternFromString(name); |
| if (descr->d_name == NULL) { |
| Py_SETREF(descr, NULL); |
| } |
| else { |
| descr->d_qualname = NULL; |
| } |
| } |
| return descr; |
| } |
| |
| PyObject * |
| PyDescr_NewMethod(PyTypeObject *type, PyMethodDef *method) |
| { |
| /* Figure out correct vectorcall function to use */ |
| vectorcallfunc vectorcall; |
| switch (method->ml_flags & (METH_VARARGS | METH_FASTCALL | METH_NOARGS | |
| METH_O | METH_KEYWORDS | METH_METHOD)) |
| { |
| case METH_VARARGS: |
| vectorcall = method_vectorcall_VARARGS; |
| break; |
| case METH_VARARGS | METH_KEYWORDS: |
| vectorcall = method_vectorcall_VARARGS_KEYWORDS; |
| break; |
| case METH_FASTCALL: |
| vectorcall = method_vectorcall_FASTCALL; |
| break; |
| case METH_FASTCALL | METH_KEYWORDS: |
| vectorcall = method_vectorcall_FASTCALL_KEYWORDS; |
| break; |
| case METH_NOARGS: |
| vectorcall = method_vectorcall_NOARGS; |
| break; |
| case METH_O: |
| vectorcall = method_vectorcall_O; |
| break; |
| case METH_METHOD | METH_FASTCALL | METH_KEYWORDS: |
| vectorcall = method_vectorcall_FASTCALL_KEYWORDS_METHOD; |
| break; |
| default: |
| PyErr_Format(PyExc_SystemError, |
| "%s() method: bad call flags", method->ml_name); |
| return NULL; |
| } |
| |
| PyMethodDescrObject *descr; |
| |
| descr = (PyMethodDescrObject *)descr_new(&PyMethodDescr_Type, |
| type, method->ml_name); |
| if (descr != NULL) { |
| descr->d_method = method; |
| descr->vectorcall = vectorcall; |
| } |
| return (PyObject *)descr; |
| } |
| |
| PyObject * |
| PyDescr_NewClassMethod(PyTypeObject *type, PyMethodDef *method) |
| { |
| PyMethodDescrObject *descr; |
| |
| descr = (PyMethodDescrObject *)descr_new(&PyClassMethodDescr_Type, |
| type, method->ml_name); |
| if (descr != NULL) |
| descr->d_method = method; |
| return (PyObject *)descr; |
| } |
| |
| PyObject * |
| PyDescr_NewMember(PyTypeObject *type, PyMemberDef *member) |
| { |
| PyMemberDescrObject *descr; |
| |
| if (member->flags & Py_RELATIVE_OFFSET) { |
| PyErr_SetString( |
| PyExc_SystemError, |
| "PyDescr_NewMember used with Py_RELATIVE_OFFSET"); |
| return NULL; |
| } |
| descr = (PyMemberDescrObject *)descr_new(&PyMemberDescr_Type, |
| type, member->name); |
| if (descr != NULL) |
| descr->d_member = member; |
| return (PyObject *)descr; |
| } |
| |
| PyObject * |
| PyDescr_NewGetSet(PyTypeObject *type, PyGetSetDef *getset) |
| { |
| PyGetSetDescrObject *descr; |
| |
| descr = (PyGetSetDescrObject *)descr_new(&PyGetSetDescr_Type, |
| type, getset->name); |
| if (descr != NULL) |
| descr->d_getset = getset; |
| return (PyObject *)descr; |
| } |
| |
| PyObject * |
| PyDescr_NewWrapper(PyTypeObject *type, struct wrapperbase *base, void *wrapped) |
| { |
| PyWrapperDescrObject *descr; |
| |
| descr = (PyWrapperDescrObject *)descr_new(&PyWrapperDescr_Type, |
| type, base->name); |
| if (descr != NULL) { |
| descr->d_base = base; |
| descr->d_wrapped = wrapped; |
| } |
| return (PyObject *)descr; |
| } |
| |
| int |
| PyDescr_IsData(PyObject *ob) |
| { |
| return Py_TYPE(ob)->tp_descr_set != NULL; |
| } |
| |
| /* --- mappingproxy: read-only proxy for mappings --- */ |
| |
| /* This has no reason to be in this file except that adding new files is a |
| bit of a pain */ |
| |
| typedef struct { |
| PyObject_HEAD |
| PyObject *mapping; |
| } mappingproxyobject; |
| |
| static Py_ssize_t |
| mappingproxy_len(PyObject *self) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_Size(pp->mapping); |
| } |
| |
| static PyObject * |
| mappingproxy_getitem(PyObject *self, PyObject *key) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_GetItem(pp->mapping, key); |
| } |
| |
| static PyMappingMethods mappingproxy_as_mapping = { |
| mappingproxy_len, /* mp_length */ |
| mappingproxy_getitem, /* mp_subscript */ |
| 0, /* mp_ass_subscript */ |
| }; |
| |
| static PyObject * |
| mappingproxy_or(PyObject *left, PyObject *right) |
| { |
| if (PyObject_TypeCheck(left, &PyDictProxy_Type)) { |
| left = ((mappingproxyobject*)left)->mapping; |
| } |
| if (PyObject_TypeCheck(right, &PyDictProxy_Type)) { |
| right = ((mappingproxyobject*)right)->mapping; |
| } |
| return PyNumber_Or(left, right); |
| } |
| |
| static PyObject * |
| mappingproxy_ior(PyObject *self, PyObject *Py_UNUSED(other)) |
| { |
| return PyErr_Format(PyExc_TypeError, |
| "'|=' is not supported by %s; use '|' instead", Py_TYPE(self)->tp_name); |
| } |
| |
| static PyNumberMethods mappingproxy_as_number = { |
| .nb_or = mappingproxy_or, |
| .nb_inplace_or = mappingproxy_ior, |
| }; |
| |
| static int |
| mappingproxy_contains(PyObject *self, PyObject *key) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| if (PyDict_CheckExact(pp->mapping)) |
| return PyDict_Contains(pp->mapping, key); |
| else |
| return PySequence_Contains(pp->mapping, key); |
| } |
| |
| static PySequenceMethods mappingproxy_as_sequence = { |
| 0, /* sq_length */ |
| 0, /* sq_concat */ |
| 0, /* sq_repeat */ |
| 0, /* sq_item */ |
| 0, /* sq_slice */ |
| 0, /* sq_ass_item */ |
| 0, /* sq_ass_slice */ |
| mappingproxy_contains, /* sq_contains */ |
| 0, /* sq_inplace_concat */ |
| 0, /* sq_inplace_repeat */ |
| }; |
| |
| static PyObject * |
| mappingproxy_get(PyObject *self, PyObject *const *args, Py_ssize_t nargs) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| /* newargs: mapping, key, default=None */ |
| PyObject *newargs[3]; |
| newargs[0] = pp->mapping; |
| newargs[2] = Py_None; |
| |
| if (!_PyArg_UnpackStack(args, nargs, "get", 1, 2, |
| &newargs[1], &newargs[2])) |
| { |
| return NULL; |
| } |
| return PyObject_VectorcallMethod(&_Py_ID(get), newargs, |
| 3 | PY_VECTORCALL_ARGUMENTS_OFFSET, |
| NULL); |
| } |
| |
| static PyObject * |
| mappingproxy_keys(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_CallMethodNoArgs(pp->mapping, &_Py_ID(keys)); |
| } |
| |
| static PyObject * |
| mappingproxy_values(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_CallMethodNoArgs(pp->mapping, &_Py_ID(values)); |
| } |
| |
| static PyObject * |
| mappingproxy_items(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_CallMethodNoArgs(pp->mapping, &_Py_ID(items)); |
| } |
| |
| static PyObject * |
| mappingproxy_copy(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_CallMethodNoArgs(pp->mapping, &_Py_ID(copy)); |
| } |
| |
| static PyObject * |
| mappingproxy_reversed(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_CallMethodNoArgs(pp->mapping, &_Py_ID(__reversed__)); |
| } |
| |
| /* WARNING: mappingproxy methods must not give access |
| to the underlying mapping */ |
| |
| static PyMethodDef mappingproxy_methods[] = { |
| {"get", _PyCFunction_CAST(mappingproxy_get), METH_FASTCALL, |
| PyDoc_STR("get($self, key, default=None, /)\n--\n\n" |
| "Return the value for key if key is in the mapping, else default.")}, |
| {"keys", mappingproxy_keys, METH_NOARGS, |
| PyDoc_STR("D.keys() -> a set-like object providing a view on D's keys")}, |
| {"values", mappingproxy_values, METH_NOARGS, |
| PyDoc_STR("D.values() -> an object providing a view on D's values")}, |
| {"items", mappingproxy_items, METH_NOARGS, |
| PyDoc_STR("D.items() -> a set-like object providing a view on D's items")}, |
| {"copy", mappingproxy_copy, METH_NOARGS, |
| PyDoc_STR("D.copy() -> a shallow copy of D")}, |
| {"__class_getitem__", Py_GenericAlias, METH_O|METH_CLASS, |
| PyDoc_STR("See PEP 585")}, |
| {"__reversed__", mappingproxy_reversed, METH_NOARGS, |
| PyDoc_STR("D.__reversed__() -> reverse iterator")}, |
| {0} |
| }; |
| |
| static void |
| mappingproxy_dealloc(PyObject *self) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| _PyObject_GC_UNTRACK(pp); |
| Py_DECREF(pp->mapping); |
| PyObject_GC_Del(pp); |
| } |
| |
| static PyObject * |
| mappingproxy_getiter(PyObject *self) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_GetIter(pp->mapping); |
| } |
| |
| static Py_hash_t |
| mappingproxy_hash(PyObject *self) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_Hash(pp->mapping); |
| } |
| |
| static PyObject * |
| mappingproxy_str(PyObject *self) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyObject_Str(pp->mapping); |
| } |
| |
| static PyObject * |
| mappingproxy_repr(PyObject *self) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| return PyUnicode_FromFormat("mappingproxy(%R)", pp->mapping); |
| } |
| |
| static int |
| mappingproxy_traverse(PyObject *self, visitproc visit, void *arg) |
| { |
| mappingproxyobject *pp = (mappingproxyobject *)self; |
| Py_VISIT(pp->mapping); |
| return 0; |
| } |
| |
| static PyObject * |
| mappingproxy_richcompare(PyObject *self, PyObject *w, int op) |
| { |
| mappingproxyobject *v = (mappingproxyobject *)self; |
| return PyObject_RichCompare(v->mapping, w, op); |
| } |
| |
| static int |
| mappingproxy_check_mapping(PyObject *mapping) |
| { |
| if (!PyMapping_Check(mapping) |
| || PyList_Check(mapping) |
| || PyTuple_Check(mapping)) { |
| PyErr_Format(PyExc_TypeError, |
| "mappingproxy() argument must be a mapping, not %s", |
| Py_TYPE(mapping)->tp_name); |
| return -1; |
| } |
| return 0; |
| } |
| |
| /*[clinic input] |
| @classmethod |
| mappingproxy.__new__ as mappingproxy_new |
| |
| mapping: object |
| |
| Read-only proxy of a mapping. |
| [clinic start generated code]*/ |
| |
| static PyObject * |
| mappingproxy_new_impl(PyTypeObject *type, PyObject *mapping) |
| /*[clinic end generated code: output=65f27f02d5b68fa7 input=c156df096ef7590c]*/ |
| { |
| mappingproxyobject *mappingproxy; |
| |
| if (mappingproxy_check_mapping(mapping) == -1) |
| return NULL; |
| |
| mappingproxy = PyObject_GC_New(mappingproxyobject, &PyDictProxy_Type); |
| if (mappingproxy == NULL) |
| return NULL; |
| mappingproxy->mapping = Py_NewRef(mapping); |
| _PyObject_GC_TRACK(mappingproxy); |
| return (PyObject *)mappingproxy; |
| } |
| |
| PyObject * |
| PyDictProxy_New(PyObject *mapping) |
| { |
| mappingproxyobject *pp; |
| |
| if (mappingproxy_check_mapping(mapping) == -1) |
| return NULL; |
| |
| pp = PyObject_GC_New(mappingproxyobject, &PyDictProxy_Type); |
| if (pp != NULL) { |
| pp->mapping = Py_NewRef(mapping); |
| _PyObject_GC_TRACK(pp); |
| } |
| return (PyObject *)pp; |
| } |
| |
| |
| /* --- Wrapper object for "slot" methods --- */ |
| |
| /* This has no reason to be in this file except that adding new files is a |
| bit of a pain */ |
| |
| typedef struct { |
| PyObject_HEAD |
| PyWrapperDescrObject *descr; |
| PyObject *self; |
| } wrapperobject; |
| |
| #define Wrapper_Check(v) Py_IS_TYPE(v, &_PyMethodWrapper_Type) |
| |
| static void |
| wrapper_dealloc(PyObject *self) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| PyObject_GC_UnTrack(wp); |
| Py_TRASHCAN_BEGIN(wp, wrapper_dealloc) |
| Py_XDECREF(wp->descr); |
| Py_XDECREF(wp->self); |
| PyObject_GC_Del(wp); |
| Py_TRASHCAN_END |
| } |
| |
| static PyObject * |
| wrapper_richcompare(PyObject *a, PyObject *b, int op) |
| { |
| wrapperobject *wa, *wb; |
| int eq; |
| |
| assert(a != NULL && b != NULL); |
| |
| /* both arguments should be wrapperobjects */ |
| if ((op != Py_EQ && op != Py_NE) |
| || !Wrapper_Check(a) || !Wrapper_Check(b)) |
| { |
| Py_RETURN_NOTIMPLEMENTED; |
| } |
| |
| wa = (wrapperobject *)a; |
| wb = (wrapperobject *)b; |
| eq = (wa->descr == wb->descr && wa->self == wb->self); |
| if (eq == (op == Py_EQ)) { |
| Py_RETURN_TRUE; |
| } |
| else { |
| Py_RETURN_FALSE; |
| } |
| } |
| |
| static Py_hash_t |
| wrapper_hash(PyObject *self) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| Py_hash_t x, y; |
| x = PyObject_GenericHash(wp->self); |
| y = _Py_HashPointer(wp->descr); |
| x = x ^ y; |
| if (x == -1) |
| x = -2; |
| return x; |
| } |
| |
| static PyObject * |
| wrapper_repr(PyObject *self) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| return PyUnicode_FromFormat("<method-wrapper '%s' of %s object at %p>", |
| wp->descr->d_base->name, |
| Py_TYPE(wp->self)->tp_name, |
| wp->self); |
| } |
| |
| static PyObject * |
| wrapper_reduce(PyObject *self, PyObject *Py_UNUSED(ignored)) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| return Py_BuildValue("N(OO)", _PyEval_GetBuiltin(&_Py_ID(getattr)), |
| wp->self, PyDescr_NAME(wp->descr)); |
| } |
| |
| static PyMethodDef wrapper_methods[] = { |
| {"__reduce__", wrapper_reduce, METH_NOARGS, NULL}, |
| {NULL, NULL} |
| }; |
| |
| static PyMemberDef wrapper_members[] = { |
| {"__self__", _Py_T_OBJECT, offsetof(wrapperobject, self), Py_READONLY}, |
| {0} |
| }; |
| |
| static PyObject * |
| wrapper_objclass(PyObject *wp, void *Py_UNUSED(ignored)) |
| { |
| PyObject *c = (PyObject *)PyDescr_TYPE(((wrapperobject *)wp)->descr); |
| |
| return Py_NewRef(c); |
| } |
| |
| static PyObject * |
| wrapper_name(PyObject *wp, void *Py_UNUSED(ignored)) |
| { |
| const char *s = ((wrapperobject *)wp)->descr->d_base->name; |
| |
| return PyUnicode_FromString(s); |
| } |
| |
| static PyObject * |
| wrapper_doc(PyObject *self, void *Py_UNUSED(ignored)) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| return _PyType_GetDocFromInternalDoc(wp->descr->d_base->name, wp->descr->d_base->doc); |
| } |
| |
| static PyObject * |
| wrapper_text_signature(PyObject *self, void *Py_UNUSED(ignored)) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| return _PyType_GetTextSignatureFromInternalDoc(wp->descr->d_base->name, |
| wp->descr->d_base->doc, 0); |
| } |
| |
| static PyObject * |
| wrapper_qualname(PyObject *self, void *Py_UNUSED(ignored)) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| return descr_get_qualname((PyObject *)wp->descr, NULL); |
| } |
| |
| static PyGetSetDef wrapper_getsets[] = { |
| {"__objclass__", wrapper_objclass}, |
| {"__name__", wrapper_name}, |
| {"__qualname__", wrapper_qualname}, |
| {"__doc__", wrapper_doc}, |
| {"__text_signature__", wrapper_text_signature}, |
| {0} |
| }; |
| |
| static PyObject * |
| wrapper_call(PyObject *self, PyObject *args, PyObject *kwds) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| return wrapperdescr_raw_call(wp->descr, wp->self, args, kwds); |
| } |
| |
| static int |
| wrapper_traverse(PyObject *self, visitproc visit, void *arg) |
| { |
| wrapperobject *wp = (wrapperobject *)self; |
| Py_VISIT(wp->descr); |
| Py_VISIT(wp->self); |
| return 0; |
| } |
| |
| PyTypeObject _PyMethodWrapper_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "method-wrapper", /* tp_name */ |
| sizeof(wrapperobject), /* tp_basicsize */ |
| 0, /* tp_itemsize */ |
| /* methods */ |
| wrapper_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| wrapper_repr, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| wrapper_hash, /* tp_hash */ |
| wrapper_call, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags */ |
| 0, /* tp_doc */ |
| wrapper_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| wrapper_richcompare, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| wrapper_methods, /* tp_methods */ |
| wrapper_members, /* tp_members */ |
| wrapper_getsets, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| }; |
| |
| PyObject * |
| PyWrapper_New(PyObject *d, PyObject *self) |
| { |
| wrapperobject *wp; |
| PyWrapperDescrObject *descr; |
| |
| assert(PyObject_TypeCheck(d, &PyWrapperDescr_Type)); |
| descr = (PyWrapperDescrObject *)d; |
| assert(_PyObject_RealIsSubclass((PyObject *)Py_TYPE(self), |
| (PyObject *)PyDescr_TYPE(descr))); |
| |
| wp = PyObject_GC_New(wrapperobject, &_PyMethodWrapper_Type); |
| if (wp != NULL) { |
| wp->descr = (PyWrapperDescrObject*)Py_NewRef(descr); |
| wp->self = Py_NewRef(self); |
| _PyObject_GC_TRACK(wp); |
| } |
| return (PyObject *)wp; |
| } |
| |
| |
| /* A built-in 'property' type */ |
| |
| /* |
| class property(object): |
| |
| def __init__(self, fget=None, fset=None, fdel=None, doc=None): |
| if doc is None and fget is not None and hasattr(fget, "__doc__"): |
| doc = fget.__doc__ |
| self.__get = fget |
| self.__set = fset |
| self.__del = fdel |
| try: |
| self.__doc__ = doc |
| except AttributeError: # read-only or dict-less class |
| pass |
| self.__name = None |
| |
| def __set_name__(self, owner, name): |
| self.__name = name |
| |
| @property |
| def __name__(self): |
| return self.__name if self.__name is not None else self.fget.__name__ |
| |
| @__name__.setter |
| def __name__(self, value): |
| self.__name = value |
| |
| def __get__(self, inst, type=None): |
| if inst is None: |
| return self |
| if self.__get is None: |
| raise AttributeError("property has no getter") |
| return self.__get(inst) |
| |
| def __set__(self, inst, value): |
| if self.__set is None: |
| raise AttributeError("property has no setter") |
| return self.__set(inst, value) |
| |
| def __delete__(self, inst): |
| if self.__del is None: |
| raise AttributeError("property has no deleter") |
| return self.__del(inst) |
| |
| */ |
| |
| static PyObject * property_copy(PyObject *, PyObject *, PyObject *, |
| PyObject *); |
| |
| static PyMemberDef property_members[] = { |
| {"fget", _Py_T_OBJECT, offsetof(propertyobject, prop_get), Py_READONLY}, |
| {"fset", _Py_T_OBJECT, offsetof(propertyobject, prop_set), Py_READONLY}, |
| {"fdel", _Py_T_OBJECT, offsetof(propertyobject, prop_del), Py_READONLY}, |
| {"__doc__", _Py_T_OBJECT, offsetof(propertyobject, prop_doc), 0}, |
| {0} |
| }; |
| |
| |
| PyDoc_STRVAR(getter_doc, |
| "Descriptor to obtain a copy of the property with a different getter."); |
| |
| static PyObject * |
| property_getter(PyObject *self, PyObject *getter) |
| { |
| return property_copy(self, getter, NULL, NULL); |
| } |
| |
| |
| PyDoc_STRVAR(setter_doc, |
| "Descriptor to obtain a copy of the property with a different setter."); |
| |
| static PyObject * |
| property_setter(PyObject *self, PyObject *setter) |
| { |
| return property_copy(self, NULL, setter, NULL); |
| } |
| |
| |
| PyDoc_STRVAR(deleter_doc, |
| "Descriptor to obtain a copy of the property with a different deleter."); |
| |
| static PyObject * |
| property_deleter(PyObject *self, PyObject *deleter) |
| { |
| return property_copy(self, NULL, NULL, deleter); |
| } |
| |
| |
| PyDoc_STRVAR(set_name_doc, |
| "__set_name__($self, owner, name, /)\n" |
| "--\n" |
| "\n" |
| "Method to set name of a property."); |
| |
| static PyObject * |
| property_set_name(PyObject *self, PyObject *args) { |
| if (PyTuple_GET_SIZE(args) != 2) { |
| PyErr_Format( |
| PyExc_TypeError, |
| "__set_name__() takes 2 positional arguments but %d were given", |
| PyTuple_GET_SIZE(args)); |
| return NULL; |
| } |
| |
| propertyobject *prop = (propertyobject *)self; |
| PyObject *name = PyTuple_GET_ITEM(args, 1); |
| |
| Py_XSETREF(prop->prop_name, Py_XNewRef(name)); |
| |
| Py_RETURN_NONE; |
| } |
| |
| static PyMethodDef property_methods[] = { |
| {"getter", property_getter, METH_O, getter_doc}, |
| {"setter", property_setter, METH_O, setter_doc}, |
| {"deleter", property_deleter, METH_O, deleter_doc}, |
| {"__set_name__", property_set_name, METH_VARARGS, set_name_doc}, |
| {0} |
| }; |
| |
| |
| static void |
| property_dealloc(PyObject *self) |
| { |
| propertyobject *gs = (propertyobject *)self; |
| |
| _PyObject_GC_UNTRACK(self); |
| Py_XDECREF(gs->prop_get); |
| Py_XDECREF(gs->prop_set); |
| Py_XDECREF(gs->prop_del); |
| Py_XDECREF(gs->prop_doc); |
| Py_XDECREF(gs->prop_name); |
| Py_TYPE(self)->tp_free(self); |
| } |
| |
| static int |
| property_name(propertyobject *prop, PyObject **name) |
| { |
| if (prop->prop_name != NULL) { |
| *name = Py_NewRef(prop->prop_name); |
| return 1; |
| } |
| if (prop->prop_get == NULL) { |
| *name = NULL; |
| return 0; |
| } |
| return PyObject_GetOptionalAttr(prop->prop_get, &_Py_ID(__name__), name); |
| } |
| |
| static PyObject * |
| property_descr_get(PyObject *self, PyObject *obj, PyObject *type) |
| { |
| if (obj == NULL || obj == Py_None) { |
| return Py_NewRef(self); |
| } |
| |
| propertyobject *gs = (propertyobject *)self; |
| if (gs->prop_get == NULL) { |
| PyObject *propname; |
| if (property_name(gs, &propname) < 0) { |
| return NULL; |
| } |
| PyObject *qualname = PyType_GetQualName(Py_TYPE(obj)); |
| if (propname != NULL && qualname != NULL) { |
| PyErr_Format(PyExc_AttributeError, |
| "property %R of %R object has no getter", |
| propname, |
| qualname); |
| } |
| else if (qualname != NULL) { |
| PyErr_Format(PyExc_AttributeError, |
| "property of %R object has no getter", |
| qualname); |
| } else { |
| PyErr_SetString(PyExc_AttributeError, |
| "property has no getter"); |
| } |
| Py_XDECREF(propname); |
| Py_XDECREF(qualname); |
| return NULL; |
| } |
| |
| return PyObject_CallOneArg(gs->prop_get, obj); |
| } |
| |
| static int |
| property_descr_set(PyObject *self, PyObject *obj, PyObject *value) |
| { |
| propertyobject *gs = (propertyobject *)self; |
| PyObject *func, *res; |
| |
| if (value == NULL) { |
| func = gs->prop_del; |
| } |
| else { |
| func = gs->prop_set; |
| } |
| |
| if (func == NULL) { |
| PyObject *propname; |
| if (property_name(gs, &propname) < 0) { |
| return -1; |
| } |
| PyObject *qualname = NULL; |
| if (obj != NULL) { |
| qualname = PyType_GetQualName(Py_TYPE(obj)); |
| } |
| if (propname != NULL && qualname != NULL) { |
| PyErr_Format(PyExc_AttributeError, |
| value == NULL ? |
| "property %R of %R object has no deleter" : |
| "property %R of %R object has no setter", |
| propname, |
| qualname); |
| } |
| else if (qualname != NULL) { |
| PyErr_Format(PyExc_AttributeError, |
| value == NULL ? |
| "property of %R object has no deleter" : |
| "property of %R object has no setter", |
| qualname); |
| } |
| else { |
| PyErr_SetString(PyExc_AttributeError, |
| value == NULL ? |
| "property has no deleter" : |
| "property has no setter"); |
| } |
| Py_XDECREF(propname); |
| Py_XDECREF(qualname); |
| return -1; |
| } |
| |
| if (value == NULL) { |
| res = PyObject_CallOneArg(func, obj); |
| } |
| else { |
| EVAL_CALL_STAT_INC_IF_FUNCTION(EVAL_CALL_API, func); |
| PyObject *args[] = { obj, value }; |
| res = PyObject_Vectorcall(func, args, 2, NULL); |
| } |
| |
| if (res == NULL) { |
| return -1; |
| } |
| |
| Py_DECREF(res); |
| return 0; |
| } |
| |
| static PyObject * |
| property_copy(PyObject *old, PyObject *get, PyObject *set, PyObject *del) |
| { |
| propertyobject *pold = (propertyobject *)old; |
| PyObject *new, *type, *doc; |
| |
| type = PyObject_Type(old); |
| if (type == NULL) |
| return NULL; |
| |
| if (get == NULL || get == Py_None) { |
| get = pold->prop_get ? pold->prop_get : Py_None; |
| } |
| if (set == NULL || set == Py_None) { |
| set = pold->prop_set ? pold->prop_set : Py_None; |
| } |
| if (del == NULL || del == Py_None) { |
| del = pold->prop_del ? pold->prop_del : Py_None; |
| } |
| if (pold->getter_doc && get != Py_None) { |
| /* make _init use __doc__ from getter */ |
| doc = Py_None; |
| } |
| else { |
| doc = pold->prop_doc ? pold->prop_doc : Py_None; |
| } |
| |
| new = PyObject_CallFunctionObjArgs(type, get, set, del, doc, NULL); |
| Py_DECREF(type); |
| if (new == NULL) |
| return NULL; |
| |
| if (PyObject_TypeCheck((new), &PyProperty_Type)) { |
| Py_XSETREF(((propertyobject *) new)->prop_name, Py_XNewRef(pold->prop_name)); |
| } |
| return new; |
| } |
| |
| /*[clinic input] |
| property.__init__ as property_init |
| |
| fget: object(c_default="NULL") = None |
| function to be used for getting an attribute value |
| fset: object(c_default="NULL") = None |
| function to be used for setting an attribute value |
| fdel: object(c_default="NULL") = None |
| function to be used for del'ing an attribute |
| doc: object(c_default="NULL") = None |
| docstring |
| |
| Property attribute. |
| |
| Typical use is to define a managed attribute x: |
| |
| class C(object): |
| def getx(self): return self._x |
| def setx(self, value): self._x = value |
| def delx(self): del self._x |
| x = property(getx, setx, delx, "I'm the 'x' property.") |
| |
| Decorators make defining new properties or modifying existing ones easy: |
| |
| class C(object): |
| @property |
| def x(self): |
| "I am the 'x' property." |
| return self._x |
| @x.setter |
| def x(self, value): |
| self._x = value |
| @x.deleter |
| def x(self): |
| del self._x |
| [clinic start generated code]*/ |
| |
| static int |
| property_init_impl(propertyobject *self, PyObject *fget, PyObject *fset, |
| PyObject *fdel, PyObject *doc) |
| /*[clinic end generated code: output=01a960742b692b57 input=dfb5dbbffc6932d5]*/ |
| { |
| if (fget == Py_None) |
| fget = NULL; |
| if (fset == Py_None) |
| fset = NULL; |
| if (fdel == Py_None) |
| fdel = NULL; |
| |
| Py_XSETREF(self->prop_get, Py_XNewRef(fget)); |
| Py_XSETREF(self->prop_set, Py_XNewRef(fset)); |
| Py_XSETREF(self->prop_del, Py_XNewRef(fdel)); |
| Py_XSETREF(self->prop_doc, NULL); |
| Py_XSETREF(self->prop_name, NULL); |
| |
| self->getter_doc = 0; |
| PyObject *prop_doc = NULL; |
| |
| if (doc != NULL && doc != Py_None) { |
| prop_doc = Py_XNewRef(doc); |
| } |
| /* if no docstring given and the getter has one, use that one */ |
| else if (fget != NULL) { |
| int rc = PyObject_GetOptionalAttr(fget, &_Py_ID(__doc__), &prop_doc); |
| if (rc < 0) { |
| return rc; |
| } |
| if (prop_doc == Py_None) { |
| prop_doc = NULL; |
| Py_DECREF(Py_None); |
| } |
| if (prop_doc != NULL){ |
| self->getter_doc = 1; |
| } |
| } |
| |
| /* At this point `prop_doc` is either NULL or |
| a non-None object with incremented ref counter */ |
| |
| if (Py_IS_TYPE(self, &PyProperty_Type)) { |
| Py_XSETREF(self->prop_doc, prop_doc); |
| } else { |
| /* If this is a property subclass, put __doc__ in the dict |
| or designated slot of the subclass instance instead, otherwise |
| it gets shadowed by __doc__ in the class's dict. */ |
| |
| if (prop_doc == NULL) { |
| prop_doc = Py_NewRef(Py_None); |
| } |
| int err = PyObject_SetAttr( |
| (PyObject *)self, &_Py_ID(__doc__), prop_doc); |
| Py_DECREF(prop_doc); |
| if (err < 0) { |
| assert(PyErr_Occurred()); |
| if (!self->getter_doc && |
| PyErr_ExceptionMatches(PyExc_AttributeError)) |
| { |
| PyErr_Clear(); |
| // https://github.com/python/cpython/issues/98963#issuecomment-1574413319 |
| // Python silently dropped this doc assignment through 3.11. |
| // We preserve that behavior for backwards compatibility. |
| // |
| // If we ever want to deprecate this behavior, only raise a |
| // warning or error when proc_doc is not None so that |
| // property without a specific doc= still works. |
| return 0; |
| } else { |
| return -1; |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| static PyObject * |
| property_get__name__(propertyobject *prop, void *Py_UNUSED(ignored)) |
| { |
| PyObject *name; |
| if (property_name(prop, &name) < 0) { |
| return NULL; |
| } |
| if (name == NULL) { |
| PyErr_SetString(PyExc_AttributeError, |
| "'property' object has no attribute '__name__'"); |
| } |
| return name; |
| } |
| |
| static int |
| property_set__name__(propertyobject *prop, PyObject *value, |
| void *Py_UNUSED(ignored)) |
| { |
| Py_XSETREF(prop->prop_name, Py_XNewRef(value)); |
| return 0; |
| } |
| |
| static PyObject * |
| property_get___isabstractmethod__(propertyobject *prop, void *closure) |
| { |
| int res = _PyObject_IsAbstract(prop->prop_get); |
| if (res == -1) { |
| return NULL; |
| } |
| else if (res) { |
| Py_RETURN_TRUE; |
| } |
| |
| res = _PyObject_IsAbstract(prop->prop_set); |
| if (res == -1) { |
| return NULL; |
| } |
| else if (res) { |
| Py_RETURN_TRUE; |
| } |
| |
| res = _PyObject_IsAbstract(prop->prop_del); |
| if (res == -1) { |
| return NULL; |
| } |
| else if (res) { |
| Py_RETURN_TRUE; |
| } |
| Py_RETURN_FALSE; |
| } |
| |
| static PyGetSetDef property_getsetlist[] = { |
| {"__name__", (getter)property_get__name__, (setter)property_set__name__}, |
| {"__isabstractmethod__", |
| (getter)property_get___isabstractmethod__, NULL, |
| NULL, |
| NULL}, |
| {NULL} /* Sentinel */ |
| }; |
| |
| static int |
| property_traverse(PyObject *self, visitproc visit, void *arg) |
| { |
| propertyobject *pp = (propertyobject *)self; |
| Py_VISIT(pp->prop_get); |
| Py_VISIT(pp->prop_set); |
| Py_VISIT(pp->prop_del); |
| Py_VISIT(pp->prop_doc); |
| Py_VISIT(pp->prop_name); |
| return 0; |
| } |
| |
| static int |
| property_clear(PyObject *self) |
| { |
| propertyobject *pp = (propertyobject *)self; |
| Py_CLEAR(pp->prop_doc); |
| return 0; |
| } |
| |
| #include "clinic/descrobject.c.h" |
| |
| PyTypeObject PyDictProxy_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "mappingproxy", /* tp_name */ |
| sizeof(mappingproxyobject), /* tp_basicsize */ |
| 0, /* tp_itemsize */ |
| /* methods */ |
| mappingproxy_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| mappingproxy_repr, /* tp_repr */ |
| &mappingproxy_as_number, /* tp_as_number */ |
| &mappingproxy_as_sequence, /* tp_as_sequence */ |
| &mappingproxy_as_mapping, /* tp_as_mapping */ |
| mappingproxy_hash, /* tp_hash */ |
| 0, /* tp_call */ |
| mappingproxy_str, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | |
| Py_TPFLAGS_MAPPING, /* tp_flags */ |
| mappingproxy_new__doc__, /* tp_doc */ |
| mappingproxy_traverse, /* tp_traverse */ |
| 0, /* tp_clear */ |
| mappingproxy_richcompare, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| mappingproxy_getiter, /* tp_iter */ |
| 0, /* tp_iternext */ |
| mappingproxy_methods, /* tp_methods */ |
| 0, /* tp_members */ |
| 0, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| 0, /* tp_descr_get */ |
| 0, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| 0, /* tp_init */ |
| 0, /* tp_alloc */ |
| mappingproxy_new, /* tp_new */ |
| }; |
| |
| PyTypeObject PyProperty_Type = { |
| PyVarObject_HEAD_INIT(&PyType_Type, 0) |
| "property", /* tp_name */ |
| sizeof(propertyobject), /* tp_basicsize */ |
| 0, /* tp_itemsize */ |
| /* methods */ |
| property_dealloc, /* tp_dealloc */ |
| 0, /* tp_vectorcall_offset */ |
| 0, /* tp_getattr */ |
| 0, /* tp_setattr */ |
| 0, /* tp_as_async */ |
| 0, /* tp_repr */ |
| 0, /* tp_as_number */ |
| 0, /* tp_as_sequence */ |
| 0, /* tp_as_mapping */ |
| 0, /* tp_hash */ |
| 0, /* tp_call */ |
| 0, /* tp_str */ |
| PyObject_GenericGetAttr, /* tp_getattro */ |
| 0, /* tp_setattro */ |
| 0, /* tp_as_buffer */ |
| Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | |
| Py_TPFLAGS_BASETYPE, /* tp_flags */ |
| property_init__doc__, /* tp_doc */ |
| property_traverse, /* tp_traverse */ |
| property_clear, /* tp_clear */ |
| 0, /* tp_richcompare */ |
| 0, /* tp_weaklistoffset */ |
| 0, /* tp_iter */ |
| 0, /* tp_iternext */ |
| property_methods, /* tp_methods */ |
| property_members, /* tp_members */ |
| property_getsetlist, /* tp_getset */ |
| 0, /* tp_base */ |
| 0, /* tp_dict */ |
| property_descr_get, /* tp_descr_get */ |
| property_descr_set, /* tp_descr_set */ |
| 0, /* tp_dictoffset */ |
| property_init, /* tp_init */ |
| PyType_GenericAlloc, /* tp_alloc */ |
| PyType_GenericNew, /* tp_new */ |
| PyObject_GC_Del, /* tp_free */ |
| }; |