summaryrefslogtreecommitdiffstats
path: root/Python/stackrefs.c
diff options
context:
space:
mode:
authorMark Shannon <mark@hotpy.org>2024-12-20 16:52:20 (GMT)
committerGitHub <noreply@github.com>2024-12-20 16:52:20 (GMT)
commit128cc47fbd44e3e09c50d9674fe4a4bba3be450c (patch)
tree12147152fb9afa9a4dffe14e9f4350dd39f1956c /Python/stackrefs.c
parent78ffba4221dcb2e39fd5db80c297d1777588bb59 (diff)
downloadcpython-128cc47fbd44e3e09c50d9674fe4a4bba3be450c.zip
cpython-128cc47fbd44e3e09c50d9674fe4a4bba3be450c.tar.gz
cpython-128cc47fbd44e3e09c50d9674fe4a4bba3be450c.tar.bz2
GH-127705: Add debug mode for `_PyStackRef`s inspired by HPy debug mode (GH-128121)
Diffstat (limited to 'Python/stackrefs.c')
-rw-r--r--Python/stackrefs.c156
1 files changed, 156 insertions, 0 deletions
diff --git a/Python/stackrefs.c b/Python/stackrefs.c
new file mode 100644
index 0000000..9bb4689
--- /dev/null
+++ b/Python/stackrefs.c
@@ -0,0 +1,156 @@
+
+#include "Python.h"
+
+#include "pycore_stackref.h"
+
+#if !defined(Py_GIL_DISABLED) && defined(Py_STACKREF_DEBUG)
+
+#if SIZEOF_VOID_P < 8
+#error "Py_STACKREF_DEBUG requires 64 bit machine"
+#endif
+
+#include "pycore_interp.h"
+#include "pycore_hashtable.h"
+
+typedef struct _table_entry {
+ PyObject *obj;
+ const char *classname;
+ const char *filename;
+ int linenumber;
+ const char *filename_borrow;
+ int linenumber_borrow;
+} TableEntry;
+
+TableEntry *
+make_table_entry(PyObject *obj, const char *filename, int linenumber)
+{
+ TableEntry *result = malloc(sizeof(TableEntry));
+ if (result == NULL) {
+ return NULL;
+ }
+ result->obj = obj;
+ result->classname = Py_TYPE(obj)->tp_name;
+ result->filename = filename;
+ result->linenumber = linenumber;
+ result->filename_borrow = NULL;
+ return result;
+}
+
+PyObject *
+_Py_stackref_get_object(_PyStackRef ref)
+{
+ if (ref.index == 0) {
+ return NULL;
+ }
+ PyInterpreterState *interp = PyInterpreterState_Get();
+ assert(interp != NULL);
+ if (ref.index >= interp->next_stackref) {
+ _Py_FatalErrorFormat(__func__, "Garbled stack ref with ID %" PRIu64 "\n", ref.index);
+ }
+ TableEntry *entry = _Py_hashtable_get(interp->stackref_debug_table, (void *)ref.index);
+ if (entry == NULL) {
+ _Py_FatalErrorFormat(__func__, "Accessing closed stack ref with ID %" PRIu64 "\n", ref.index);
+ }
+ return entry->obj;
+}
+
+PyObject *
+_Py_stackref_close(_PyStackRef ref)
+{
+ PyInterpreterState *interp = PyInterpreterState_Get();
+ if (ref.index >= interp->next_stackref) {
+ _Py_FatalErrorFormat(__func__, "Garbled stack ref with ID %" PRIu64 "\n", ref.index);
+ }
+ PyObject *obj;
+ if (ref.index <= LAST_PREDEFINED_STACKREF_INDEX) {
+ // Pre-allocated reference to None, False or True -- Do not clear
+ TableEntry *entry = _Py_hashtable_get(interp->stackref_debug_table, (void *)ref.index);
+ obj = entry->obj;
+ }
+ else {
+ TableEntry *entry = _Py_hashtable_steal(interp->stackref_debug_table, (void *)ref.index);
+ if (entry == NULL) {
+ _Py_FatalErrorFormat(__func__, "Invalid StackRef with ID %" PRIu64 "\n", (void *)ref.index);
+ }
+ obj = entry->obj;
+ free(entry);
+ }
+ return obj;
+}
+
+_PyStackRef
+_Py_stackref_create(PyObject *obj, const char *filename, int linenumber)
+{
+ if (obj == NULL) {
+ Py_FatalError("Cannot create a stackref for NULL");
+ }
+ PyInterpreterState *interp = PyInterpreterState_Get();
+ uint64_t new_id = interp->next_stackref++;
+ TableEntry *entry = make_table_entry(obj, filename, linenumber);
+ if (entry == NULL) {
+ Py_FatalError("No memory left for stackref debug table");
+ }
+ if (_Py_hashtable_set(interp->stackref_debug_table, (void *)new_id, entry) < 0) {
+ Py_FatalError("No memory left for stackref debug table");
+ }
+ return (_PyStackRef){ .index = new_id };
+}
+
+void
+_Py_stackref_record_borrow(_PyStackRef ref, const char *filename, int linenumber)
+{
+ if (ref.index <= LAST_PREDEFINED_STACKREF_INDEX) {
+ return;
+ }
+ PyInterpreterState *interp = PyInterpreterState_Get();
+ TableEntry *entry = _Py_hashtable_get(interp->stackref_debug_table, (void *)ref.index);
+ if (entry == NULL) {
+ _Py_FatalErrorFormat(__func__, "Invalid StackRef with ID %" PRIu64 "\n", (void *)ref.index);
+ }
+ entry->filename_borrow = filename;
+ entry->linenumber_borrow = linenumber;
+}
+
+
+void
+_Py_stackref_associate(PyInterpreterState *interp, PyObject *obj, _PyStackRef ref)
+{
+ assert(interp->next_stackref >= ref.index);
+ interp->next_stackref = ref.index+1;
+ TableEntry *entry = make_table_entry(obj, "builtin-object", 0);
+ if (entry == NULL) {
+ Py_FatalError("No memory left for stackref debug table");
+ }
+ if (_Py_hashtable_set(interp->stackref_debug_table, (void *)ref.index, (void *)entry) < 0) {
+ Py_FatalError("No memory left for stackref debug table");
+ }
+}
+
+
+static int
+report_leak(_Py_hashtable_t *ht, const void *key, const void *value, void *leak)
+{
+ TableEntry *entry = (TableEntry *)value;
+ if (!_Py_IsStaticImmortal(entry->obj)) {
+ *(int *)leak = 1;
+ printf("Stackref leak. Refers to instance of %s at %p. Created at %s:%d",
+ entry->classname, entry->obj, entry->filename, entry->linenumber);
+ if (entry->filename_borrow != NULL) {
+ printf(". Last borrow at %s:%d",entry->filename_borrow, entry->linenumber_borrow);
+ }
+ printf("\n");
+ }
+ return 0;
+}
+
+void
+_Py_stackref_report_leaks(PyInterpreterState *interp)
+{
+ int leak = 0;
+ _Py_hashtable_foreach(interp->stackref_debug_table, report_leak, &leak);
+ if (leak) {
+ Py_FatalError("Stackrefs leaked.");
+ }
+}
+
+#endif