15

我已经在寻找答案,但是对于一个简单的示例,我没有得到任何快速响应。

我想使用 g++ 编译一个 flex/bison 扫描器+解析器,只是因为我想使用 C++ 类来创建 AST 和类似的东西。

在互联网上搜索我发现了一些漏洞,都说唯一需要的是在 lex 文件中使用 extern "C" 声明一些函数原型。

所以我的 shady.y 文件是

%{
#include <stdio.h>
#include "opcodes.h"
#include "utils.h"

void yyerror(const char *s)
{
    fprintf(stderr, "error: %s\n", s);
}

int counter = 0;

extern "C"
{
        int yyparse(void);
        int yylex(void);  
        int yywrap()
        {
                return 1;
        }

}

%}

%token INTEGER FLOAT
%token T_SEMICOL T_COMMA T_LPAR T_RPAR T_GRID T_LSPAR T_RSPAR
%token EOL

%token T_MOV T_NOP


%% 

... GRAMMAR OMITTED ...

%%

main(int argc, char **argv)
{
    yyparse();
}

而 shady.l 文件是

%{
    #include "shady.tab.h"
%}

%%

"MOV"|"mov" { return T_MOV; }
"NOP"|"nop" { return T_NOP; }

";" { return T_SEMICOL; }
"," { return T_COMMA; }
"(" { return T_LPAR; }
")" { return T_RPAR; }
"#" { return T_GRID; }
"[" { return T_LSPAR; }
"]" { return T_RSPAR; }
[1-9][0-9]? { yylval = atoi(yytext); return INTEGER;}
[0-9]+"."[0-9]+ | "."?[0-9]? { yylval.d = atof(yytext); return FLOAT; }
\n { return EOL; }
[ \t] { /* ignore whitespace */ }
. { printf("Mystery character %c\n", *yytext); }

%%

最后在makefile中我使用g++而不是gcc:

shady: shady.l shady.y
bison -d shady.y -o shady.tab.c
flex shady.l
g++ -o $@ shady.tab.c lex.yy.c -lfl

flex 和 bison 工作正常,但在链接时出现以下错误:

Undefined symbols:
  "_yylex", referenced from:
  _yyparse in ccwb57x0.o

当然,如果我尝试更改野牛文件中有关函数的任何内容,它会说 yylex 未在 yyparse 的范围内声明。

我是否试图简单地解决比看起来更复杂的问题?实际上,我不需要一个封闭的结构来以面向对象的方式访问解析器和词法分析器,我只是想让它工作。

我只想能够在野牛文件中使用 C++(创建 AST)并从 C++ 对象调用 yyparse() ..

提前致谢

4

1 回答 1

12

extern "C" {}需要yylexshady.l

%{
    extern "C"
    {
        int yylex(void);
    }

    #include "shady.tab.h"
%}

%%

"MOV"|"mov" { return T_MOV; }
"NOP"|"nop" { return T_NOP; }

...etc...

此外,在添加了一个虚拟语法规则之后,我能够构建并运行它:

  559  flex shady.l
  560  bison -d shady.y
  561  g++ shady.tab.c lex.yy.c 
于 2009-10-20T18:02:52.363 回答