如何在flex-lexer中使用GNU readline?

时间:2016-04-19 01:18:42

标签: c readline flex-lexer lemon

我认为使用GNU Readline库进行命令行提示很好,我想要我正在使用的shell的功能。现在readline适合我(我的环境是CLion,CMake,Ubuntu,BSD,C,flex-lexer和lemon-parser)但是我还需要flex和yacc同时工作来扫描和解析输入,但代码似乎“不相容” - 他们真的吗?

    params[0] = NULL;
   printf("> ");

    i=1;
    do {
        lexCode = yylex(scanner);

        /*  snprintf(shell_prompt, sizeof(shell_prompt), "%s:%s $ ", getenv("USER"), getcwd(NULL, 1024));
         Display prompt and read input (NB: input must be freed after use)...*/



        text = strdup(yyget_text(scanner));
        /*
        input = readline(text);

        if (!input)
            break;

        add_history(input);

        free(input);*/
        printf("lexcode %i Text %s\n", lexCode, text);
        if (lexCode == 4) {
            params[i++] = mystring;
            if (strcmp(text, "\'\0")) {
                params[i++] = mystring;
            }
        } else
        if (lexCode != EOL) {
                params[i++] = text;
                printf("B%s\n", text);
        }
        Parse(shellParser, lexCode, text);
        if (lexCode == EOL) {
            dump_argv("Before exec_arguments", i, params);
            exec_arguments(i, params);
            corpse_collector();
            Parse(shellParser, 0, NULL);
            i=1;
        }
    } while (lexCode > 0);
    if (-1 == lexCode) {
        fprintf(stderr, "The scanner encountered an error.\n");
    }

上面的代码有解析和扫描工作,并注释掉了readline功能,如果我想要同时使用这两个功能。我可以让它发挥作用吗?

2 个答案:

答案 0 :(得分:4)

阅读GNU readline的文档。阅读tty demystified页面。仅当您的 stdin 是tty时,您才会使用readline,因此请使用isatty(3)作为isatty(STDIN_FILENO)来检测。{/ p>

使用readline的shell的基本行为只是使用

char *readline (const char *prompt);

功能。因此,您希望调整解析器以从缓冲区读取,而不是从stdin读取。这是通常的做法。在完成后,不要忘记测试(反对失败)对readlinefree结果缓冲区的调用。

然后,您想要添加一些completion。那是困难的部分。看看其他贝壳(zshbashfish ......)正在做什么,至少是为了灵感。请注意,默认文件名完成可能就足够了,至少在启动时。

顺便说一句,我不会同时使用 lemon bison 来解析相同的输入。实际上,对于shell(因为它的语法非常简单),我只会使用一些手写的recursive-descent parser

答案 1 :(得分:2)

flex + readline示例

快速弯曲" shell"只能lsdate

%{
  #include <stdlib.h>
  #include <readline/readline.h>
  #include <readline/history.h>
  #define YY_INPUT(buf,result,max_size) result = mygetinput(buf, max_size);

  static int mygetinput(char *buf, int size) {
    char *line;
    if (feof(yyin))  return YY_NULL;
    line = readline("> ");
    if(!line)        return YY_NULL;
    if(strlen(line) > size-2){
       fprintf(stderr,"input line too long\n"); return YY_NULL; }
    sprintf(buf,"%s\n",line);
    add_history(line);
    free(line);
    return strlen(buf);
  }   
%}

%option noyywrap    
%%
ls.*         system(yytext);
date.*       system(yytext);
.+           fprintf(stderr, "Error: unknown comand\n");
[ \t\n]+     {}
%%

用以下内容构建它:

flex mysh.fl
cc -o mysh lex.yy.c -lreadline -lfl