1 %option prefix="expr_"
2 %option reentrant
3 %option bison-bridge
4
5 %{
6 #include <linux/compiler.h>
7 #include "expr.h"
8 #include "expr-bison.h"
9 #include <math.h>
10
11 char *expr_get_text(yyscan_t yyscanner);
12 YYSTYPE *expr_get_lval(yyscan_t yyscanner);
13
__value(YYSTYPE * yylval,char * str,int token)14 static double __value(YYSTYPE *yylval, char *str, int token)
15 {
16 double num;
17
18 errno = 0;
19 num = strtod(str, NULL);
20 if (errno)
21 return EXPR_ERROR;
22
23 yylval->num = num;
24 return token;
25 }
26
value(yyscan_t scanner)27 static int value(yyscan_t scanner)
28 {
29 YYSTYPE *yylval = expr_get_lval(scanner);
30 char *text = expr_get_text(scanner);
31
32 return __value(yylval, text, NUMBER);
33 }
34
35 /*
36 * Allow @ instead of / to be able to specify pmu/event/ without
37 * conflicts with normal division.
38 */
normalize(char * str,int runtime)39 static char *normalize(char *str, int runtime)
40 {
41 char *ret = str;
42 char *dst = str;
43
44 while (*str) {
45 if (*str == '\\') {
46 *dst++ = *++str;
47 if (!*str)
48 break;
49 }
50 else if (*str == '?') {
51 char *paramval;
52 int i = 0;
53 int size = asprintf(¶mval, "%d", runtime);
54
55 if (size < 0)
56 *dst++ = '0';
57 else {
58 while (i < size)
59 *dst++ = paramval[i++];
60 free(paramval);
61 }
62 }
63 else
64 *dst++ = *str;
65 str++;
66 }
67
68 *dst = 0x0;
69 return ret;
70 }
71
str(yyscan_t scanner,int token,int runtime)72 static int str(yyscan_t scanner, int token, int runtime)
73 {
74 YYSTYPE *yylval = expr_get_lval(scanner);
75 char *text = expr_get_text(scanner);
76
77 yylval->str = normalize(strdup(text), runtime);
78 if (!yylval->str)
79 return EXPR_ERROR;
80
81 yylval->str = normalize(yylval->str, runtime);
82 return token;
83 }
84
literal(yyscan_t scanner,const struct expr_scanner_ctx * sctx)85 static int literal(yyscan_t scanner, const struct expr_scanner_ctx *sctx)
86 {
87 YYSTYPE *yylval = expr_get_lval(scanner);
88
89 yylval->num = expr__get_literal(expr_get_text(scanner), sctx);
90 if (isnan(yylval->num)) {
91 if (!sctx->is_test)
92 return EXPR_ERROR;
93 yylval->num = 1;
94 }
95 return LITERAL;
96 }
97
nan_value(yyscan_t scanner)98 static int nan_value(yyscan_t scanner)
99 {
100 YYSTYPE *yylval = expr_get_lval(scanner);
101
102 yylval->num = NAN;
103 return NUMBER;
104 }
105 %}
106
107 number ([0-9]+\.?[0-9]*|[0-9]*\.?[0-9]+)(e-?[0-9]+)?
108
109 sch [-,=]
110 spec \\{sch}
111 sym [0-9a-zA-Z_\.:@?]+
112 symbol ({spec}|{sym})+
113 literal #[0-9a-zA-Z_\.\-]+
114
115 %%
116 struct expr_scanner_ctx *sctx = expr_get_extra(yyscanner);
117
118 d_ratio { return D_RATIO; }
119 max { return MAX; }
120 min { return MIN; }
121 if { return IF; }
122 else { return ELSE; }
123 source_count { return SOURCE_COUNT; }
124 has_event { return HAS_EVENT; }
125 strcmp_cpuid_str { return STRCMP_CPUID_STR; }
126 NaN { return nan_value(yyscanner); }
127 {literal} { return literal(yyscanner, sctx); }
128 {number} { return value(yyscanner); }
129 {symbol} { return str(yyscanner, ID, sctx->runtime); }
130 "|" { return '|'; }
131 "^" { return '^'; }
132 "&" { return '&'; }
133 "<" { return '<'; }
134 ">" { return '>'; }
135 "-" { return '-'; }
136 "+" { return '+'; }
137 "*" { return '*'; }
138 "/" { return '/'; }
139 "%" { return '%'; }
140 "(" { return '('; }
141 ")" { return ')'; }
142 "," { return ','; }
143 . { }
144 %%
145
146 int expr_wrap(void *scanner __maybe_unused)
147 {
148 return 1;
149 }
150