aboutsummaryrefslogtreecommitdiff
path: root/Tools
diff options
context:
space:
mode:
authorPablo Galindo <Pablogsal@gmail.com>2020-05-25 18:38:45 +0100
committerGitHub <noreply@github.com>2020-05-25 18:38:45 +0100
commit800a35c623bbcdb5793c7d7a4974524286311479 (patch)
tree01980ede9eca05a61ae1636d1443b9a34106ba04 /Tools
parentbpo-39573: Fix buildbot failure for tupleobject.c (GH-20391) (diff)
downloadcpython-800a35c623bbcdb5793c7d7a4974524286311479.tar.gz
cpython-800a35c623bbcdb5793c7d7a4974524286311479.tar.bz2
cpython-800a35c623bbcdb5793c7d7a4974524286311479.zip
bpo-40750: Support -d flag in the new parser (GH-20340)
Diffstat (limited to 'Tools')
-rw-r--r--Tools/peg_generator/pegen/c_generator.py84
-rw-r--r--Tools/peg_generator/pegen/testutil.py2
2 files changed, 61 insertions, 25 deletions
diff --git a/Tools/peg_generator/pegen/c_generator.py b/Tools/peg_generator/pegen/c_generator.py
index 1249d4f683e..ff7e75a9fdf 100644
--- a/Tools/peg_generator/pegen/c_generator.py
+++ b/Tools/peg_generator/pegen/c_generator.py
@@ -29,6 +29,13 @@ from pegen.parser_generator import ParserGenerator
EXTENSION_PREFIX = """\
#include "pegen.h"
+#ifdef Py_DEBUG
+extern int Py_DebugFlag;
+#define D(x) if (Py_DebugFlag) x;
+#else
+#define D(x)
+#endif
+
"""
@@ -300,6 +307,16 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.debug = debug
self.skip_actions = skip_actions
+ def add_level(self) -> None:
+ self.print("D(p->level++);")
+
+ def remove_level(self) -> None:
+ self.print("D(p->level--);")
+
+ def add_return(self, ret_val: str) -> None:
+ self.remove_level()
+ self.print(f"return {ret_val};")
+
def unique_varname(self, name: str = "tmpvar") -> str:
new_var = name + "_" + str(self._varname_counter)
self._varname_counter += 1
@@ -310,8 +327,8 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print(f"int {error_var} = {call_text};")
self.print(f"if ({error_var}) {{")
with self.indent():
- self.print(f"return {returnval};")
- self.print(f"}}")
+ self.add_return(returnval)
+ self.print("}")
def call_with_errorcheck_goto(self, call_text: str, goto_target: str) -> None:
error_var = self.unique_varname()
@@ -328,7 +345,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print(cleanup_code)
self.print("p->error_indicator = 1;")
self.print("PyErr_NoMemory();")
- self.print("return NULL;")
+ self.add_return("NULL")
self.print(f"}}")
def out_of_memory_goto(self, expr: str, goto_target: str) -> None:
@@ -415,7 +432,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print("if (p->mark == p->fill && _PyPegen_fill_token(p) < 0) {")
with self.indent():
self.print("p->error_indicator = 1;")
- self.print("return NULL;")
+ self.add_return("NULL")
self.print("}")
self.print("int _start_lineno = p->tokens[_mark]->lineno;")
self.print("UNUSED(_start_lineno); // Only used by EXTRA macro")
@@ -426,7 +443,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print("Token *_token = _PyPegen_get_last_nonnwhitespace_token(p);")
self.print("if (_token == NULL) {")
with self.indent():
- self.print("return NULL;")
+ self.add_return("NULL")
self.print("}")
self.print("int _end_lineno = _token->end_lineno;")
self.print("UNUSED(_end_lineno); // Only used by EXTRA macro")
@@ -436,16 +453,18 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
def _check_for_errors(self) -> None:
self.print("if (p->error_indicator) {")
with self.indent():
- self.print("return NULL;")
+ self.add_return("NULL")
self.print("}")
def _set_up_rule_memoization(self, node: Rule, result_type: str) -> None:
self.print("{")
with self.indent():
+ self.add_level()
self.print(f"{result_type} _res = NULL;")
- self.print(f"if (_PyPegen_is_memoized(p, {node.name}_type, &_res))")
+ self.print(f"if (_PyPegen_is_memoized(p, {node.name}_type, &_res)) {{")
with self.indent():
- self.print("return _res;")
+ self.add_return("_res")
+ self.print("}")
self.print("int _mark = p->mark;")
self.print("int _resmark = p->mark;")
self.print("while (1) {")
@@ -462,7 +481,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print("_res = _raw;")
self.print("}")
self.print(f"p->mark = _resmark;")
- self.print("return _res;")
+ self.add_return("_res")
self.print("}")
self.print(f"static {result_type}")
self.print(f"{node.name}_raw(Parser *p)")
@@ -474,12 +493,14 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
memoize = self._should_memoize(node)
with self.indent():
+ self.add_level()
self._check_for_errors()
self.print(f"{result_type} _res = NULL;")
if memoize:
- self.print(f"if (_PyPegen_is_memoized(p, {node.name}_type, &_res))")
+ self.print(f"if (_PyPegen_is_memoized(p, {node.name}_type, &_res)) {{")
with self.indent():
- self.print("return _res;")
+ self.add_return("_res")
+ self.print("}")
self.print("int _mark = p->mark;")
if any(alt.action and "EXTRA" in alt.action for alt in rhs.alts):
self._set_up_token_start_metadata_extraction()
@@ -487,25 +508,27 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
rhs, is_loop=False, is_gather=node.is_gather(), rulename=node.name,
)
if self.debug:
- self.print(f'fprintf(stderr, "Fail at %d: {node.name}\\n", p->mark);')
+ self.print(f'D(fprintf(stderr, "Fail at %d: {node.name}\\n", p->mark));')
self.print("_res = NULL;")
self.print(" done:")
with self.indent():
if memoize:
self.print(f"_PyPegen_insert_memo(p, _mark, {node.name}_type, _res);")
- self.print("return _res;")
+ self.add_return("_res")
def _handle_loop_rule_body(self, node: Rule, rhs: Rhs) -> None:
memoize = self._should_memoize(node)
is_repeat1 = node.name.startswith("_loop1")
with self.indent():
+ self.add_level()
self._check_for_errors()
self.print("void *_res = NULL;")
if memoize:
- self.print(f"if (_PyPegen_is_memoized(p, {node.name}_type, &_res))")
+ self.print(f"if (_PyPegen_is_memoized(p, {node.name}_type, &_res)) {{")
with self.indent():
- self.print("return _res;")
+ self.add_return("_res")
+ self.print("}")
self.print("int _mark = p->mark;")
self.print("int _start_mark = p->mark;")
self.print("void **_children = PyMem_Malloc(sizeof(void *));")
@@ -521,7 +544,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print("if (_n == 0 || p->error_indicator) {")
with self.indent():
self.print("PyMem_Free(_children);")
- self.print("return NULL;")
+ self.add_return("NULL")
self.print("}")
self.print("asdl_seq *_seq = _Py_asdl_seq_new(_n, p->arena);")
self.out_of_memory_return(f"!_seq", cleanup_code="PyMem_Free(_children);")
@@ -529,7 +552,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print("PyMem_Free(_children);")
if node.name:
self.print(f"_PyPegen_insert_memo(p, _start_mark, {node.name}_type, _seq);")
- self.print("return _seq;")
+ self.add_return("_seq")
def visit_Rule(self, node: Rule) -> None:
is_loop = node.is_loop()
@@ -594,12 +617,12 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print("p->error_indicator = 1;")
if cleanup_code:
self.print(cleanup_code)
- self.print("return NULL;")
+ self.add_return("NULL")
self.print("}")
if self.debug:
self.print(
- f'fprintf(stderr, "Hit with action [%d-%d]: %s\\n", _mark, p->mark, "{node}");'
+ f'D(fprintf(stderr, "Hit with action [%d-%d]: %s\\n", _mark, p->mark, "{node}"));'
)
def emit_default_action(self, is_gather: bool, node: Alt) -> None:
@@ -613,7 +636,7 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
else:
if self.debug:
self.print(
- f'fprintf(stderr, "Hit without action [%d:%d]: %s\\n", _mark, p->mark, "{node}");'
+ f'D(fprintf(stderr, "Hit without action [%d:%d]: %s\\n", _mark, p->mark, "{node}"));'
)
self.print(
f"_res = _PyPegen_dummy_name(p, {', '.join(self.local_variable_names)});"
@@ -621,18 +644,21 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
else:
if self.debug:
self.print(
- f'fprintf(stderr, "Hit with default action [%d:%d]: %s\\n", _mark, p->mark, "{node}");'
+ f'D(fprintf(stderr, "Hit with default action [%d:%d]: %s\\n", _mark, p->mark, "{node}"));'
)
self.print(f"_res = {self.local_variable_names[0]};")
def emit_dummy_action(self) -> None:
self.print("_res = _PyPegen_dummy_name(p);")
- def handle_alt_normal(self, node: Alt, is_gather: bool) -> None:
+ def handle_alt_normal(self, node: Alt, is_gather: bool, rulename: Optional[str]) -> None:
self.join_conditions(keyword="if", node=node)
self.print("{")
# We have parsed successfully all the conditions for the option.
with self.indent():
+ self.print(
+ f'D(fprintf(stderr, "%*c+ {rulename}[%d-%d]: %s succeeded!\\n", p->level, \' \', _mark, p->mark, "{node}"));'
+ )
# Prepare to emmit the rule action and do so
if node.action and "EXTRA" in node.action:
self._set_up_token_end_metadata_extraction()
@@ -684,6 +710,9 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
self.print(f"{{ // {node}")
with self.indent():
self._check_for_errors()
+ self.print(
+ f'D(fprintf(stderr, "%*c> {rulename}[%d-%d]: %s\\n", p->level, \' \', _mark, p->mark, "{node}"));'
+ )
# Prepare variable declarations for the alternative
vars = self.collect_vars(node)
for v, var_type in sorted(item for item in vars.items() if item[0] is not None):
@@ -701,11 +730,18 @@ class CParserGenerator(ParserGenerator, GrammarVisitor):
if is_loop:
self.handle_alt_loop(node, is_gather, rulename)
else:
- self.handle_alt_normal(node, is_gather)
+ self.handle_alt_normal(node, is_gather, rulename)
self.print("p->mark = _mark;")
+ self.print(
+ f"D(fprintf(stderr, \"%*c%s {rulename}[%d-%d]: %s failed!\\n\", p->level, ' ',\n"
+ f' p->error_indicator ? "ERROR!" : "-", _mark, p->mark, "{node}"));'
+ )
if "_cut_var" in vars:
- self.print("if (_cut_var) return NULL;")
+ self.print("if (_cut_var) {")
+ with self.indent():
+ self.add_return("NULL")
+ self.print("}")
self.print("}")
def collect_vars(self, node: Alt) -> Dict[Optional[str], Optional[str]]:
diff --git a/Tools/peg_generator/pegen/testutil.py b/Tools/peg_generator/pegen/testutil.py
index 264659e7176..920d24657c5 100644
--- a/Tools/peg_generator/pegen/testutil.py
+++ b/Tools/peg_generator/pegen/testutil.py
@@ -96,7 +96,7 @@ def generate_parser_c_extension(
# context.
assert not os.listdir(path)
source = path / "parse.c"
- with open(source, "w") as file:
+ with open(source, "w", encoding="utf-8") as file:
genr = CParserGenerator(
grammar, ALL_TOKENS, EXACT_TOKENS, NON_EXACT_TOKENS, file, debug=debug
)