سؤال

OK, so I suppose my question is quite self-explanatory.

I'm currently building a parser in Bison, and I want to make error reporting somewhat better.

Currently, I've set %define parse.error verbose (which actually gives messages like syntax error, unexpected ***********************, expecting ********************.

All I want is to add some more information in the error messages, e.g. line number (in input/file/etc)

My current yyerror (well nothing... unusual... lol) :

void yyerror(const char *str)
{
    fprintf(stderr,"\x1B[35mInterpreter : \x1B[37m%s\n",str);
}

P.S.

  • I've gone through the latest Bison documentation, but I seem quite lost...
  • I've also had a look into the %locations directive, which most likely is very close to what I need - however, I still found no complete working example and I'm not sure how this is to be used.
هل كانت مفيدة؟

المحلول

So, here I'm a with a step-by-step solution :

  • We add the %locations directive in our grammar file (between %} and the first %%)
  • We make sure that our lexer file contains an include for our parser (e.g. #include "mygrammar.tab.h"), at the top
  • We add the %option yylineno option in our lexer file (between %} and the first %%)

And now, in our yyerror function (which will supposedly be in our lexer file), we may freely use this... yylineno (= current line in file being processed) :

void yyerror(const char *str)
{
    fprintf(stderr,"Error | Line: %d\n%s\n",yylineno,str);
}

Yep. Simple as that! :-)

نصائح أخرى

Whats worked for me was adding extern int yylineno in .ypp file:

/* parser.ypp */
%{
    extern int yylineno;
%}

/* scanner.lex */
...
%option yylineno

Bison ships with a number of examples to demonstrate its features, see /usr/local/share/doc/bison/examples on your machine (where the prefix /usr/local depends on your configuration.

These examples in particular might be of interest to you:

  • lexcalc uses precedence directives and location tracking. It uses Flex to generate the scanner.
  • bistromathic demonstrates best practices when using Bison.
    • Its hand-written scanner tracks locations.
    • Its interface is pure.
    • It uses %params to pass user information to the parser and scanner.
    • Its scanner uses the error token to signal lexical errors and enter error recovery.
    • Its interface is "incremental", well suited for interaction: it uses the push-parser API to feed the parser with the incoming tokens.
    • It features an interactive command line with completion based on the parser state, based on yyexpected_tokens.
    • It uses Bison's standard catalog for internationalization of generated messages.
    • It uses a custom syntax error with location, lookahead correction and token internationalization.
    • Error messages quote the source with squiggles that underline the error:
    > 123 456
    1.5-7: syntax error: expected end of file or + or - or * or / or ^ before number
        1 | 123 456
          |     ^~~
    
    • It supports debug traces with semantic values.
    • It uses named references instead of the traditional $1, $2, etc.
مرخصة بموجب: CC-BY-SA مع الإسناد
لا تنتمي إلى StackOverflow
scroll top