3 %lex-param { yyscan_t scanner }
4 %parse-param { struct grub2_parser *parser }
8 #include <talloc/talloc.h>
14 static void print_token(FILE *fp, int type, YYSTYPE value);
16 #define YYLEX_PARAM parser->scanner
17 #define YYPRINT(f, t, v) print_token(f, t, v)
19 static void yyerror(struct grub2_parser *, char const *s);
23 struct grub2_word *word;
24 struct grub2_argv *argv;
25 struct grub2_statement *statement;
26 struct grub2_statements *statements;
30 %token TOKEN_LDSQBRACKET "[["
31 %token TOKEN_RDSQBRACKET "]]"
32 %token TOKEN_CASE "case"
34 %token TOKEN_DONE "done"
35 %token TOKEN_ELIF "elif"
36 %token TOKEN_ELSE "else"
37 %token TOKEN_ESAC "esac"
39 %token TOKEN_FOR "for"
40 %token TOKEN_FUNCTION "function"
43 %token TOKEN_MENUENTRY "menuentry"
44 %token TOKEN_SELECT "select"
45 %token TOKEN_SUBMENU "submenu"
46 %token TOKEN_THEN "then"
47 %token TOKEN_TIME "time"
48 %token TOKEN_UTIL "until"
49 %token TOKEN_WHILE "while"
51 %type <statement> statement
52 %type <statements> statements
53 %type <statement> conditional
54 %type <statement> elif
55 %type <statements> elifs
62 %token <word> TOKEN_WORD
70 parser->script->statements = $1;
73 statements: /* empty */ {
74 $$ = create_statements(parser);
76 | statements statement TOKEN_EOL {
77 statement_append($1, $2);
80 | statements TOKEN_EOL {
84 conditional: statement TOKEN_EOL "then" TOKEN_EOL statements {
85 $$ = create_statement_conditional(parser, $1, $5);
88 elif: "elif" TOKEN_DELIM conditional {
93 $$ = create_statements(parser);
96 statement_append($1, $2);
102 $$ = create_statement_simple(parser, $1);
104 | '{' statements '}' {
105 $$ = create_statement_block(parser, $2);
107 | "if" TOKEN_DELIM conditional elifs "fi" {
108 $$ = create_statement_if(parser, $3, $4, NULL);
110 | "if" TOKEN_DELIM conditional
115 $$ = create_statement_if(parser, $3, $4, $7);
117 | "function" TOKEN_DELIM word TOKEN_DELIM '{' statements '}' {
118 $$ = create_statement_function(parser, $3, $6);
120 | "menuentry" TOKEN_DELIM words TOKEN_DELIM
122 $$ = create_statement_menuentry(parser, $3, $6);
124 | "submenu" TOKEN_DELIM words TOKEN_DELIM
126 /* we just flatten everything */
127 $$ = create_statement_block(parser, $6);
131 $$ = create_argv(parser);
134 | words TOKEN_DELIM word {
146 void yyerror(struct grub2_parser *parser, char const *s)
148 fprintf(stderr, "%d: error: %s '%s'\n",
149 yyget_lineno(parser->scanner),
150 s, yyget_text(parser->scanner));
153 static void print_token(FILE *fp, int type, YYSTYPE value)
155 if (type != TOKEN_WORD)
157 fprintf(fp, "%s", value.word->text);
160 struct grub2_statements *create_statements(struct grub2_parser *parser)
162 struct grub2_statements *stmts = talloc(parser,
163 struct grub2_statements);
164 list_init(&stmts->list);
168 struct grub2_statement *create_statement_simple(struct grub2_parser *parser,
169 struct grub2_argv *argv)
171 struct grub2_statement_simple *stmt =
172 talloc(parser, struct grub2_statement_simple);
173 stmt->st.type = STMT_TYPE_SIMPLE;
174 stmt->st.exec = statement_simple_execute;
179 struct grub2_statement *create_statement_menuentry(struct grub2_parser *parser,
180 struct grub2_argv *argv, struct grub2_statements *stmts)
182 struct grub2_statement_menuentry *stmt =
183 talloc(parser, struct grub2_statement_menuentry);
184 stmt->st.type = STMT_TYPE_MENUENTRY;
185 stmt->st.exec = statement_menuentry_execute;
187 stmt->statements = stmts;
191 struct grub2_statement *create_statement_conditional(
192 struct grub2_parser *parser,
193 struct grub2_statement *condition,
194 struct grub2_statements *statements)
196 struct grub2_statement_conditional *stmt =
197 talloc(parser, struct grub2_statement_conditional);
198 stmt->st.type = STMT_TYPE_CONDITIONAL;
199 stmt->condition = condition;
200 stmt->statements = statements;
204 struct grub2_statement *create_statement_if(struct grub2_parser *parser,
205 struct grub2_statement *conditional,
206 struct grub2_statements *elifs,
207 struct grub2_statements *else_case)
209 struct grub2_statement_if *stmt =
210 talloc(parser, struct grub2_statement_if);
212 list_add(&elifs->list, &conditional->list);
214 stmt->st.type = STMT_TYPE_IF;
215 stmt->st.exec = statement_if_execute;
216 stmt->conditionals = elifs;
217 stmt->else_case = else_case;
221 struct grub2_statement *create_statement_block(struct grub2_parser *parser,
222 struct grub2_statements *stmts)
224 struct grub2_statement_block *stmt =
225 talloc(parser, struct grub2_statement_block);
226 stmt->st.type = STMT_TYPE_BLOCK;
227 stmt->st.exec = NULL;
228 stmt->statements = stmts;
232 struct grub2_statement *create_statement_function(struct grub2_parser *parser,
233 struct grub2_word *name, struct grub2_statements *body)
235 struct grub2_statement_function *stmt =
236 talloc(parser, struct grub2_statement_function);
237 stmt->st.exec = statement_function_execute;
243 void statement_append(struct grub2_statements *stmts,
244 struct grub2_statement *stmt)
248 list_add_tail(&stmts->list, &stmt->list);
251 struct grub2_word *create_word_text(struct grub2_parser *parser,
254 struct grub2_word *word = talloc(parser, struct grub2_word);
255 word->type = GRUB2_WORD_TEXT;
257 word->text = talloc_strdup(word, text);
263 struct grub2_word *create_word_var(struct grub2_parser *parser,
264 const char *name, bool split)
266 struct grub2_word *word = talloc(parser, struct grub2_word);
267 word->type = GRUB2_WORD_VAR;
268 word->name = talloc_strdup(word, name);
275 struct grub2_argv *create_argv(struct grub2_parser *parser)
277 struct grub2_argv *argv = talloc(parser, struct grub2_argv);
278 list_init(&argv->words);
282 void argv_append(struct grub2_argv *argv, struct grub2_word *word)
284 list_add_tail(&argv->words, &word->argv_list);
287 void word_append(struct grub2_word *w1, struct grub2_word *w2)
293 struct grub2_parser *grub2_parser_create(struct discover_context *ctx)
295 struct grub2_parser *parser;
297 parser = talloc(ctx, struct grub2_parser);
298 yylex_init_extra(parser, &parser->scanner);
299 parser->script = create_script(parser, ctx);
304 void grub2_parser_parse(struct grub2_parser *parser, char *buf, int len)
306 YY_BUFFER_STATE bufstate;
309 bufstate = yy_scan_bytes(buf, len - 1, parser->scanner);
311 rc = yyparse(parser);
313 yy_delete_buffer(bufstate, parser->scanner);
316 script_execute(parser->script);