2009-06-09 02:35:39 +02:00
|
|
|
%{
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2009 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "expr.h"
|
2009-06-12 21:24:39 +02:00
|
|
|
#include "yydefs.h"
|
2009-06-09 02:35:39 +02:00
|
|
|
#include "parser.h"
|
|
|
|
|
|
|
|
extern int gLine;
|
|
|
|
extern int gColumn;
|
|
|
|
|
2009-06-12 02:21:44 +02:00
|
|
|
void yyerror(Expr** root, int* error_count, const char* s);
|
|
|
|
int yyparse(Expr** root, int* error_count);
|
2009-06-09 02:35:39 +02:00
|
|
|
|
2014-02-14 00:07:56 +01:00
|
|
|
struct yy_buffer_state;
|
|
|
|
void yy_switch_to_buffer(struct yy_buffer_state* new_buffer);
|
|
|
|
struct yy_buffer_state* yy_scan_string(const char* yystr);
|
|
|
|
|
2016-10-11 07:52:18 +02:00
|
|
|
// Convenience function for building expressions with a fixed number
|
|
|
|
// of arguments.
|
|
|
|
static Expr* Build(Function fn, YYLTYPE loc, size_t count, ...) {
|
|
|
|
va_list v;
|
|
|
|
va_start(v, count);
|
|
|
|
Expr* e = static_cast<Expr*>(malloc(sizeof(Expr)));
|
|
|
|
e->fn = fn;
|
|
|
|
e->name = "(operator)";
|
|
|
|
e->argc = count;
|
|
|
|
e->argv = static_cast<Expr**>(malloc(count * sizeof(Expr*)));
|
|
|
|
for (size_t i = 0; i < count; ++i) {
|
|
|
|
e->argv[i] = va_arg(v, Expr*);
|
|
|
|
}
|
|
|
|
va_end(v);
|
|
|
|
e->start = loc.start;
|
|
|
|
e->end = loc.end;
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
2009-06-09 02:35:39 +02:00
|
|
|
%}
|
|
|
|
|
2009-06-12 21:24:39 +02:00
|
|
|
%locations
|
|
|
|
|
2009-06-09 02:35:39 +02:00
|
|
|
%union {
|
|
|
|
char* str;
|
|
|
|
Expr* expr;
|
|
|
|
struct {
|
|
|
|
int argc;
|
|
|
|
Expr** argv;
|
|
|
|
} args;
|
|
|
|
}
|
|
|
|
|
|
|
|
%token AND OR SUBSTR SUPERSTR EQ NE IF THEN ELSE ENDIF
|
|
|
|
%token <str> STRING BAD
|
|
|
|
%type <expr> expr
|
|
|
|
%type <args> arglist
|
|
|
|
|
|
|
|
%parse-param {Expr** root}
|
2009-06-12 02:21:44 +02:00
|
|
|
%parse-param {int* error_count}
|
2009-06-09 02:35:39 +02:00
|
|
|
%error-verbose
|
|
|
|
|
|
|
|
/* declarations in increasing order of precedence */
|
|
|
|
%left ';'
|
|
|
|
%left ','
|
|
|
|
%left OR
|
|
|
|
%left AND
|
|
|
|
%left EQ NE
|
|
|
|
%left '+'
|
|
|
|
%right '!'
|
|
|
|
|
|
|
|
%%
|
|
|
|
|
|
|
|
input: expr { *root = $1; }
|
|
|
|
;
|
|
|
|
|
|
|
|
expr: STRING {
|
2016-10-11 07:52:18 +02:00
|
|
|
$$ = static_cast<Expr*>(malloc(sizeof(Expr)));
|
2009-06-09 02:35:39 +02:00
|
|
|
$$->fn = Literal;
|
|
|
|
$$->name = $1;
|
|
|
|
$$->argc = 0;
|
|
|
|
$$->argv = NULL;
|
2009-06-12 21:24:39 +02:00
|
|
|
$$->start = @$.start;
|
|
|
|
$$->end = @$.end;
|
2009-06-09 02:35:39 +02:00
|
|
|
}
|
2009-06-12 21:24:39 +02:00
|
|
|
| '(' expr ')' { $$ = $2; $$->start=@$.start; $$->end=@$.end; }
|
|
|
|
| expr ';' { $$ = $1; $$->start=@1.start; $$->end=@1.end; }
|
|
|
|
| expr ';' expr { $$ = Build(SequenceFn, @$, 2, $1, $3); }
|
|
|
|
| error ';' expr { $$ = $3; $$->start=@$.start; $$->end=@$.end; }
|
|
|
|
| expr '+' expr { $$ = Build(ConcatFn, @$, 2, $1, $3); }
|
|
|
|
| expr EQ expr { $$ = Build(EqualityFn, @$, 2, $1, $3); }
|
|
|
|
| expr NE expr { $$ = Build(InequalityFn, @$, 2, $1, $3); }
|
|
|
|
| expr AND expr { $$ = Build(LogicalAndFn, @$, 2, $1, $3); }
|
|
|
|
| expr OR expr { $$ = Build(LogicalOrFn, @$, 2, $1, $3); }
|
|
|
|
| '!' expr { $$ = Build(LogicalNotFn, @$, 1, $2); }
|
|
|
|
| IF expr THEN expr ENDIF { $$ = Build(IfElseFn, @$, 2, $2, $4); }
|
|
|
|
| IF expr THEN expr ELSE expr ENDIF { $$ = Build(IfElseFn, @$, 3, $2, $4, $6); }
|
2009-06-09 02:35:39 +02:00
|
|
|
| STRING '(' arglist ')' {
|
2016-10-11 07:52:18 +02:00
|
|
|
$$ = static_cast<Expr*>(malloc(sizeof(Expr)));
|
2009-06-09 02:35:39 +02:00
|
|
|
$$->fn = FindFunction($1);
|
2016-10-11 07:52:18 +02:00
|
|
|
if ($$->fn == nullptr) {
|
2009-06-09 02:35:39 +02:00
|
|
|
char buffer[256];
|
|
|
|
snprintf(buffer, sizeof(buffer), "unknown function \"%s\"", $1);
|
2009-06-12 02:21:44 +02:00
|
|
|
yyerror(root, error_count, buffer);
|
2009-06-09 02:35:39 +02:00
|
|
|
YYERROR;
|
|
|
|
}
|
|
|
|
$$->name = $1;
|
|
|
|
$$->argc = $3.argc;
|
|
|
|
$$->argv = $3.argv;
|
2009-06-12 21:24:39 +02:00
|
|
|
$$->start = @$.start;
|
|
|
|
$$->end = @$.end;
|
2009-06-09 02:35:39 +02:00
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
arglist: /* empty */ {
|
|
|
|
$$.argc = 0;
|
|
|
|
$$.argv = NULL;
|
|
|
|
}
|
|
|
|
| expr {
|
|
|
|
$$.argc = 1;
|
2016-10-11 07:52:18 +02:00
|
|
|
$$.argv = static_cast<Expr**>(malloc(sizeof(Expr*)));
|
2009-06-09 02:35:39 +02:00
|
|
|
$$.argv[0] = $1;
|
|
|
|
}
|
|
|
|
| arglist ',' expr {
|
|
|
|
$$.argc = $1.argc + 1;
|
2016-10-11 07:52:18 +02:00
|
|
|
$$.argv = static_cast<Expr**>(realloc($$.argv, $$.argc * sizeof(Expr*)));
|
2009-06-09 02:35:39 +02:00
|
|
|
$$.argv[$$.argc-1] = $3;
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
%%
|
|
|
|
|
2009-06-12 02:21:44 +02:00
|
|
|
void yyerror(Expr** root, int* error_count, const char* s) {
|
2009-06-09 02:35:39 +02:00
|
|
|
if (strlen(s) == 0) {
|
|
|
|
s = "syntax error";
|
|
|
|
}
|
|
|
|
printf("line %d col %d: %s\n", gLine, gColumn, s);
|
2009-06-12 02:21:44 +02:00
|
|
|
++*error_count;
|
2009-06-09 02:35:39 +02:00
|
|
|
}
|
2014-02-14 00:07:56 +01:00
|
|
|
|
|
|
|
int parse_string(const char* str, Expr** root, int* error_count) {
|
|
|
|
yy_switch_to_buffer(yy_scan_string(str));
|
|
|
|
return yyparse(root, error_count);
|
|
|
|
}
|