summaryrefslogtreecommitdiffstats
path: root/Python/symtable.c
diff options
context:
space:
mode:
authorCarl Meyer <carl@oddbird.net>2023-09-20 16:55:56 (GMT)
committerGitHub <noreply@github.com>2023-09-20 16:55:56 (GMT)
commit32ffe58c1298b0082ff6fe96ad45c4efe49f4338 (patch)
tree19686c5a8fdc82495ef401caf3efbf467c1e9249 /Python/symtable.c
parentd41d2e69f621ce25e7719f5057a6be6776bc6783 (diff)
downloadcpython-32ffe58c1298b0082ff6fe96ad45c4efe49f4338.zip
cpython-32ffe58c1298b0082ff6fe96ad45c4efe49f4338.tar.gz
cpython-32ffe58c1298b0082ff6fe96ad45c4efe49f4338.tar.bz2
gh-109390: add dump_symtable utility under #if 0 (#109391)
Co-authored-by: Jelle Zijlstra <jelle.zijlstra@gmail.com>
Diffstat (limited to 'Python/symtable.c')
-rw-r--r--Python/symtable.c111
1 files changed, 110 insertions, 1 deletions
diff --git a/Python/symtable.c b/Python/symtable.c
index b0c7240..75ea9e9 100644
--- a/Python/symtable.c
+++ b/Python/symtable.c
@@ -4,6 +4,8 @@
#include "pycore_pystate.h" // _PyThreadState_GET()
#include "pycore_symtable.h" // PySTEntryObject
+// Set this to 1 to dump all symtables to stdout for debugging
+#define _PY_DUMP_SYMTABLE 0
/* error strings used for warnings */
#define GLOBAL_PARAM \
@@ -251,6 +253,109 @@ static int symtable_visit_pattern(struct symtable *st, pattern_ty s);
static int symtable_raise_if_annotation_block(struct symtable *st, const char *, expr_ty);
static int symtable_raise_if_comprehension_block(struct symtable *st, expr_ty);
+/* For debugging purposes only */
+#if _PY_DUMP_SYMTABLE
+static void _dump_symtable(PySTEntryObject* ste, PyObject* prefix)
+{
+ const char *blocktype = "";
+ switch (ste->ste_type) {
+ case FunctionBlock: blocktype = "FunctionBlock"; break;
+ case ClassBlock: blocktype = "ClassBlock"; break;
+ case ModuleBlock: blocktype = "ModuleBlock"; break;
+ case AnnotationBlock: blocktype = "AnnotationBlock"; break;
+ case TypeVarBoundBlock: blocktype = "TypeVarBoundBlock"; break;
+ case TypeAliasBlock: blocktype = "TypeAliasBlock"; break;
+ case TypeParamBlock: blocktype = "TypeParamBlock"; break;
+ }
+ const char *comptype = "";
+ switch (ste->ste_comprehension) {
+ case ListComprehension: comptype = " ListComprehension"; break;
+ case DictComprehension: comptype = " DictComprehension"; break;
+ case SetComprehension: comptype = " SetComprehension"; break;
+ case GeneratorExpression: comptype = " GeneratorExpression"; break;
+ case NoComprehension: break;
+ }
+ PyObject* msg = PyUnicode_FromFormat(
+ (
+ "%U=== Symtable for %U ===\n"
+ "%U%s%s\n"
+ "%U%s%s%s%s%s%s%s%s%s%s%s%s%s\n"
+ "%Ulineno: %d col_offset: %d\n"
+ "%U--- Symbols ---\n"
+ ),
+ prefix,
+ ste->ste_name,
+ prefix,
+ blocktype,
+ comptype,
+ prefix,
+ ste->ste_nested ? " nested" : "",
+ ste->ste_free ? " free" : "",
+ ste->ste_child_free ? " child_free" : "",
+ ste->ste_generator ? " generator" : "",
+ ste->ste_coroutine ? " coroutine" : "",
+ ste->ste_varargs ? " varargs" : "",
+ ste->ste_varkeywords ? " varkeywords" : "",
+ ste->ste_returns_value ? " returns_value" : "",
+ ste->ste_needs_class_closure ? " needs_class_closure" : "",
+ ste->ste_needs_classdict ? " needs_classdict" : "",
+ ste->ste_comp_inlined ? " comp_inlined" : "",
+ ste->ste_comp_iter_target ? " comp_iter_target" : "",
+ ste->ste_can_see_class_scope ? " can_see_class_scope" : "",
+ prefix,
+ ste->ste_lineno,
+ ste->ste_col_offset,
+ prefix
+ );
+ assert(msg != NULL);
+ printf("%s", PyUnicode_AsUTF8(msg));
+ Py_DECREF(msg);
+ PyObject *name, *value;
+ Py_ssize_t pos = 0;
+ while (PyDict_Next(ste->ste_symbols, &pos, &name, &value)) {
+ int scope = _PyST_GetScope(ste, name);
+ long flags = _PyST_GetSymbol(ste, name);
+ printf("%s %s: ", PyUnicode_AsUTF8(prefix), PyUnicode_AsUTF8(name));
+ if (flags & DEF_GLOBAL) printf(" DEF_GLOBAL");
+ if (flags & DEF_LOCAL) printf(" DEF_LOCAL");
+ if (flags & DEF_PARAM) printf(" DEF_PARAM");
+ if (flags & DEF_NONLOCAL) printf(" DEF_NONLOCAL");
+ if (flags & USE) printf(" USE");
+ if (flags & DEF_FREE) printf(" DEF_FREE");
+ if (flags & DEF_FREE_CLASS) printf(" DEF_FREE_CLASS");
+ if (flags & DEF_IMPORT) printf(" DEF_IMPORT");
+ if (flags & DEF_ANNOT) printf(" DEF_ANNOT");
+ if (flags & DEF_COMP_ITER) printf(" DEF_COMP_ITER");
+ if (flags & DEF_TYPE_PARAM) printf(" DEF_TYPE_PARAM");
+ if (flags & DEF_COMP_CELL) printf(" DEF_COMP_CELL");
+ switch (scope) {
+ case LOCAL: printf(" LOCAL"); break;
+ case GLOBAL_EXPLICIT: printf(" GLOBAL_EXPLICIT"); break;
+ case GLOBAL_IMPLICIT: printf(" GLOBAL_IMPLICIT"); break;
+ case FREE: printf(" FREE"); break;
+ case CELL: printf(" CELL"); break;
+ }
+ printf("\n");
+ }
+ printf("%s--- Children ---\n", PyUnicode_AsUTF8(prefix));
+ PyObject *new_prefix = PyUnicode_FromFormat(" %U", prefix);
+ assert(new_prefix != NULL);
+ for (Py_ssize_t i = 0; i < PyList_GET_SIZE(ste->ste_children); i++) {
+ PyObject *child = PyList_GetItem(ste->ste_children, i);
+ assert(child != NULL && PySTEntry_Check(child));
+ _dump_symtable((PySTEntryObject *)child, new_prefix);
+ }
+ Py_DECREF(new_prefix);
+}
+
+static void dump_symtable(PySTEntryObject* ste)
+{
+ PyObject *empty = PyUnicode_FromString("");
+ assert(empty != NULL);
+ _dump_symtable(ste, empty);
+ Py_DECREF(empty);
+}
+#endif
#define DUPLICATE_ARGUMENT \
"duplicate argument '%U' in function definition"
@@ -360,8 +465,12 @@ _PySymtable_Build(mod_ty mod, PyObject *filename, PyFutureFeatures *future)
return NULL;
}
/* Make the second symbol analysis pass */
- if (symtable_analyze(st))
+ if (symtable_analyze(st)) {
+#if _PY_DUMP_SYMTABLE
+ dump_symtable(st->st_top);
+#endif
return st;
+ }
_PySymtable_Free(st);
return NULL;
error: