#include "Python.h" #include "pycore_dict.h" // _PyDict_DelItemIf() #include "pycore_object.h" // _PyObject_GET_WEAKREFS_LISTPTR() #include "pycore_weakref.h" // _PyWeakref_IS_DEAD() #define GET_WEAKREFS_LISTPTR(o) \ ((PyWeakReference **) _PyObject_GET_WEAKREFS_LISTPTR(o)) /*[clinic input] module _weakref [clinic start generated code]*/ /*[clinic end generated code: output=da39a3ee5e6b4b0d input=ffec73b85846596d]*/ #include "clinic/_weakref.c.h" /*[clinic input] @critical_section object _weakref.getweakrefcount -> Py_ssize_t object: object / Return the number of weak references to 'object'. [clinic start generated code]*/ static Py_ssize_t _weakref_getweakrefcount_impl(PyObject *module, PyObject *object) /*[clinic end generated code: output=301806d59558ff3e input=6535a580f1d0ebdc]*/ { if (!_PyType_SUPPORTS_WEAKREFS(Py_TYPE(object))) { return 0; } PyWeakReference **list = GET_WEAKREFS_LISTPTR(object); Py_ssize_t count = _PyWeakref_GetWeakrefCount(*list); return count; } static int is_dead_weakref(PyObject *value) { if (!PyWeakref_Check(value)) { PyErr_SetString(PyExc_TypeError, "not a weakref"); return -1; } return _PyWeakref_IS_DEAD(value); } /*[clinic input] _weakref._remove_dead_weakref -> object dct: object(subclass_of='&PyDict_Type') key: object / Atomically remove key from dict if it points to a dead weakref. [clinic start generated code]*/ static PyObject * _weakref__remove_dead_weakref_impl(PyObject *module, PyObject *dct, PyObject *key) /*[clinic end generated code: output=d9ff53061fcb875c input=19fc91f257f96a1d]*/ { if (_PyDict_DelItemIf(dct, key, is_dead_weakref) < 0) { if (PyErr_ExceptionMatches(PyExc_KeyError)) /* This function is meant to allow safe weak-value dicts with GC in another thread (see issue #28427), so it's ok if the key doesn't exist anymore. */ PyErr_Clear(); else return NULL; } Py_RETURN_NONE; } /*[clinic input] @critical_section object _weakref.getweakrefs object: object / Return a list of all weak reference objects pointing to 'object'. [clinic start generated code]*/ static PyObject * _weakref_getweakrefs_impl(PyObject *module, PyObject *object) /*[clinic end generated code: output=5ec268989fb8f035 input=3dea95b8f5b31bbb]*/ { if (!_PyType_SUPPORTS_WEAKREFS(Py_TYPE(object))) { return PyList_New(0); } PyWeakReference **list = GET_WEAKREFS_LISTPTR(object); Py_ssize_t count = _PyWeakref_GetWeakrefCount(*list); PyObject *result = PyList_New(count); if (result == NULL) { return NULL; } PyWeakReference *current = *list; for (Py_ssize_t i = 0; i < count; ++i) { PyList_SET_ITEM(result, i, Py_NewRef(current)); current = current->wr_next; } return result; } /*[clinic input] _weakref.proxy object: object callback: object(c_default="NULL") = None / Create a proxy object that weakly references 'object'. 'callback', if given, is called with a reference to the proxy when 'object' is about to be finalized. [clinic start generated code]*/ static PyObject * _weakref_proxy_impl(PyObject *module, PyObject *object, PyObject *callback) /*[clinic end generated code: output=d68fa4ad9ea40519 input=4808adf22fd137e7]*/ { return PyWeakref_NewProxy(object, callback); } static PyMethodDef weakref_functions[] = { _WEAKREF_GETWEAKREFCOUNT_METHODDEF _WEAKREF__REMOVE_DEAD_WEAKREF_METHODDEF _WEAKREF_GETWEAKREFS_METHODDEF _WEAKREF_PROXY_METHODDEF {NULL, NULL, 0, NULL} }; static int weakref_exec(PyObject *module) { if (PyModule_AddObjectRef(module, "ref", (PyObject *) &_PyWeakref_RefType) < 0) { return -1; } if (PyModule_AddObjectRef(module, "ReferenceType", (PyObject *) &_PyWeakref_RefType) < 0) { return -1; } if (PyModule_AddObjectRef(module, "ProxyType", (PyObject *) &_PyWeakref_ProxyType) < 0) { return -1; } if (PyModule_AddObjectRef(module, "CallableProxyType", (PyObject *) &_PyWeakref_CallableProxyType) < 0) { return -1; } return 0; } static struct PyModuleDef_Slot weakref_slots[] = { {Py_mod_exec, weakref_exec}, {Py_mod_multiple_interpreters, Py_MOD_PER_INTERPRETER_GIL_SUPPORTED}, {0, NULL} }; static struct PyModuleDef weakrefmodule = { PyModuleDef_HEAD_INIT, "_weakref", "Weak-reference support module.", 0, weakref_functions, weakref_slots, NULL, NULL, NULL }; PyMODINIT_FUNC PyInit__weakref(void) { return PyModuleDef_Init(&weakrefmodule); }