БНФК Плохое принуждение в правиле
Я пытаюсь написать компилятор с BNFC. Я собираюсь использовать BNFC для генерации абстрактного синтаксического дерева. Но я получаю ошибки и не могу понять, почему. Там, кажется, не так много документации по этому вопросу.
Вот ошибки, которые я получаю:
Bad coercion in rule _. Prog ::= Block
Bad coercion in rule _. Declarations ::= Declaration ";" Declarations
Bad coercion in rule _. Declarations ::=
Bad coercion in rule _. Declaration ::= Var_declaration
Bad coercion in rule _. Declaration ::= Fun_declaration
Bad coercion in rule _. Type ::= "int"
Bad coercion in rule _. Type ::= "real"
Bad coercion in rule _. Type ::= "bool"
Bad coercion in rule _. Array_dimensions ::= "[" Expr "]" Array_dimensions
Bad coercion in rule _. Array_dimensions ::=
Bad coercion in rule _. Fun_block ::= Declarations Fun_body
Bad coercion in rule _. Param_list ::= "(" Parameters ")"
Bad coercion in rule _. Parameters ::= Basic_declaration More_parameters
Bad coercion in rule _. Parameters ::=
Bad coercion in rule _. More_parameters ::= "," Basic_declaration More_parameters
Bad coercion in rule _. More_parameters ::=
Bad coercion in rule _. Basic_declaration ::= Ident Basic_array_dimensions ":" Type
Bad coercion in rule _. Basic_array_dimensions ::=
Bad coercion in rule _. Program_body ::= "begin" Prog_stmts "end"
Bad coercion in rule _. Fun_body ::= "begin" Prog_stmts "return" Expr ";" "end"
Bad coercion in rule _. Prog_stmts ::= Prog_stmt ";" Prog_stmts
Bad coercion in rule _. Prog_stmts ::=
Bad coercion in rule _. Identifier ::= Ident Array_dimensions
Bad coercion in rule _. Expr ::= Bint_term
Bad coercion in rule _. Bint_term ::= Bint_factor
Bad coercion in rule _. Bint_factor ::= Int_expr Compare_op Int_expr
Bad coercion in rule _. Bint_factor ::= Int_expr
Bad coercion in rule _. Int_expr ::= Int_expr Addop Int_term
Bad coercion in rule _. Int_expr ::= Int_term
Bad coercion in rule _. Int_term ::= Int_term Mulop Int_factor
Bad coercion in rule _. Int_term ::= Int_factor
Bad coercion in rule _. Int_factor ::= "(" Expr ")"
Bad coercion in rule _. Modifier_list ::= "(" Arguments ")"
Bad coercion in rule _. Modifier_list ::= Array_dimensions
Bad coercion in rule _. Arguments ::= Expr More_arguments
Bad coercion in rule _. Arguments ::=
Bad coercion in rule _. More_arguments ::= "," Expr More_arguments
Bad coercion in rule _. More_arguments ::=
Вот пример файла BNFC:
_.Prog ::= Block;
M_Prog. Block ::= Declarations Program_body;
_.Declarations ::= Declaration ";" Declarations;
_. Declarations ::= ;
_. Declaration ::= Var_declaration;
_. Declaration ::= Fun_declaration;
M_Var. Var_declaration ::= "var" Ident Array_dimensions ":" Type;
_. Type ::= "int";
_. Type ::= "real";
_. Type ::= "bool";
_. Array_dimensions ::= "[" Expr "]" Array_dimensions;
_. Array_dimensions ::=;
M_Fun. Fun_declaration ::= "fun" Ident Param_list ":" Type "{" Fun_block "}";
_. Fun_block ::= Declarations Fun_body;
_. Param_list ::= "(" Parameters ")";
_. Parameters ::= Basic_declaration More_parameters;
_. Parameters ::= ;
_. More_parameters ::= "," Basic_declaration More_parameters;
_. More_parameters ::= ;
_. Basic_declaration ::= Ident Basic_array_dimensions ":" Type;
_. Basic_array_dimensions ::= "[" "]" Basic_array_dimensions;
_. Basic_array_dimensions ::=;
Кажется, я использую _.
неправильно маркировать. Но в руководстве есть только строка или две, описывающие его использование. Что я здесь не так делаю?
1 ответ
Из документации:
Подчеркивания, конечно, имеют смысл только в качестве замены конструкторов с одним аргументом, где тип значения совпадает с типом аргумента.
Это означает, что вы можете использовать только _
если у вас есть ровно один нетерминал в правой части вашего правила, и этот нетерминал такой же, как и на левом сайте. Таким образом, вы можете сделать что-то вроде _. A ::= "(" A ")" ;
но нет _. A ::= "(" B ")" ;
ни _. A ::= "(" A A ")" ;
,
В вашем примере я предлагаю вам дать ярлык каждому правилу, _
в основном используется для упрощения AST в некоторых угловых случаях.
Кстати, есть также ярлык синтаксиса для списков вещей.