Commit 474eedfb authored by Eddie Elizondo's avatar Eddie Elizondo Committed by Petr Viktorin

bpo-34784: Fix PyStructSequence_NewType with heap-allocated StructSequence (GH-9665)

parent 1a6be91e
Fix the implementation of PyStructSequence_NewType in order to create heap
allocated StructSequences.
...@@ -3313,6 +3313,31 @@ test_decref_doesnt_leak(PyObject *ob, PyObject *Py_UNUSED(ignored)) ...@@ -3313,6 +3313,31 @@ test_decref_doesnt_leak(PyObject *ob, PyObject *Py_UNUSED(ignored))
Py_RETURN_NONE; Py_RETURN_NONE;
} }
static PyObject *
test_structseq_newtype_doesnt_leak(PyObject *Py_UNUSED(self),
PyObject *Py_UNUSED(args))
{
PyStructSequence_Desc descr;
PyStructSequence_Field descr_fields[3];
descr_fields[0] = (PyStructSequence_Field){"foo", "foo value"};
descr_fields[1] = (PyStructSequence_Field){NULL, "some hidden value"};
descr_fields[2] = (PyStructSequence_Field){0, NULL};
descr.name = "_testcapi.test_descr";
descr.doc = "This is used to test for memory leaks in NewType";
descr.fields = descr_fields;
descr.n_in_sequence = 1;
PyTypeObject* structseq_type = PyStructSequence_NewType(&descr);
assert(structseq_type != NULL);
assert(PyType_Check(structseq_type));
assert(PyType_FastSubclass(structseq_type, Py_TPFLAGS_TUPLE_SUBCLASS));
Py_DECREF(structseq_type);
Py_RETURN_NONE;
}
static PyObject * static PyObject *
test_incref_decref_API(PyObject *ob, PyObject *Py_UNUSED(ignored)) test_incref_decref_API(PyObject *ob, PyObject *Py_UNUSED(ignored))
{ {
...@@ -4721,6 +4746,8 @@ static PyMethodDef TestMethods[] = { ...@@ -4721,6 +4746,8 @@ static PyMethodDef TestMethods[] = {
{"test_incref_doesnt_leak", test_incref_doesnt_leak, METH_NOARGS}, {"test_incref_doesnt_leak", test_incref_doesnt_leak, METH_NOARGS},
{"test_xdecref_doesnt_leak",test_xdecref_doesnt_leak, METH_NOARGS}, {"test_xdecref_doesnt_leak",test_xdecref_doesnt_leak, METH_NOARGS},
{"test_decref_doesnt_leak", test_decref_doesnt_leak, METH_NOARGS}, {"test_decref_doesnt_leak", test_decref_doesnt_leak, METH_NOARGS},
{"test_structseq_newtype_doesnt_leak",
test_structseq_newtype_doesnt_leak, METH_NOARGS},
{"test_incref_decref_API", test_incref_decref_API, METH_NOARGS}, {"test_incref_decref_API", test_incref_decref_API, METH_NOARGS},
{"test_long_and_overflow", test_long_and_overflow, METH_NOARGS}, {"test_long_and_overflow", test_long_and_overflow, METH_NOARGS},
{"test_long_as_double", test_long_as_double, METH_NOARGS}, {"test_long_as_double", test_long_as_double, METH_NOARGS},
......
...@@ -1948,14 +1948,14 @@ static PyStructSequence_Desc waitid_result_desc = { ...@@ -1948,14 +1948,14 @@ static PyStructSequence_Desc waitid_result_desc = {
waitid_result_fields, waitid_result_fields,
5 5
}; };
static PyTypeObject WaitidResultType; static PyTypeObject* WaitidResultType;
#endif #endif
static int initialized; static int initialized;
static PyTypeObject StatResultType; static PyTypeObject* StatResultType;
static PyTypeObject StatVFSResultType; static PyTypeObject* StatVFSResultType;
#if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDPARAM) #if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDPARAM)
static PyTypeObject SchedParamType; static PyTypeObject* SchedParamType;
#endif #endif
static newfunc structseq_new; static newfunc structseq_new;
...@@ -2029,7 +2029,7 @@ static PyObject* ...@@ -2029,7 +2029,7 @@ static PyObject*
_pystat_fromstructstat(STRUCT_STAT *st) _pystat_fromstructstat(STRUCT_STAT *st)
{ {
unsigned long ansec, mnsec, cnsec; unsigned long ansec, mnsec, cnsec;
PyObject *v = PyStructSequence_New(&StatResultType); PyObject *v = PyStructSequence_New(StatResultType);
if (v == NULL) if (v == NULL)
return NULL; return NULL;
...@@ -4407,7 +4407,7 @@ static PyStructSequence_Desc uname_result_desc = { ...@@ -4407,7 +4407,7 @@ static PyStructSequence_Desc uname_result_desc = {
5 5
}; };
static PyTypeObject UnameResultType; static PyTypeObject* UnameResultType;
#ifdef HAVE_UNAME #ifdef HAVE_UNAME
...@@ -4435,7 +4435,7 @@ os_uname_impl(PyObject *module) ...@@ -4435,7 +4435,7 @@ os_uname_impl(PyObject *module)
if (res < 0) if (res < 0)
return posix_error(); return posix_error();
value = PyStructSequence_New(&UnameResultType); value = PyStructSequence_New(UnameResultType);
if (value == NULL) if (value == NULL)
return NULL; return NULL;
...@@ -5941,7 +5941,7 @@ os_sched_getscheduler_impl(PyObject *module, pid_t pid) ...@@ -5941,7 +5941,7 @@ os_sched_getscheduler_impl(PyObject *module, pid_t pid)
#if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDPARAM) #if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDPARAM)
/*[clinic input] /*[clinic input]
class os.sched_param "PyObject *" "&SchedParamType" class os.sched_param "PyObject *" "SchedParamType"
@classmethod @classmethod
os.sched_param.__new__ os.sched_param.__new__
...@@ -5954,7 +5954,7 @@ Current has only one field: sched_priority"); ...@@ -5954,7 +5954,7 @@ Current has only one field: sched_priority");
static PyObject * static PyObject *
os_sched_param_impl(PyTypeObject *type, PyObject *sched_priority) os_sched_param_impl(PyTypeObject *type, PyObject *sched_priority)
/*[clinic end generated code: output=48f4067d60f48c13 input=73a4c22f7071fc62]*/ /*[clinic end generated code: output=48f4067d60f48c13 input=ab4de35a9a7811f2]*/
{ {
PyObject *res; PyObject *res;
...@@ -5986,7 +5986,7 @@ convert_sched_param(PyObject *param, struct sched_param *res) ...@@ -5986,7 +5986,7 @@ convert_sched_param(PyObject *param, struct sched_param *res)
{ {
long priority; long priority;
if (Py_TYPE(param) != &SchedParamType) { if (Py_TYPE(param) != SchedParamType) {
PyErr_SetString(PyExc_TypeError, "must have a sched_param object"); PyErr_SetString(PyExc_TypeError, "must have a sched_param object");
return 0; return 0;
} }
...@@ -6057,7 +6057,7 @@ os_sched_getparam_impl(PyObject *module, pid_t pid) ...@@ -6057,7 +6057,7 @@ os_sched_getparam_impl(PyObject *module, pid_t pid)
if (sched_getparam(pid, &param)) if (sched_getparam(pid, &param))
return posix_error(); return posix_error();
result = PyStructSequence_New(&SchedParamType); result = PyStructSequence_New(SchedParamType);
if (!result) if (!result)
return NULL; return NULL;
priority = PyLong_FromLong(param.sched_priority); priority = PyLong_FromLong(param.sched_priority);
...@@ -7422,7 +7422,7 @@ os_waitid_impl(PyObject *module, idtype_t idtype, id_t id, int options) ...@@ -7422,7 +7422,7 @@ os_waitid_impl(PyObject *module, idtype_t idtype, id_t id, int options)
if (si.si_pid == 0) if (si.si_pid == 0)
Py_RETURN_NONE; Py_RETURN_NONE;
result = PyStructSequence_New(&WaitidResultType); result = PyStructSequence_New(WaitidResultType);
if (!result) if (!result)
return NULL; return NULL;
...@@ -7857,7 +7857,7 @@ static PyStructSequence_Desc times_result_desc = { ...@@ -7857,7 +7857,7 @@ static PyStructSequence_Desc times_result_desc = {
5 5
}; };
static PyTypeObject TimesResultType; static PyTypeObject* TimesResultType;
#ifdef MS_WINDOWS #ifdef MS_WINDOWS
#define HAVE_TIMES /* mandatory, for the method table */ #define HAVE_TIMES /* mandatory, for the method table */
...@@ -7870,7 +7870,7 @@ build_times_result(double user, double system, ...@@ -7870,7 +7870,7 @@ build_times_result(double user, double system,
double children_user, double children_system, double children_user, double children_system,
double elapsed) double elapsed)
{ {
PyObject *value = PyStructSequence_New(&TimesResultType); PyObject *value = PyStructSequence_New(TimesResultType);
if (value == NULL) if (value == NULL)
return NULL; return NULL;
...@@ -9950,7 +9950,7 @@ os_WSTOPSIG_impl(PyObject *module, int status) ...@@ -9950,7 +9950,7 @@ os_WSTOPSIG_impl(PyObject *module, int status)
static PyObject* static PyObject*
_pystatvfs_fromstructstatvfs(struct statvfs st) { _pystatvfs_fromstructstatvfs(struct statvfs st) {
PyObject *v = PyStructSequence_New(&StatVFSResultType); PyObject *v = PyStructSequence_New(StatVFSResultType);
if (v == NULL) if (v == NULL)
return NULL; return NULL;
...@@ -11703,7 +11703,7 @@ os_urandom_impl(PyObject *module, Py_ssize_t size) ...@@ -11703,7 +11703,7 @@ os_urandom_impl(PyObject *module, Py_ssize_t size)
/* Terminal size querying */ /* Terminal size querying */
static PyTypeObject TerminalSizeType; static PyTypeObject* TerminalSizeType;
PyDoc_STRVAR(TerminalSize_docstring, PyDoc_STRVAR(TerminalSize_docstring,
"A tuple of (columns, lines) for holding terminal window size"); "A tuple of (columns, lines) for holding terminal window size");
...@@ -11795,7 +11795,7 @@ get_terminal_size(PyObject *self, PyObject *args) ...@@ -11795,7 +11795,7 @@ get_terminal_size(PyObject *self, PyObject *args)
} }
#endif /* TERMSIZE_USE_CONIO */ #endif /* TERMSIZE_USE_CONIO */
termsize = PyStructSequence_New(&TerminalSizeType); termsize = PyStructSequence_New(TerminalSizeType);
if (termsize == NULL) if (termsize == NULL)
return NULL; return NULL;
PyStructSequence_SET_ITEM(termsize, 0, PyLong_FromLong(columns)); PyStructSequence_SET_ITEM(termsize, 0, PyLong_FromLong(columns));
...@@ -13912,23 +13912,28 @@ INITFUNC(void) ...@@ -13912,23 +13912,28 @@ INITFUNC(void)
if (!initialized) { if (!initialized) {
#if defined(HAVE_WAITID) && !defined(__APPLE__) #if defined(HAVE_WAITID) && !defined(__APPLE__)
waitid_result_desc.name = MODNAME ".waitid_result"; waitid_result_desc.name = MODNAME ".waitid_result";
if (PyStructSequence_InitType2(&WaitidResultType, &waitid_result_desc) < 0) WaitidResultType = PyStructSequence_NewType(&waitid_result_desc);
if (WaitidResultType == NULL) {
return NULL; return NULL;
}
#endif #endif
stat_result_desc.name = "os.stat_result"; /* see issue #19209 */ stat_result_desc.name = "os.stat_result"; /* see issue #19209 */
stat_result_desc.fields[7].name = PyStructSequence_UnnamedField; stat_result_desc.fields[7].name = PyStructSequence_UnnamedField;
stat_result_desc.fields[8].name = PyStructSequence_UnnamedField; stat_result_desc.fields[8].name = PyStructSequence_UnnamedField;
stat_result_desc.fields[9].name = PyStructSequence_UnnamedField; stat_result_desc.fields[9].name = PyStructSequence_UnnamedField;
if (PyStructSequence_InitType2(&StatResultType, &stat_result_desc) < 0) StatResultType = PyStructSequence_NewType(&stat_result_desc);
if (StatResultType == NULL) {
return NULL; return NULL;
structseq_new = StatResultType.tp_new; }
StatResultType.tp_new = statresult_new; structseq_new = StatResultType->tp_new;
StatResultType->tp_new = statresult_new;
statvfs_result_desc.name = "os.statvfs_result"; /* see issue #19209 */ statvfs_result_desc.name = "os.statvfs_result"; /* see issue #19209 */
if (PyStructSequence_InitType2(&StatVFSResultType, StatVFSResultType = PyStructSequence_NewType(&statvfs_result_desc);
&statvfs_result_desc) < 0) if (StatVFSResultType == NULL) {
return NULL; return NULL;
}
#ifdef NEED_TICKS_PER_SECOND #ifdef NEED_TICKS_PER_SECOND
# if defined(HAVE_SYSCONF) && defined(_SC_CLK_TCK) # if defined(HAVE_SYSCONF) && defined(_SC_CLK_TCK)
ticks_per_second = sysconf(_SC_CLK_TCK); ticks_per_second = sysconf(_SC_CLK_TCK);
...@@ -13941,15 +13946,18 @@ INITFUNC(void) ...@@ -13941,15 +13946,18 @@ INITFUNC(void)
#if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDPARAM) #if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDULER) || defined(POSIX_SPAWN_SETSCHEDPARAM)
sched_param_desc.name = MODNAME ".sched_param"; sched_param_desc.name = MODNAME ".sched_param";
if (PyStructSequence_InitType2(&SchedParamType, &sched_param_desc) < 0) SchedParamType = PyStructSequence_NewType(&sched_param_desc);
if (SchedParamType == NULL) {
return NULL; return NULL;
SchedParamType.tp_new = os_sched_param; }
SchedParamType->tp_new = os_sched_param;
#endif #endif
/* initialize TerminalSize_info */ /* initialize TerminalSize_info */
if (PyStructSequence_InitType2(&TerminalSizeType, TerminalSizeType = PyStructSequence_NewType(&TerminalSize_desc);
&TerminalSize_desc) < 0) if (TerminalSizeType == NULL) {
return NULL; return NULL;
}
/* initialize scandir types */ /* initialize scandir types */
if (PyType_Ready(&ScandirIteratorType) < 0) if (PyType_Ready(&ScandirIteratorType) < 0)
...@@ -13958,29 +13966,33 @@ INITFUNC(void) ...@@ -13958,29 +13966,33 @@ INITFUNC(void)
return NULL; return NULL;
} }
#if defined(HAVE_WAITID) && !defined(__APPLE__) #if defined(HAVE_WAITID) && !defined(__APPLE__)
Py_INCREF((PyObject*) &WaitidResultType); Py_INCREF((PyObject*) WaitidResultType);
PyModule_AddObject(m, "waitid_result", (PyObject*) &WaitidResultType); PyModule_AddObject(m, "waitid_result", (PyObject*) WaitidResultType);
#endif #endif
Py_INCREF((PyObject*) &StatResultType); Py_INCREF((PyObject*) StatResultType);
PyModule_AddObject(m, "stat_result", (PyObject*) &StatResultType); PyModule_AddObject(m, "stat_result", (PyObject*) StatResultType);
Py_INCREF((PyObject*) &StatVFSResultType); Py_INCREF((PyObject*) StatVFSResultType);
PyModule_AddObject(m, "statvfs_result", PyModule_AddObject(m, "statvfs_result",
(PyObject*) &StatVFSResultType); (PyObject*) StatVFSResultType);
#if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER) #if defined(HAVE_SCHED_SETPARAM) || defined(HAVE_SCHED_SETSCHEDULER)
Py_INCREF(&SchedParamType); Py_INCREF(SchedParamType);
PyModule_AddObject(m, "sched_param", (PyObject *)&SchedParamType); PyModule_AddObject(m, "sched_param", (PyObject *)SchedParamType);
#endif #endif
times_result_desc.name = MODNAME ".times_result"; times_result_desc.name = MODNAME ".times_result";
if (PyStructSequence_InitType2(&TimesResultType, &times_result_desc) < 0) TimesResultType = PyStructSequence_NewType(&times_result_desc);
if (TimesResultType == NULL) {
return NULL; return NULL;
PyModule_AddObject(m, "times_result", (PyObject *)&TimesResultType); }
PyModule_AddObject(m, "times_result", (PyObject *)TimesResultType);
uname_result_desc.name = MODNAME ".uname_result"; uname_result_desc.name = MODNAME ".uname_result";
if (PyStructSequence_InitType2(&UnameResultType, &uname_result_desc) < 0) UnameResultType = PyStructSequence_NewType(&uname_result_desc);
if (UnameResultType == NULL) {
return NULL; return NULL;
PyModule_AddObject(m, "uname_result", (PyObject *)&UnameResultType); }
PyModule_AddObject(m, "uname_result", (PyObject *)UnameResultType);
#ifdef __APPLE__ #ifdef __APPLE__
/* /*
...@@ -14020,8 +14032,8 @@ INITFUNC(void) ...@@ -14020,8 +14032,8 @@ INITFUNC(void)
#endif /* __APPLE__ */ #endif /* __APPLE__ */
Py_INCREF(&TerminalSizeType); Py_INCREF(TerminalSizeType);
PyModule_AddObject(m, "terminal_size", (PyObject*) &TerminalSizeType); PyModule_AddObject(m, "terminal_size", (PyObject*)TerminalSizeType);
billion = PyLong_FromLong(1000000000); billion = PyLong_FromLong(1000000000);
if (!billion) if (!billion)
......
This diff is collapsed.
...@@ -2846,15 +2846,28 @@ static const short slotoffsets[] = { ...@@ -2846,15 +2846,28 @@ static const short slotoffsets[] = {
PyObject * PyObject *
PyType_FromSpecWithBases(PyType_Spec *spec, PyObject *bases) PyType_FromSpecWithBases(PyType_Spec *spec, PyObject *bases)
{ {
PyHeapTypeObject *res = (PyHeapTypeObject*)PyType_GenericAlloc(&PyType_Type, 0); PyHeapTypeObject *res;
PyTypeObject *type, *base; PyMemberDef *memb;
PyObject *modname; PyObject *modname;
char *s; PyTypeObject *type, *base;
char *res_start = (char*)res;
PyType_Slot *slot; PyType_Slot *slot;
Py_ssize_t nmembers;
char *s, *res_start;
nmembers = 0;
for (slot = spec->slots; slot->slot; slot++) {
if (slot->slot == Py_tp_members) {
for (memb = slot->pfunc; memb->name != NULL; memb++) {
nmembers++;
}
}
}
res = (PyHeapTypeObject*)PyType_GenericAlloc(&PyType_Type, nmembers);
if (res == NULL) if (res == NULL)
return NULL; return NULL;
res_start = (char*)res;
if (spec->name == NULL) { if (spec->name == NULL) {
PyErr_SetString(PyExc_SystemError, PyErr_SetString(PyExc_SystemError,
...@@ -2950,6 +2963,13 @@ PyType_FromSpecWithBases(PyType_Spec *spec, PyObject *bases) ...@@ -2950,6 +2963,13 @@ PyType_FromSpecWithBases(PyType_Spec *spec, PyObject *bases)
memcpy(tp_doc, old_doc, len); memcpy(tp_doc, old_doc, len);
type->tp_doc = tp_doc; type->tp_doc = tp_doc;
} }
/* Move the slots to the heap type itself */
if (slot->slot == Py_tp_members) {
size_t len = Py_TYPE(type)->tp_itemsize * nmembers;
memcpy(PyHeapType_GET_MEMBERS(res), slot->pfunc, len);
type->tp_members = PyHeapType_GET_MEMBERS(res);
}
} }
if (type->tp_dealloc == NULL) { if (type->tp_dealloc == NULL) {
/* It's a heap type, so needs the heap types' dealloc. /* It's a heap type, so needs the heap types' dealloc.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment