/* * yosys -- Yosys Open SYnthesis Suite * * Copyright (C) 2012 Claire Xenia Wolf * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * * --- * * The Verilog frontend. * * This frontend is using the AST frontend library (see frontends/ast/). * Thus this frontend does not generate RTLIL code directly but creates an * AST directly from the Verilog parse tree and then passes this AST to * the AST frontend library. * * --- * * A simple lexer for Verilog code. Non-preprocessor compiler directives are * handled here. The preprocessor stuff is handled in preproc.cc. Everything * else is left to the bison parser (see verilog_parser.y). * */ %option c++ %option yyclass="VerilogLexer" %option noyywrap %option nounput %option yylineno %option prefix="frontend_verilog_yy" %{ #ifdef __clang__ // bison generates code using the 'register' storage class specifier #pragma clang diagnostic ignored "-Wdeprecated-register" // flex generates weirdly-indented code #pragma clang diagnostic ignored "-Wmisleading-indentation" #endif #include "frontends/verilog/verilog_lexer.h" #include "frontends/ast/ast.h" #include "kernel/log.h" #include USING_YOSYS_NAMESPACE using namespace AST; using namespace VERILOG_FRONTEND; using parser = frontend_verilog_yy::parser; //#define YYSTYPE FRONTEND_VERILOG_YYSTYPE //#define YYLTYPE FRONTEND_VERILOG_YYLTYPE YOSYS_NAMESPACE_BEGIN #undef YY_DECL #define YY_DECL parser::symbol_type VerilogLexer::nextToken() #undef yyterminate #define yyterminate() terminate() YOSYS_NAMESPACE_END #define SV_KEYWORD(_tok) \ if (mode->sv) return _tok; \ log("Lexer warning: The SystemVerilog keyword `%s' (at %s:%d) is not "\ "recognized unless read_verilog is called with -sv!\n", YYText(), \ AST::current_filename.c_str(), yylineno); \ string_t val = new std::string(std::string("\\") + YYText()); \ return parser::make_TOK_ID(val, out_loc); #define NON_KEYWORD() \ string_t val = new std::string(std::string("\\") + YYText()); \ return parser::make_TOK_ID(val, out_loc); // #define YY_INPUT(buf,result,max_size) \ // result = readsome(*extra->lexin, buf, max_size) #define YY_USER_ACTION \ old_loc = real_loc; \ real_loc.begin = real_loc.end; \ for(int i = 0; YYText()[i] != '\0'; ++i){ \ if(YYText()[i] == '\n') { \ real_loc.end.line++; \ real_loc.end.column = 1; \ } \ else { \ real_loc.end.column++; \ } \ } #define YY_BREAK \ out_loc = old_loc; \ break; #undef YY_BUF_SIZE #define YY_BUF_SIZE 65536 static bool isUserType(ParseState* extra, std::string &s) { // check current scope then outer scopes for a name for (auto it = extra->user_type_stack.rbegin(); it != extra->user_type_stack.rend(); ++it) { if (it->count(s) > 0) { return true; } } return false; } parser::symbol_type char_tok(char c, parser::location_type loc) { switch (c) { case '!': return parser::make_TOK_EXCL(loc); case '#': return parser::make_TOK_HASH(loc); case '%': return parser::make_TOK_PERC(loc); case '&': return parser::make_TOK_AMP(loc); case '(': return parser::make_TOK_LPAREN(loc); case ')': return parser::make_TOK_RPAREN(loc); case '*': return parser::make_TOK_ASTER(loc); case '+': return parser::make_TOK_PLUS(loc); case ',': return parser::make_TOK_COMMA(loc); case '-': return parser::make_TOK_MINUS(loc); case '.': return parser::make_TOK_DOT(loc); case '/': return parser::make_TOK_SLASH(loc); case ':': return parser::make_TOK_COL(loc); case ';': return parser::make_TOK_SEMICOL(loc); case '<': return parser::make_TOK_LT(loc); case '=': return parser::make_TOK_EQ(loc); case '>': return parser::make_TOK_GT(loc); case '?': return parser::make_TOK_QUE(loc); case '@': return parser::make_TOK_AT(loc); case '[': return parser::make_TOK_LBRA(loc); case ']': return parser::make_TOK_RBRA(loc); case '^': return parser::make_TOK_CARET(loc); case '_': return parser::make_TOK_UNDER(loc); case '{': return parser::make_TOK_LCURL(loc); case '|': return parser::make_TOK_PIPE(loc); case '}': return parser::make_TOK_RCURL(loc); case '~': return parser::make_TOK_TILDE(loc); case 'n': return parser::make_TOK_n(loc); case 'p': return parser::make_TOK_p(loc); case 'x': return parser::make_TOK_x(loc); case 'z': return parser::make_TOK_z(loc); case 0: return parser::make_FRONTEND_VERILOG_YYEOF(loc); default: std::cout << ".." << +c << "..\n"; return parser::make_ch_t(c, loc); } } %} %x COMMENT %x STRING %x SYNOPSYS_TRANSLATE_OFF %x SYNOPSYS_FLAGS %x IMPORT_DPI %x BASED_CONST UNSIGNED_NUMBER [0-9][0-9_]* FIXED_POINT_NUMBER_DEC [0-9][0-9_]*\.[0-9][0-9_]*([eE][-+]?[0-9_]+)? FIXED_POINT_NUMBER_NO_DEC [0-9][0-9_]*[eE][-+]?[0-9_]+ TIME_SCALE_SUFFIX [munpf]?s %% // Initialise comment_caller to something to avoid a "maybe undefined" // warning from GCC. int comment_caller = INITIAL; "`file_push "[^\n]* { fn_stack.push_back(current_filename); ln_stack.push_back(yylineno); current_filename = YYText()+11; if (!current_filename.empty() && current_filename.front() == '"') current_filename = current_filename.substr(1); if (!current_filename.empty() && current_filename.back() == '"') current_filename = current_filename.substr(0, current_filename.size()-1); yylineno = (0); out_loc.begin.line = out_loc.end.line = 0; real_loc.begin.line = real_loc.end.line = 0; } "`file_pop"[^\n]*\n { current_filename = fn_stack.back(); fn_stack.pop_back(); yylineno = (ln_stack.back()); out_loc.begin.line = out_loc.end.line = ln_stack.back(); real_loc.begin.line = real_loc.end.line = ln_stack.back(); ln_stack.pop_back(); } "`line"[ \t]+[^ \t\r\n]+[ \t]+\"[^ \r\n]+\"[^\r\n]*\n { const char *p = YYText() + 5; while (*p == ' ' || *p == '\t') p++; yylineno = (atoi(p)); out_loc.begin.line = out_loc.end.line = atoi(p); real_loc.begin.line = real_loc.end.line = atoi(p); while (*p && *p != ' ' && *p != '\t') p++; while (*p == ' ' || *p == '\t') p++; const char *q = *p ? p + 1 : p; while (*q && *q != '"') q++; current_filename = std::string(p).substr(1, q-p-1); } "`file_notfound "[^\n]* { log_error("Can't open include file `%s'!\n", YYText() + 15); } "`timescale"[ \t]+[^ \t\r\n/]+[ \t]*"/"[ \t]*[^ \t\r\n]* /* ignore timescale directive */ "`celldefine"[^\n]* /* ignore `celldefine */ "`endcelldefine"[^\n]* /* ignore `endcelldefine */ "`default_nettype"[ \t]+[^ \t\r\n/]+ { const char *p = YYText(); while (*p != 0 && *p != ' ' && *p != '\t') p++; while (*p == ' ' || *p == '\t') p++; if (!strcmp(p, "none")) extra->default_nettype_wire = false; else if (!strcmp(p, "wire")) extra->default_nettype_wire = true; else frontend_verilog_yyerror("Unsupported default nettype: %s", p); } "`protect"[^\n]* /* ignore `protect*/ "`endprotect"[^\n]* /* ignore `endprotect*/ "`"[a-zA-Z_$][a-zA-Z0-9_$]* { frontend_verilog_yyerror("Unimplemented compiler directive or undefined macro %s.", YYText()); } "module" { return parser::make_TOK_MODULE(out_loc); } "endmodule" { return parser::make_TOK_ENDMODULE(out_loc); } "function" { return parser::make_TOK_FUNCTION(out_loc); } "endfunction" { return parser::make_TOK_ENDFUNCTION(out_loc); } "task" { return parser::make_TOK_TASK(out_loc); } "endtask" { return parser::make_TOK_ENDTASK(out_loc); } "specify" { return mode->specify ? parser::make_TOK_SPECIFY(out_loc) : parser::make_TOK_IGNORED_SPECIFY(out_loc); } "endspecify" { return parser::make_TOK_ENDSPECIFY(out_loc); } "specparam" { return parser::make_TOK_SPECPARAM(out_loc); } "package" { SV_KEYWORD(parser::make_TOK_PACKAGE(out_loc)); } "endpackage" { SV_KEYWORD(parser::make_TOK_ENDPACKAGE(out_loc)); } "interface" { SV_KEYWORD(parser::make_TOK_INTERFACE(out_loc)); } "endinterface" { SV_KEYWORD(parser::make_TOK_ENDINTERFACE(out_loc)); } "modport" { SV_KEYWORD(parser::make_TOK_MODPORT(out_loc)); } "parameter" { return parser::make_TOK_PARAMETER(out_loc); } "localparam" { return parser::make_TOK_LOCALPARAM(out_loc); } "defparam" { return parser::make_TOK_DEFPARAM(out_loc); } "assign" { return parser::make_TOK_ASSIGN(out_loc); } "always" { return parser::make_TOK_ALWAYS(out_loc); } "initial" { return parser::make_TOK_INITIAL(out_loc); } "begin" { return parser::make_TOK_BEGIN(out_loc); } "end" { return parser::make_TOK_END(out_loc); } "if" { return parser::make_TOK_IF(out_loc); } "else" { return parser::make_TOK_ELSE(out_loc); } "for" { return parser::make_TOK_FOR(out_loc); } "posedge" { return parser::make_TOK_POSEDGE(out_loc); } "negedge" { return parser::make_TOK_NEGEDGE(out_loc); } "or" { return parser::make_TOK_OR(out_loc); } "case" { return parser::make_TOK_CASE(out_loc); } "casex" { return parser::make_TOK_CASEX(out_loc); } "casez" { return parser::make_TOK_CASEZ(out_loc); } "endcase" { return parser::make_TOK_ENDCASE(out_loc); } "default" { return parser::make_TOK_DEFAULT(out_loc); } "generate" { return parser::make_TOK_GENERATE(out_loc); } "endgenerate" { return parser::make_TOK_ENDGENERATE(out_loc); } "while" { return parser::make_TOK_WHILE(out_loc); } "repeat" { return parser::make_TOK_REPEAT(out_loc); } "automatic" { return parser::make_TOK_AUTOMATIC(out_loc); } "unique" { SV_KEYWORD(parser::make_TOK_UNIQUE(out_loc)); } "unique0" { SV_KEYWORD(parser::make_TOK_UNIQUE0(out_loc)); } "priority" { SV_KEYWORD(parser::make_TOK_PRIORITY(out_loc)); } "always_comb" { SV_KEYWORD(parser::make_TOK_ALWAYS_COMB(out_loc)); } "always_ff" { SV_KEYWORD(parser::make_TOK_ALWAYS_FF(out_loc)); } "always_latch" { SV_KEYWORD(parser::make_TOK_ALWAYS_LATCH(out_loc)); } /* use special token for labels on assert, assume, cover, and restrict because it's insanley complex to fix parsing of cells otherwise. (the current cell parser forces a reduce very early to update some global state.. its a mess) */ [a-zA-Z_$][a-zA-Z0-9_$]*/[ \t\r\n]*:[ \t\r\n]*(assert|assume|cover|restrict)[^a-zA-Z0-9_$\.] { if (!strcmp(YYText(), "default")) return parser::make_TOK_DEFAULT(out_loc); string_t val = new std::string(std::string("\\") + YYText()); return parser::make_TOK_SVA_LABEL(val, out_loc); } "assert" { if (mode->formal) return parser::make_TOK_ASSERT(out_loc); SV_KEYWORD(parser::make_TOK_ASSERT(out_loc)); } "assume" { if (mode->formal) return parser::make_TOK_ASSUME(out_loc); SV_KEYWORD(parser::make_TOK_ASSUME(out_loc)); } "cover" { if (mode->formal) return parser::make_TOK_COVER(out_loc); SV_KEYWORD(parser::make_TOK_COVER(out_loc)); } "restrict" { if (mode->formal) return parser::make_TOK_RESTRICT(out_loc); SV_KEYWORD(parser::make_TOK_RESTRICT(out_loc)); } "property" { if (mode->formal) return parser::make_TOK_PROPERTY(out_loc); SV_KEYWORD(parser::make_TOK_PROPERTY(out_loc)); } "rand" { if (mode->formal) return parser::make_TOK_RAND(out_loc); SV_KEYWORD(parser::make_TOK_RAND(out_loc)); } "const" { if (mode->formal) return parser::make_TOK_CONST(out_loc); SV_KEYWORD(parser::make_TOK_CONST(out_loc)); } "checker" { if (mode->formal) return parser::make_TOK_CHECKER(out_loc); SV_KEYWORD(parser::make_TOK_CHECKER(out_loc)); } "endchecker" { if (mode->formal) return parser::make_TOK_ENDCHECKER(out_loc); SV_KEYWORD(parser::make_TOK_ENDCHECKER(out_loc)); } "bind" { if (mode->formal) return parser::make_TOK_BIND(out_loc); SV_KEYWORD(parser::make_TOK_BIND(out_loc)); } "final" { SV_KEYWORD(parser::make_TOK_FINAL(out_loc)); } "logic" { SV_KEYWORD(parser::make_TOK_LOGIC(out_loc)); } "var" { SV_KEYWORD(parser::make_TOK_VAR(out_loc)); } "bit" { SV_KEYWORD(parser::make_TOK_LOGIC(out_loc)); } "int" { SV_KEYWORD(parser::make_TOK_INT(out_loc)); } "byte" { SV_KEYWORD(parser::make_TOK_BYTE(out_loc)); } "shortint" { SV_KEYWORD(parser::make_TOK_SHORTINT(out_loc)); } "longint" { SV_KEYWORD(parser::make_TOK_LONGINT(out_loc)); } "void" { SV_KEYWORD(parser::make_TOK_VOID(out_loc)); } "eventually" { if (mode->formal) return parser::make_TOK_EVENTUALLY(out_loc); SV_KEYWORD(parser::make_TOK_EVENTUALLY(out_loc)); } "s_eventually" { if (mode->formal) return parser::make_TOK_EVENTUALLY(out_loc); SV_KEYWORD(parser::make_TOK_EVENTUALLY(out_loc)); } "input" { return parser::make_TOK_INPUT(out_loc); } "output" { return parser::make_TOK_OUTPUT(out_loc); } "inout" { return parser::make_TOK_INOUT(out_loc); } "wire" { return parser::make_TOK_WIRE(out_loc); } "tri" { return parser::make_TOK_WIRE(out_loc); } "wor" { return parser::make_TOK_WOR(out_loc); } "trior" { return parser::make_TOK_WOR(out_loc); } "wand" { return parser::make_TOK_WAND(out_loc); } "triand" { return parser::make_TOK_WAND(out_loc); } "reg" { return parser::make_TOK_REG(out_loc); } "integer" { return parser::make_TOK_INTEGER(out_loc); } "signed" { return parser::make_TOK_SIGNED(out_loc); } "unsigned" { SV_KEYWORD(parser::make_TOK_UNSIGNED(out_loc)); } "genvar" { return parser::make_TOK_GENVAR(out_loc); } "real" { return parser::make_TOK_REAL(out_loc); } "enum" { SV_KEYWORD(parser::make_TOK_ENUM(out_loc)); } "typedef" { SV_KEYWORD(parser::make_TOK_TYPEDEF(out_loc)); } "struct" { SV_KEYWORD(parser::make_TOK_STRUCT(out_loc)); } "union" { SV_KEYWORD(parser::make_TOK_UNION(out_loc)); } "packed" { SV_KEYWORD(parser::make_TOK_PACKED(out_loc)); } {UNSIGNED_NUMBER} { string_t val = new std::string(YYText()); return parser::make_TOK_CONSTVAL(val, out_loc); } \'[01zxZX] { string_t val = new std::string(YYText()); return parser::make_TOK_UNBASED_UNSIZED_CONSTVAL(val, out_loc); } \'[sS]?[bodhBODH] { BEGIN(BASED_CONST); string_t val = new std::string(YYText()); return parser::make_TOK_BASE(val, out_loc); } [0-9a-fA-FzxZX?][0-9a-fA-FzxZX?_]* { BEGIN(0); string_t val = new std::string(YYText()); return parser::make_TOK_BASED_CONSTVAL(val, out_loc); } {FIXED_POINT_NUMBER_DEC} { string_t val = new std::string(YYText()); return parser::make_TOK_REALVAL(val, out_loc); } {FIXED_POINT_NUMBER_NO_DEC} { string_t val = new std::string(YYText()); return parser::make_TOK_REALVAL(val, out_loc); } \" { BEGIN(STRING); } \\. { yymore(); real_loc = old_loc; } \" { BEGIN(0); char *yystr = strdup(YYText()); yystr[strlen(YYText()) - 1] = 0; int i = 0, j = 0; while (yystr[i]) { if (yystr[i] == '\\' && yystr[i + 1]) { i++; if (yystr[i] == 'a') yystr[i] = '\a'; else if (yystr[i] == 'f') yystr[i] = '\f'; else if (yystr[i] == 'n') yystr[i] = '\n'; else if (yystr[i] == 'r') yystr[i] = '\r'; else if (yystr[i] == 't') yystr[i] = '\t'; else if (yystr[i] == 'v') yystr[i] = '\v'; else if ('0' <= yystr[i] && yystr[i] <= '7') { yystr[i] = yystr[i] - '0'; if ('0' <= yystr[i + 1] && yystr[i + 1] <= '7') { yystr[i + 1] = yystr[i] * 8 + yystr[i + 1] - '0'; i++; } if ('0' <= yystr[i + 1] && yystr[i + 1] <= '7') { yystr[i + 1] = yystr[i] * 8 + yystr[i + 1] - '0'; i++; } } } yystr[j++] = yystr[i++]; } yystr[j] = 0; string_t val = new std::string(yystr, j); free(yystr); return parser::make_TOK_STRING(val, out_loc); } . { yymore(); real_loc = old_loc; } and|nand|or|nor|xor|xnor|not|buf|bufif0|bufif1|notif0|notif1 { auto val = new std::string(YYText()); return parser::make_TOK_PRIMITIVE(val, out_loc); } supply0 { return parser::make_TOK_SUPPLY0(out_loc); } supply1 { return parser::make_TOK_SUPPLY1(out_loc); } "$"(display[bho]?|write[bho]?|strobe|monitor|time|realtime|stop|finish|dumpfile|dumpvars|dumpon|dumpoff|dumpall) { auto val = new std::string(YYText()); return parser::make_TOK_ID(val, out_loc); } "$"(setup|hold|setuphold|removal|recovery|recrem|skew|timeskew|fullskew|nochange) { if (!mode->specify) REJECT; auto val = new std::string(YYText()); return parser::make_TOK_ID(val, out_loc); } "$"(info|warning|error|fatal) { auto val = new std::string(YYText()); return parser::make_TOK_MSG_TASKS(val, out_loc); } "$signed" { return parser::make_TOK_TO_SIGNED(out_loc); } "$unsigned" { return parser::make_TOK_TO_UNSIGNED(out_loc); } [a-zA-Z_][a-zA-Z0-9_]*::[a-zA-Z_$][a-zA-Z0-9_$]* { // package qualifier auto s = std::string("\\") + YYText(); if (extra->pkg_user_types.count(s) > 0) { // package qualified typedefed name auto val = new std::string(s); return parser::make_TOK_PKG_USER_TYPE(val, out_loc); } else { // backup before :: just return first part size_t len = strchr(YYText(), ':') - YYText(); yyless(len); auto val = new std::string(std::string("\\") + YYText()); return parser::make_TOK_ID(val, out_loc); } } [a-zA-Z_$][a-zA-Z0-9_$]* { auto s = std::string("\\") + YYText(); if (isUserType(extra, s)) { // previously typedefed name auto val = new std::string(s); return parser::make_TOK_USER_TYPE(val, out_loc); } else { auto val = new std::string(std::string("\\") + YYText()); return parser::make_TOK_ID(val, out_loc); } } [a-zA-Z_$][a-zA-Z0-9_$\.]* { auto val = new std::string(std::string("\\") + YYText()); return parser::make_TOK_ID(val, out_loc); } "/*"[ \t]*(synopsys|synthesis)[ \t]*translate_off[ \t]*"*/" { static bool printed_warning = false; if (!printed_warning) { log_warning( "Encountered `translate_off' comment! Such legacy hot " "comments are supported by Yosys, but are not part of " "any formal language specification. Using a portable " "and standards-compliant construct such as `ifdef is " "recommended!\n" ); printed_warning = true; } BEGIN(SYNOPSYS_TRANSLATE_OFF); } . /* ignore synopsys translate_off body */ \n /* ignore synopsys translate_off body */ "/*"[ \t]*(synopsys|synthesis)[ \t]*"translate_on"[ \t]*"*/" { BEGIN(0); } "/*"[ \t]*(synopsys|synthesis)[ \t]+ { BEGIN(SYNOPSYS_FLAGS); } full_case { static bool printed_warning = false; if (!printed_warning) { log_warning( "Encountered `full_case' comment! Such legacy hot " "comments are supported by Yosys, but are not part of " "any formal language specification. Using the Verilog " "`full_case' attribute or the SystemVerilog `unique' " "or `unique0' keywords is recommended!\n" ); printed_warning = true; } return parser::make_TOK_SYNOPSYS_FULL_CASE(out_loc); } parallel_case { static bool printed_warning = false; if (!printed_warning) { log_warning( "Encountered `parallel_case' comment! Such legacy hot " "comments are supported by Yosys, but are not part of " "any formal language specification. Using the Verilog " "`parallel_case' attribute or the SystemVerilog " "`unique' or `priority' keywords is recommended!\n" ); printed_warning = true; } return parser::make_TOK_SYNOPSYS_PARALLEL_CASE(out_loc); } . /* ignore everything else */ "*/" { BEGIN(0); } import[ \t\r\n]+\"(DPI|DPI-C)\"[ \t\r\n]+function[ \t\r\n]+ { BEGIN(IMPORT_DPI); return parser::make_TOK_DPI_FUNCTION(out_loc); } [a-zA-Z_$][a-zA-Z0-9_$]* { auto val = new std::string(std::string("\\") + YYText()); return parser::make_TOK_ID(val, out_loc); } [ \t\r\n] /* ignore whitespaces */ ";" { BEGIN(0); return char_tok(*YYText(), out_loc); } . { return char_tok(*YYText(), out_loc); } "\\"[^ \t\r\n]+ { auto val = new std::string(YYText()); return parser::make_TOK_ID(val, out_loc); } "(*" { return parser::make_ATTR_BEGIN(out_loc); } "*)" { return parser::make_ATTR_END(out_loc); } "{*" { return parser::make_DEFATTR_BEGIN(out_loc); } "*}" { return parser::make_DEFATTR_END(out_loc); } "**" { return parser::make_OP_POW(out_loc); } "||" { return parser::make_OP_LOR(out_loc); } "&&" { return parser::make_OP_LAND(out_loc); } "==" { return parser::make_OP_EQ(out_loc); } "!=" { return parser::make_OP_NE(out_loc); } "<=" { return parser::make_OP_LE(out_loc); } ">=" { return parser::make_OP_GE(out_loc); } "===" { return parser::make_OP_EQX(out_loc); } "!==" { return parser::make_OP_NEX(out_loc); } "~&" { return parser::make_OP_NAND(out_loc); } "~|" { return parser::make_OP_NOR(out_loc); } "~^" { return parser::make_OP_XNOR(out_loc); } "^~" { return parser::make_OP_XNOR(out_loc); } "<<" { return parser::make_OP_SHL(out_loc); } ">>" { return parser::make_OP_SHR(out_loc); } "<<<" { return parser::make_OP_SSHL(out_loc); } ">>>" { return parser::make_OP_SSHR(out_loc); } "'" { return parser::make_OP_CAST(out_loc); } "::" { return parser::make_TOK_PACKAGESEP(out_loc); } "++" { return parser::make_TOK_INCREMENT(out_loc); } "--" { return parser::make_TOK_DECREMENT(out_loc); } "+:" { return parser::make_TOK_POS_INDEXED(out_loc); } "-:" { return parser::make_TOK_NEG_INDEXED(out_loc); } ".*" { return parser::make_TOK_WILDCARD_CONNECT(out_loc); } "|=" { SV_KEYWORD(parser::make_TOK_BIT_OR_ASSIGN(out_loc)); } "&=" { SV_KEYWORD(parser::make_TOK_BIT_AND_ASSIGN(out_loc)); } "+=" { SV_KEYWORD(parser::make_TOK_ADD_ASSIGN(out_loc)); } "-=" { SV_KEYWORD(parser::make_TOK_SUB_ASSIGN(out_loc)); } "^=" { SV_KEYWORD(parser::make_TOK_BIT_XOR_ASSIGN(out_loc)); } "/=" { SV_KEYWORD(parser::make_TOK_DIV_ASSIGN(out_loc)); } "%=" { SV_KEYWORD(parser::make_TOK_MOD_ASSIGN(out_loc)); } "*=" { SV_KEYWORD(parser::make_TOK_MUL_ASSIGN(out_loc)); } "<<=" { SV_KEYWORD(parser::make_TOK_SHL_ASSIGN(out_loc)); } ">>=" { SV_KEYWORD(parser::make_TOK_SHR_ASSIGN(out_loc)); } "<<<=" { SV_KEYWORD(parser::make_TOK_SSHL_ASSIGN(out_loc)); } ">>>=" { SV_KEYWORD(parser::make_TOK_SSHR_ASSIGN(out_loc)); } [-+]?[=*]> { if (!mode->specify) REJECT; auto val = new std::string(YYText()); return parser::make_TOK_SPECIFY_OPER(val, out_loc); } "&&&" { if (!mode->specify) return parser::make_TOK_IGNORED_SPECIFY_AND(out_loc); return parser::make_TOK_SPECIFY_AND(out_loc); } {UNSIGNED_NUMBER}{TIME_SCALE_SUFFIX} { return parser::make_TOK_TIME_SCALE(out_loc); } {FIXED_POINT_NUMBER_DEC}{TIME_SCALE_SUFFIX} { return parser::make_TOK_TIME_SCALE(out_loc); } {FIXED_POINT_NUMBER_NO_DEC}{TIME_SCALE_SUFFIX} { return parser::make_TOK_TIME_SCALE(out_loc); } "/*" { comment_caller=YY_START; BEGIN(COMMENT); } . /* ignore comment body */ \n /* ignore comment body */ "*/" { BEGIN(comment_caller); } [ \t\r\n] /* ignore whitespaces */ \\[\r\n] /* ignore continuation sequence */ "//"[^\r\n]* /* ignore one-line comments */ . { return char_tok(*YYText(), out_loc); } <*>. { BEGIN(0); return char_tok(*YYText(), out_loc); } %%