繁体   English   中英

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

[英]How to use GNU readline with flex-lexer?

我认为使用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功能,如果我想同时使用它们将无法工作。 我可以让它运作吗?

阅读GNU readline的文档。 阅读tty揭秘页面。 只有当你的stdin是tty时你才会想要使用readline ,所以使用isatty(3) isatty(STDIN_FILENO)来检测它。

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

char *readline (const char *prompt);

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

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

顺便说一下,我不会同时使用柠檬野牛来解析相同的输入。 实际上,对于shell(因为它的语法很简单),我只会使用一些手写的递归下降解析器

flex + readline示例

继快速柔性“空壳”,它只是可以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

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM