yyparse正在打印一个前导标签

时间:2021-12-30 09:43:50

In my bison/flex program, right after yyparse() is called, a leading tab is printed, but I don't know why. Can you see what's wrong?

在我的bison / flex程序中,在调用yyparse()之后,会打印一个前导选项卡,但我不知道为什么。你能看出什么是错的吗?

This calls the bison code, and right after yyparse() returns, a tab is printed.

这会调用bison代码,并在yyparse()返回后立即打印一个选项卡。

void parseArguments(int argc, char** argv)
130 {
131     int i;
132 
133     int sum = 0;
134     // calculate the length of buffer we need
135     for(i = 1; i < argc; i++)
136     {
137         sum += strlen(argv[i]) + 1;
138     }
139 
140     if(sum <= 0)
141         return;
142 
143     // make us a buffer and zero it out
144     char tempBuffer[sum];
145     memset(tempBuffer, 0, sum);
146 
147     // pointer to walk through our buffer
148     int pos = 0;
149 
150     // copy arguments into the buffer
151     for(i = 1; i < argc; i++)
152     {
153         memcpy(tempBuffer+pos, argv[i], strlen(argv[i]));
154         pos += strlen(argv[i]);
155         sprintf(tempBuffer+pos++, " ");
156     }
157 
158     // give our arguments to lex for parsing
159     configBuffer(tempBuffer);
160 


// use bison parsing
163     int returnVal = yyparse(); // after this returns a tab character has been printed


164     if(returnVal != 0)                                                                                            
   165     {   
   166         printf("yyparse failed!\n");                                                                              
   167     }                                                                                                             
   168     

All my bison rules are just regular expressions paired with a return statement. The only code of interest in bison that I could see affecting this would be this:

我的所有野牛规则都只是正则表达式与return语句配对。我能看到影响这个的野牛唯一感兴趣的代码是:

64 %%
 65 void configBuffer(char* arguments)
 66 {
 67 #ifdef DEBUG
 68     printf("Given the buffer: %s\n", arguments);
 69 #endif
 70     yy_delete_buffer(YY_CURRENT_BUFFER);
 71 
 72     yy_scan_string(arguments);
 73 }

I tried the suggestions given by several people, but still not luck. Here is my full flex file:

我尝试了几个人给出的建议,但仍然没有运气。这是我的完整flex文件:

%{
#include <string.h>
#include "CommandParser.tab.h"
%}

%%

\t {
    printf("TAB!\n");
}

" " {
    printf("SPACE!\n");
}

\n {
    return;
}

-p {
    return PRINTMODE; 
}

-x {
    return XORMODE;
}

-n {
    return NOTMODE;
}

-a {
    return ANDMODE;
}

-o {
    return ORMODE;
}

-r {
    return RANGEFLAG;
}

-l {
    return LENGTHFLAG;
}

0[xX][0-9a-fA-F]+ {
    int retVal = sscanf(yytext, "%x",&(yylval.int_val));
    if(retVal != 1)
        return;
    return NUMBER;
}

[0-9]+ {
    yylval.int_val = atoi(yytext);
    return NUMBER;
}

['"].*+['"] {
    yylval.string_val = strdup(yytext);
    return ARGUMENT;
}

[^ \t\n]+ {
    yylval.string_val = strdup(yytext);
    return ARGUMENT;
}


%%
void configBuffer(char* arguments)
{
#define DEBUG
#ifdef DEBUG
    printf("Given the buffer: %s:\n", arguments);
#endif
    yy_delete_buffer(YY_CURRENT_BUFFER);
    yy_scan_string(arguments);

}

1 个解决方案

#1


Is the tab not handled in you lexer and therefore the default rule matching and echoed is being applied?

是否lexer中没有处理选项卡,因此正在应用默认规则匹配和回显?

Put a extra match

加一场比赛

\t { printf("TAB"); }

into the code before your end code section.

进入最终代码部分之前的代码。

if that shows TAB instead of the \t, then turn the printf into an empty statement

如果显示TAB而不是\ t,则将printf转换为空语句

\t { /*printf("TAB")*/; }

After lex posting Edit:

lex发布后编辑:

Ok, after testing your lex it would seem you are matching things correctly.

好的,在测试了你的lex后,你似乎正确地匹配了东西。

I used this code to test it

我用这段代码来测试它

#include <stdio.h>
#include "CommandParser.tab.h"

YYSTYPE yylval;

int main(int argc, char* argv[])
{
    while(1)
    {
        printf("lex:%d\r\n",yylex());
    }
    return 0;
}

extern "C" int yywrap();

int yywrap ()
{
    return 1;
}

So with the input (via stdin)

所以输入(通过stdin)

-a<\ >-x<\t>-p<space>-c<\r>

I get

lex:103
SPACE!
lex:101
TAB!
lex:100
SPACE!
lex:108
lex:3

for this header file

这个头文件

#define PRINTMODE   100
#define XORMODE     101
#define NOTMODE     102
#define ANDMODE     103
#define ORMODE      104
#define LENGTHFLAG  105
#define RANGEFLAG   106
#define NUMBER      107
#define ARGUMENT    108
#define DEFUALT     0

typedef union {
    int int_val;
    char* string_val;
} YYSTYPE;

#ifdef __cplusplus
extern "C" int yylex();

extern "C" YYSTYPE yylval;
#else // __cplusplus
extern YYSTYPE yylval;
#endif // __cplusplus

So what I'd try next is replace the yyparse with this code and see what you get.

所以我接下来尝试用这段代码替换yyparse,看看你得到了什么。

while(1)
{
    printf("lex:%d\r\n",yylex());
}

If you still get the tab printed it is somehow you lexer, otherwise it is somehow your parser/main program.

如果你仍然打印出标签,那么它就是lexer,否则它就是你的解析器/主程序。

To find that out I'd replace the magic string building you do with a const string, and see what happen in that case. Basically binary search your code to find the problem spot.

为了找到这个,我将用一个const字符串替换你所做的魔术字符串,并看看在这种情况下会发生什么。基本上二进制搜索你的代码来找到问题点。

#1


Is the tab not handled in you lexer and therefore the default rule matching and echoed is being applied?

是否lexer中没有处理选项卡,因此正在应用默认规则匹配和回显?

Put a extra match

加一场比赛

\t { printf("TAB"); }

into the code before your end code section.

进入最终代码部分之前的代码。

if that shows TAB instead of the \t, then turn the printf into an empty statement

如果显示TAB而不是\ t,则将printf转换为空语句

\t { /*printf("TAB")*/; }

After lex posting Edit:

lex发布后编辑:

Ok, after testing your lex it would seem you are matching things correctly.

好的,在测试了你的lex后,你似乎正确地匹配了东西。

I used this code to test it

我用这段代码来测试它

#include <stdio.h>
#include "CommandParser.tab.h"

YYSTYPE yylval;

int main(int argc, char* argv[])
{
    while(1)
    {
        printf("lex:%d\r\n",yylex());
    }
    return 0;
}

extern "C" int yywrap();

int yywrap ()
{
    return 1;
}

So with the input (via stdin)

所以输入(通过stdin)

-a<\ >-x<\t>-p<space>-c<\r>

I get

lex:103
SPACE!
lex:101
TAB!
lex:100
SPACE!
lex:108
lex:3

for this header file

这个头文件

#define PRINTMODE   100
#define XORMODE     101
#define NOTMODE     102
#define ANDMODE     103
#define ORMODE      104
#define LENGTHFLAG  105
#define RANGEFLAG   106
#define NUMBER      107
#define ARGUMENT    108
#define DEFUALT     0

typedef union {
    int int_val;
    char* string_val;
} YYSTYPE;

#ifdef __cplusplus
extern "C" int yylex();

extern "C" YYSTYPE yylval;
#else // __cplusplus
extern YYSTYPE yylval;
#endif // __cplusplus

So what I'd try next is replace the yyparse with this code and see what you get.

所以我接下来尝试用这段代码替换yyparse,看看你得到了什么。

while(1)
{
    printf("lex:%d\r\n",yylex());
}

If you still get the tab printed it is somehow you lexer, otherwise it is somehow your parser/main program.

如果你仍然打印出标签,那么它就是lexer,否则它就是你的解析器/主程序。

To find that out I'd replace the magic string building you do with a const string, and see what happen in that case. Basically binary search your code to find the problem spot.

为了找到这个,我将用一个const字符串替换你所做的魔术字符串,并看看在这种情况下会发生什么。基本上二进制搜索你的代码来找到问题点。