%{ /* Distributed under the OSI-approved BSD 3-Clause License. See accompanying file Copyright.txt or https://cmake.org/licensing for details. */ /* This file must be translated to C and modified to build everywhere. Run bison like this: bison --name-prefix=cmExpr_yy --defines=cmExprParserTokens.h -ocmExprParser.cxx cmExprParser.y */ #include "cmConfigure.h" // IWYU pragma: keep #include #include #include #include /*-------------------------------------------------------------------------*/ #define YYDEBUG 1 #include "cmExprParserHelper.h" /* Interface to parser object. */ #include "cmExprLexer.h" /* Interface to lexer object. */ #include "cmExprParserTokens.h" /* Need YYSTYPE for YY_DECL. */ /* Forward declare the lexer entry point. */ YY_DECL; /* Helper function to forward error callback from parser. */ static void cmExpr_yyerror(yyscan_t yyscanner, const char* message); /* Disable some warnings in the generated code. */ #ifdef _MSC_VER # pragma warning (disable: 4102) /* Unused goto label. */ # pragma warning (disable: 4065) /* Switch statement contains default but no case. */ #endif #if defined(__GNUC__) && __GNUC__ >= 8 # pragma GCC diagnostic ignored "-Wconversion" # pragma GCC diagnostic ignored "-Wfree-nonheap-object" #endif #if defined(__clang__) && defined(__has_warning) # if __has_warning("-Wused-but-marked-unused") # pragma clang diagnostic ignored "-Wused-but-marked-unused" # endif #endif %} /* Generate a reentrant parser object. */ %define api.pure /* Configure the parser to use a lexer object. */ %lex-param {yyscan_t yyscanner} %parse-param {yyscan_t yyscanner} %define parse.error verbose /*-------------------------------------------------------------------------*/ /* Tokens */ %token exp_PLUS %token exp_MINUS %token exp_TIMES %token exp_DIVIDE %token exp_MOD %token exp_SHIFTLEFT %token exp_SHIFTRIGHT %token exp_OPENPARENT %token exp_CLOSEPARENT %token exp_OR; %token exp_AND; %token exp_XOR; %token exp_NOT; %token exp_NUMBER; /*-------------------------------------------------------------------------*/ /* grammar */ %% start: exp { cmExpr_yyget_extra(yyscanner)->SetResult($1); } exp: bitwiseor { $$ = $1; } | exp exp_OR bitwiseor { $$ = $1 | $3; } bitwiseor: bitwisexor { $$ = $1; } | bitwiseor exp_XOR bitwisexor { $$ = $1 ^ $3; } bitwisexor: bitwiseand { $$ = $1; } | bitwisexor exp_AND bitwiseand { $$ = $1 & $3; } bitwiseand: shift { $$ = $1; } | bitwiseand exp_SHIFTLEFT shift { $$ = $1 << $3; } | bitwiseand exp_SHIFTRIGHT shift { $$ = $1 >> $3; } shift: term { $$ = $1; } | shift exp_PLUS term { $$ = $1 + $3; } | shift exp_MINUS term { $$ = $1 - $3; } term: unary { $$ = $1; } | term exp_TIMES unary { $$ = $1 * $3; } | term exp_DIVIDE unary { if (yyvsp[0].Number == 0) { throw std::overflow_error("divide by zero"); } $$ = $1 / $3; } | term exp_MOD unary { $$ = $1 % $3; } unary: factor { $$ = $1; } | exp_PLUS unary { $$ = + $2; } | exp_MINUS unary { $$ = - $2; } | exp_NOT unary { $$ = ~ $2; } factor: exp_NUMBER { $$ = $1; } | exp_OPENPARENT exp exp_CLOSEPARENT { $$ = $2; } ; %% /* End of grammar */ /*--------------------------------------------------------------------------*/ void cmExpr_yyerror(yyscan_t yyscanner, const char* message) { cmExpr_yyget_extra(yyscanner)->Error(message); }