From 5b56c808301e5eaeccefa174fb2468eb41176d7b Mon Sep 17 00:00:00 2001 From: Arkadiusz Kozdra Date: Fri, 27 Sep 2024 20:38:49 +0200 Subject: [PATCH] Better error recovery (#5493) Signed-off-by: Arkadiusz Kozdra --- src/verilog.y | 36 ++++++++++++++++--- test_regress/t/t_source_sync.out | 7 ---- test_regress/t/t_source_sync.v | 12 ------- test_regress/t/t_source_sync_bad.out | 16 +++++++++ ...{t_source_sync.py => t_source_sync_bad.py} | 0 test_regress/t/t_source_sync_bad.v | 34 ++++++++++++++++++ 6 files changed, 82 insertions(+), 23 deletions(-) delete mode 100644 test_regress/t/t_source_sync.out delete mode 100644 test_regress/t/t_source_sync.v create mode 100644 test_regress/t/t_source_sync_bad.out rename test_regress/t/{t_source_sync.py => t_source_sync_bad.py} (100%) create mode 100644 test_regress/t/t_source_sync_bad.v diff --git a/src/verilog.y b/src/verilog.y index 6c6589f4f..cf1a36b18 100644 --- a/src/verilog.y +++ b/src/verilog.y @@ -2248,18 +2248,28 @@ struct_unionDecl: // IEEE: part of data_type // // packedSigningE is NOP for unpacked ySTRUCT packedSigningE '{' /*mid*/ { $$ = new AstStructDType{$1, $2}; SYMP->pushNew($$); } - /*cont*/ struct_union_memberList '}' + /*cont*/ struct_union_memberListEnd { $$ = $4; $$->addMembersp($5); SYMP->popScope($$); } | yUNION taggedSoftE packedSigningE '{' /*mid*/ { $$ = new AstUnionDType{$1, $3}; SYMP->pushNew($$); } - /*cont*/ struct_union_memberList '}' + /*cont*/ struct_union_memberListEnd { $$ = $5; $$->addMembersp($6); SYMP->popScope($$); } ; +struct_union_memberListEnd: // IEEE: { struct_union_member } '}' + struct_union_memberList '}' { $$ = $1; } + // + | struct_union_memberList error '}' { $$ = $1; } + | error '}' { $$ = nullptr; } + ; + struct_union_memberList: // IEEE: { struct_union_member } struct_union_member { $$ = $1; } | struct_union_memberList struct_union_member { $$ = addNextNull($1, $2); } + // + | struct_union_memberList error ';' { $$ = $1; } + | error ';' { $$ = nullptr; } ; struct_union_member: // ==IEEE: struct_union_member @@ -2634,6 +2644,8 @@ type_declaration: // ==IEEE: type_declaration | yTYPEDEF yUNION idAny ';' { $$ = GRAMMARP->createTypedefFwd($3, *$3); } | yTYPEDEF yCLASS idAny ';' { $$ = GRAMMARP->createTypedefFwd($3, *$3); } | yTYPEDEF yINTERFACE yCLASS idAny ';' { $$ = GRAMMARP->createTypedefFwd($4, *$4); } + // + | yTYPEDEF error idAny ';' { $$ = GRAMMARP->createTypedefFwd($3, *$3); } ; dtypeAttrListE: @@ -3559,6 +3571,9 @@ blockDeclStmtListE: // IEEE: [ { block_item_declaration } { statemen block_item_declarationList: // IEEE: [ block_item_declaration ] block_item_declaration { $$ = $1; } | block_item_declarationList block_item_declaration { $$ = addNextNull($1, $2); } + // + | block_item_declarationList error ';' { $$ = $1; } + | error ';' { $$ = nullptr; } ; block_item_declaration: // ==IEEE: block_item_declaration @@ -3570,6 +3585,8 @@ block_item_declaration: // ==IEEE: block_item_declaration stmtList: stmtBlock { $$ = $1; } | stmtList stmtBlock { $$ = addNextNull($1, $2); } + // + | stmtList error ';' { $$ = $1; } ; stmt: // IEEE: statement_or_null == function_statement_or_null @@ -3752,8 +3769,6 @@ statement_item: // IEEE: statement_item { $$ = nullptr; BBUNSUP($1, "Unsupported: expect"); } | yEXPECT '(' property_spec ')' yELSE stmt { $$ = nullptr; BBUNSUP($1, "Unsupported: expect"); } - // - | error ';' { $$ = nullptr; } ; statementFor: // IEEE: part of statement @@ -6655,11 +6670,17 @@ bins_or_empty: // ==IEEE: bins_or_empty '{' bins_or_optionsList '}' { $$ = $2; } | '{' '}' { $$ = nullptr; } | ';' { $$ = nullptr; } + // + | '{' bins_or_optionsList error '}' { $$ = $2; } + | '{' error '}' { $$ = nullptr; } ; bins_or_optionsList: // IEEE: { bins_or_options ';' } bins_or_options ';' { $$ = $1; } | bins_or_optionsList bins_or_options ';' { $$ = addNextNull($1, $2); } + // + | bins_or_optionsList error ';' { $$ = $1; } + | error ';' { $$ = nullptr; } ; bins_or_options: // ==IEEE: bins_or_options @@ -6764,11 +6785,15 @@ cross_body: // ==IEEE: cross_body // // IEEE-2012: No semicolon here, mistake in spec | '{' cross_body_itemSemiList '}' { $$ = $2; } | ';' { $$ = nullptr; } + // + | '{' cross_body_itemSemiList error '}' { $$ = $2; } + | '{' error '}' { $$ = nullptr; } ; cross_body_itemSemiList: // IEEE: part of cross_body cross_body_item ';' { $$ = $1; } | cross_body_itemSemiList cross_body_item ';' { $$ = addNextNull($1, $2); } + // | error ';' { $$ = nullptr; } | cross_body_itemSemiList error ';' { $$ = $1; } ; @@ -7369,6 +7394,7 @@ constraintIdNew: // IEEE: id part of class_constraint constraint_block: // ==IEEE: constraint_block '{' constraint_block_itemList '}' { $$ = $2; } + // | '{' error '}' { $$ = nullptr; } | '{' constraint_block_itemList error '}' { $$ = $2; } ; @@ -7427,6 +7453,7 @@ constraint_expression: // ==IEEE: constraint_expression { AstConstraintExpr* const newp = new AstConstraintExpr{$1, $3}; newp->isDisableSoft(true); $$ = newp; } + // | error ';' { $$ = nullptr; } ; @@ -7434,6 +7461,7 @@ constraint_expression: // ==IEEE: constraint_expression constraint_set: // ==IEEE: constraint_set constraint_expression { $$ = $1; } | '{' constraint_expressionList '}' { $$ = $2; } + // | '{' error '}' { $$ = nullptr; } | '{' constraint_expressionList error '}' { $$ = $2; } ; diff --git a/test_regress/t/t_source_sync.out b/test_regress/t/t_source_sync.out deleted file mode 100644 index 92dd48e27..000000000 --- a/test_regress/t/t_source_sync.out +++ /dev/null @@ -1,7 +0,0 @@ -%Error: t/t_source_sync.v:8:14: syntax error, unexpected /*verilator clocker*/, expecting ',' or ';' - 8 | logic clk /*verilator clocker*/ ; - | ^~~~~~~~~~~~~~~~~~~~~ -%Error: t/t_source_sync.v:10:1: syntax error, unexpected '}' - 10 | } ss_s; - | ^ -%Error: Exiting due to diff --git a/test_regress/t/t_source_sync.v b/test_regress/t/t_source_sync.v deleted file mode 100644 index 2ca3a7282..000000000 --- a/test_regress/t/t_source_sync.v +++ /dev/null @@ -1,12 +0,0 @@ -// DESCRIPTION: Verilator: Verilog Test module -// -// This file ONLY is placed under the Creative Commons Public Domain, for -// any use, without warranty, 2020 by Dan Petrisko. -// SPDX-License-Identifier: CC0-1.0 - -typedef struct packed { - logic clk /*verilator clocker*/; - logic data; -} ss_s; - -endmodule diff --git a/test_regress/t/t_source_sync_bad.out b/test_regress/t/t_source_sync_bad.out new file mode 100644 index 000000000..081308b24 --- /dev/null +++ b/test_regress/t/t_source_sync_bad.out @@ -0,0 +1,16 @@ +%Error: t/t_source_sync_bad.v:17:16: syntax error, unexpected IDENTIFIER + 17 | Invalid1 invalid1; + | ^~~~~~~~ +%Error: t/t_source_sync_bad.v:20:16: syntax error, unexpected IDENTIFIER + 20 | Invalid2 invalid2; + | ^~~~~~~~ +%Error: t/t_source_sync_bad.v:24:22: syntax error, unexpected IDENTIFIER, expecting "'{" + 24 | pkg::cls::defi invalid; + | ^~~~~~~ +%Error: t/t_source_sync_bad.v:30:14: syntax error, unexpected /*verilator clocker*/, expecting ',' or ';' + 30 | logic clk /*verilator clocker*/ ; + | ^~~~~~~~~~~~~~~~~~~~~ +%Error: t/t_source_sync_bad.v:34:1: syntax error, unexpected endmodule + 34 | endmodule + | ^~~~~~~~~ +%Error: Exiting due to diff --git a/test_regress/t/t_source_sync.py b/test_regress/t/t_source_sync_bad.py similarity index 100% rename from test_regress/t/t_source_sync.py rename to test_regress/t/t_source_sync_bad.py diff --git a/test_regress/t/t_source_sync_bad.v b/test_regress/t/t_source_sync_bad.v new file mode 100644 index 000000000..32b7559ab --- /dev/null +++ b/test_regress/t/t_source_sync_bad.v @@ -0,0 +1,34 @@ +// DESCRIPTION: Verilator: Verilog Test module +// +// This file ONLY is placed under the Creative Commons Public Domain, for +// any use, without warranty, 2020 by Dan Petrisko. +// SPDX-License-Identifier: CC0-1.0 + +package pkg; + class cls; + typedef unknown defu; + typedef int defi; + endclass +endpackage + +module t; + task tsk; + begin + Invalid1 invalid1; // invalid declaration + pkg::cls::defi valid1; // valid declaration + pkg::cls::defu valid2; // valid declaration + Invalid2 invalid2; // invalid declaration + + valid1 = 5; // valid statement + + pkg::cls::defi invalid; // invalid statement + end + endtask +endmodule + +typedef struct packed { + logic clk /*verilator clocker*/; + logic data; +} ss_s; + +endmodule