mirror of
https://github.com/python/cpython.git
synced 2025-09-26 18:29:57 +00:00
GH-92678: Expose managed dict clear and visit functions (#95246)
This commit is contained in:
parent
2d26449b06
commit
27055d766a
5 changed files with 57 additions and 0 deletions
|
@ -83,3 +83,6 @@ typedef struct {
|
||||||
|
|
||||||
PyAPI_FUNC(PyObject *) _PyDictView_New(PyObject *, PyTypeObject *);
|
PyAPI_FUNC(PyObject *) _PyDictView_New(PyObject *, PyTypeObject *);
|
||||||
PyAPI_FUNC(PyObject *) _PyDictView_Intersect(PyObject* self, PyObject *other);
|
PyAPI_FUNC(PyObject *) _PyDictView_Intersect(PyObject* self, PyObject *other);
|
||||||
|
|
||||||
|
PyAPI_FUNC(int) _PyObject_VisitManagedDict(PyObject *self, visitproc visit, void *arg);
|
||||||
|
PyAPI_FUNC(void) _PyObject_ClearManagedDict(PyObject *self);
|
||||||
|
|
|
@ -722,6 +722,20 @@ class CAPITest(unittest.TestCase):
|
||||||
with self.subTest(name=name):
|
with self.subTest(name=name):
|
||||||
self.assertTrue(hasattr(ctypes.pythonapi, name))
|
self.assertTrue(hasattr(ctypes.pythonapi, name))
|
||||||
|
|
||||||
|
def test_clear_managed_dict(self):
|
||||||
|
|
||||||
|
class C:
|
||||||
|
def __init__(self):
|
||||||
|
self.a = 1
|
||||||
|
|
||||||
|
c = C()
|
||||||
|
_testcapi.clear_managed_dict(c)
|
||||||
|
self.assertEqual(c.__dict__, {})
|
||||||
|
c = C()
|
||||||
|
self.assertEqual(c.__dict__, {'a':1})
|
||||||
|
_testcapi.clear_managed_dict(c)
|
||||||
|
self.assertEqual(c.__dict__, {})
|
||||||
|
|
||||||
|
|
||||||
class TestPendingCalls(unittest.TestCase):
|
class TestPendingCalls(unittest.TestCase):
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,3 @@
|
||||||
|
Adds unstable C-API functions ``_PyObject_VisitManagedDict`` and
|
||||||
|
``_PyObject_ClearManagedDict`` to allow C extensions to allow the VM to
|
||||||
|
manage their object's dictionaries.
|
|
@ -6009,6 +6009,13 @@ settrace_to_record(PyObject *self, PyObject *list)
|
||||||
Py_RETURN_NONE;
|
Py_RETURN_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static PyObject *
|
||||||
|
clear_managed_dict(PyObject *self, PyObject *obj)
|
||||||
|
{
|
||||||
|
_PyObject_ClearManagedDict(obj);
|
||||||
|
Py_RETURN_NONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
test_macros(PyObject *self, PyObject *Py_UNUSED(args))
|
test_macros(PyObject *self, PyObject *Py_UNUSED(args))
|
||||||
|
@ -6347,6 +6354,7 @@ static PyMethodDef TestMethods[] = {
|
||||||
{"test_code_api", test_code_api, METH_NOARGS, NULL},
|
{"test_code_api", test_code_api, METH_NOARGS, NULL},
|
||||||
{"settrace_to_record", settrace_to_record, METH_O, NULL},
|
{"settrace_to_record", settrace_to_record, METH_O, NULL},
|
||||||
{"test_macros", test_macros, METH_NOARGS, NULL},
|
{"test_macros", test_macros, METH_NOARGS, NULL},
|
||||||
|
{"clear_managed_dict", clear_managed_dict, METH_O, NULL},
|
||||||
{NULL, NULL} /* sentinel */
|
{NULL, NULL} /* sentinel */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -5583,6 +5583,35 @@ _PyObject_FreeInstanceAttributes(PyObject *self)
|
||||||
free_values(*values_ptr);
|
free_values(*values_ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
_PyObject_VisitManagedDict(PyObject *self, visitproc visit, void *arg)
|
||||||
|
{
|
||||||
|
PyTypeObject *tp = Py_TYPE(self);
|
||||||
|
if((tp->tp_flags & Py_TPFLAGS_MANAGED_DICT) == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
assert(tp->tp_dictoffset);
|
||||||
|
int err = _PyObject_VisitInstanceAttributes(self, visit, arg);
|
||||||
|
if (err) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
Py_VISIT(*_PyObject_ManagedDictPointer(self));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
_PyObject_ClearManagedDict(PyObject *self)
|
||||||
|
{
|
||||||
|
PyTypeObject *tp = Py_TYPE(self);
|
||||||
|
if((tp->tp_flags & Py_TPFLAGS_MANAGED_DICT) == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_PyObject_FreeInstanceAttributes(self);
|
||||||
|
*_PyObject_ValuesPointer(self) = NULL;
|
||||||
|
Py_CLEAR(*_PyObject_ManagedDictPointer(self));
|
||||||
|
}
|
||||||
|
|
||||||
PyObject *
|
PyObject *
|
||||||
PyObject_GenericGetDict(PyObject *obj, void *context)
|
PyObject_GenericGetDict(PyObject *obj, void *context)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue