mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 18:54:53 +00:00 
			
		
		
		
	gh-109390: add dump_symtable utility under #if 0 (#109391)
Co-authored-by: Jelle Zijlstra <jelle.zijlstra@gmail.com>
This commit is contained in:
		
							parent
							
								
									d41d2e69f6
								
							
						
					
					
						commit
						32ffe58c12
					
				
					 1 changed files with 110 additions and 1 deletions
				
			
		|  | @ -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: | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Carl Meyer
						Carl Meyer