1 /*
2 * Copyright (C) 2009 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "edify/expr.h"
18
19 #include <stdarg.h>
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <string.h>
23 #include <unistd.h>
24
25 #include <memory>
26 #include <string>
27 #include <unordered_map>
28 #include <vector>
29
30 #include <android-base/parseint.h>
31 #include <android-base/stringprintf.h>
32 #include <android-base/strings.h>
33
34 #include "otautil/error_code.h"
35
36 // Functions should:
37 //
38 // - return a malloc()'d string
39 // - if Evaluate() on any argument returns nullptr, return nullptr.
40
BooleanString(const std::string & s)41 static bool BooleanString(const std::string& s) {
42 return !s.empty();
43 }
44
Evaluate(State * state,const std::unique_ptr<Expr> & expr,std::string * result)45 bool Evaluate(State* state, const std::unique_ptr<Expr>& expr, std::string* result) {
46 if (result == nullptr) {
47 return false;
48 }
49
50 std::unique_ptr<Value> v(expr->fn(expr->name.c_str(), state, expr->argv));
51 if (!v) {
52 return false;
53 }
54 if (v->type != Value::Type::STRING) {
55 ErrorAbort(state, kArgsParsingFailure, "expecting string, got value type %d", v->type);
56 return false;
57 }
58
59 *result = v->data;
60 return true;
61 }
62
EvaluateValue(State * state,const std::unique_ptr<Expr> & expr)63 Value* EvaluateValue(State* state, const std::unique_ptr<Expr>& expr) {
64 return expr->fn(expr->name.c_str(), state, expr->argv);
65 }
66
StringValue(const char * str)67 Value* StringValue(const char* str) {
68 if (str == nullptr) {
69 return nullptr;
70 }
71 return new Value(Value::Type::STRING, str);
72 }
73
StringValue(const std::string & str)74 Value* StringValue(const std::string& str) {
75 return StringValue(str.c_str());
76 }
77
ConcatFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)78 Value* ConcatFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
79 if (argv.empty()) {
80 return StringValue("");
81 }
82 std::string result;
83 for (size_t i = 0; i < argv.size(); ++i) {
84 std::string str;
85 if (!Evaluate(state, argv[i], &str)) {
86 return nullptr;
87 }
88 result += str;
89 }
90
91 return StringValue(result);
92 }
93
IfElseFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)94 Value* IfElseFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
95 if (argv.size() != 2 && argv.size() != 3) {
96 state->errmsg = "ifelse expects 2 or 3 arguments";
97 return nullptr;
98 }
99
100 std::string cond;
101 if (!Evaluate(state, argv[0], &cond)) {
102 return nullptr;
103 }
104
105 if (!cond.empty()) {
106 return EvaluateValue(state, argv[1]);
107 } else if (argv.size() == 3) {
108 return EvaluateValue(state, argv[2]);
109 }
110
111 return StringValue("");
112 }
113
AbortFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)114 Value* AbortFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
115 std::string msg;
116 if (!argv.empty() && Evaluate(state, argv[0], &msg)) {
117 state->errmsg += msg;
118 } else {
119 state->errmsg += "called abort()";
120 }
121 return nullptr;
122 }
123
AssertFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)124 Value* AssertFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
125 for (size_t i = 0; i < argv.size(); ++i) {
126 std::string result;
127 if (!Evaluate(state, argv[i], &result)) {
128 return nullptr;
129 }
130 if (result.empty()) {
131 int len = argv[i]->end - argv[i]->start;
132 state->errmsg = "assert failed: " + state->script.substr(argv[i]->start, len);
133 return nullptr;
134 }
135 }
136 return StringValue("");
137 }
138
SleepFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)139 Value* SleepFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
140 std::string val;
141 if (!Evaluate(state, argv[0], &val)) {
142 return nullptr;
143 }
144
145 int v;
146 if (!android::base::ParseInt(val.c_str(), &v, 0)) {
147 return nullptr;
148 }
149 sleep(v);
150
151 return StringValue(val);
152 }
153
StdoutFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)154 Value* StdoutFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
155 for (size_t i = 0; i < argv.size(); ++i) {
156 std::string v;
157 if (!Evaluate(state, argv[i], &v)) {
158 return nullptr;
159 }
160 fputs(v.c_str(), stdout);
161 }
162 return StringValue("");
163 }
164
LogicalAndFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)165 Value* LogicalAndFn(const char* name, State* state,
166 const std::vector<std::unique_ptr<Expr>>& argv) {
167 std::string left;
168 if (!Evaluate(state, argv[0], &left)) {
169 return nullptr;
170 }
171 if (BooleanString(left)) {
172 return EvaluateValue(state, argv[1]);
173 } else {
174 return StringValue("");
175 }
176 }
177
LogicalOrFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)178 Value* LogicalOrFn(const char* name, State* state,
179 const std::vector<std::unique_ptr<Expr>>& argv) {
180 std::string left;
181 if (!Evaluate(state, argv[0], &left)) {
182 return nullptr;
183 }
184 if (!BooleanString(left)) {
185 return EvaluateValue(state, argv[1]);
186 } else {
187 return StringValue(left);
188 }
189 }
190
LogicalNotFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)191 Value* LogicalNotFn(const char* name, State* state,
192 const std::vector<std::unique_ptr<Expr>>& argv) {
193 std::string val;
194 if (!Evaluate(state, argv[0], &val)) {
195 return nullptr;
196 }
197
198 return StringValue(BooleanString(val) ? "" : "t");
199 }
200
SubstringFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)201 Value* SubstringFn(const char* name, State* state,
202 const std::vector<std::unique_ptr<Expr>>& argv) {
203 std::string needle;
204 if (!Evaluate(state, argv[0], &needle)) {
205 return nullptr;
206 }
207
208 std::string haystack;
209 if (!Evaluate(state, argv[1], &haystack)) {
210 return nullptr;
211 }
212
213 std::string result = (haystack.find(needle) != std::string::npos) ? "t" : "";
214 return StringValue(result);
215 }
216
EqualityFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)217 Value* EqualityFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
218 std::string left;
219 if (!Evaluate(state, argv[0], &left)) {
220 return nullptr;
221 }
222 std::string right;
223 if (!Evaluate(state, argv[1], &right)) {
224 return nullptr;
225 }
226
227 const char* result = (left == right) ? "t" : "";
228 return StringValue(result);
229 }
230
InequalityFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)231 Value* InequalityFn(const char* name, State* state,
232 const std::vector<std::unique_ptr<Expr>>& argv) {
233 std::string left;
234 if (!Evaluate(state, argv[0], &left)) {
235 return nullptr;
236 }
237 std::string right;
238 if (!Evaluate(state, argv[1], &right)) {
239 return nullptr;
240 }
241
242 const char* result = (left != right) ? "t" : "";
243 return StringValue(result);
244 }
245
SequenceFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)246 Value* SequenceFn(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
247 std::unique_ptr<Value> left(EvaluateValue(state, argv[0]));
248 if (!left) {
249 return nullptr;
250 }
251 return EvaluateValue(state, argv[1]);
252 }
253
LessThanIntFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)254 Value* LessThanIntFn(const char* name, State* state,
255 const std::vector<std::unique_ptr<Expr>>& argv) {
256 if (argv.size() != 2) {
257 state->errmsg = "less_than_int expects 2 arguments";
258 return nullptr;
259 }
260
261 std::vector<std::string> args;
262 if (!ReadArgs(state, argv, &args)) {
263 return nullptr;
264 }
265
266 // Parse up to at least long long or 64-bit integers.
267 int64_t l_int;
268 if (!android::base::ParseInt(args[0].c_str(), &l_int)) {
269 state->errmsg = "failed to parse int in " + args[0];
270 return nullptr;
271 }
272
273 int64_t r_int;
274 if (!android::base::ParseInt(args[1].c_str(), &r_int)) {
275 state->errmsg = "failed to parse int in " + args[1];
276 return nullptr;
277 }
278
279 return StringValue(l_int < r_int ? "t" : "");
280 }
281
GreaterThanIntFn(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)282 Value* GreaterThanIntFn(const char* name, State* state,
283 const std::vector<std::unique_ptr<Expr>>& argv) {
284 if (argv.size() != 2) {
285 state->errmsg = "greater_than_int expects 2 arguments";
286 return nullptr;
287 }
288
289 std::vector<std::string> args;
290 if (!ReadArgs(state, argv, &args)) {
291 return nullptr;
292 }
293
294 // Parse up to at least long long or 64-bit integers.
295 int64_t l_int;
296 if (!android::base::ParseInt(args[0].c_str(), &l_int)) {
297 state->errmsg = "failed to parse int in " + args[0];
298 return nullptr;
299 }
300
301 int64_t r_int;
302 if (!android::base::ParseInt(args[1].c_str(), &r_int)) {
303 state->errmsg = "failed to parse int in " + args[1];
304 return nullptr;
305 }
306
307 return StringValue(l_int > r_int ? "t" : "");
308 }
309
Literal(const char * name,State * state,const std::vector<std::unique_ptr<Expr>> & argv)310 Value* Literal(const char* name, State* state, const std::vector<std::unique_ptr<Expr>>& argv) {
311 return StringValue(name);
312 }
313
314 // -----------------------------------------------------------------
315 // the function table
316 // -----------------------------------------------------------------
317
318 static std::unordered_map<std::string, Function> fn_table;
319
RegisterFunction(const std::string & name,Function fn)320 void RegisterFunction(const std::string& name, Function fn) {
321 fn_table[name] = fn;
322 }
323
FindFunction(const std::string & name)324 Function FindFunction(const std::string& name) {
325 if (fn_table.find(name) == fn_table.end()) {
326 return nullptr;
327 } else {
328 return fn_table[name];
329 }
330 }
331
RegisterBuiltins()332 void RegisterBuiltins() {
333 RegisterFunction("ifelse", IfElseFn);
334 RegisterFunction("abort", AbortFn);
335 RegisterFunction("assert", AssertFn);
336 RegisterFunction("concat", ConcatFn);
337 RegisterFunction("is_substring", SubstringFn);
338 RegisterFunction("stdout", StdoutFn);
339 RegisterFunction("sleep", SleepFn);
340
341 RegisterFunction("less_than_int", LessThanIntFn);
342 RegisterFunction("greater_than_int", GreaterThanIntFn);
343 }
344
345
346 // -----------------------------------------------------------------
347 // convenience methods for functions
348 // -----------------------------------------------------------------
349
350 // Evaluate the expressions in argv, and put the results of strings in args. If any expression
351 // evaluates to nullptr, return false. Return true on success.
ReadArgs(State * state,const std::vector<std::unique_ptr<Expr>> & argv,std::vector<std::string> * args)352 bool ReadArgs(State* state, const std::vector<std::unique_ptr<Expr>>& argv,
353 std::vector<std::string>* args) {
354 return ReadArgs(state, argv, args, 0, argv.size());
355 }
356
ReadArgs(State * state,const std::vector<std::unique_ptr<Expr>> & argv,std::vector<std::string> * args,size_t start,size_t len)357 bool ReadArgs(State* state, const std::vector<std::unique_ptr<Expr>>& argv,
358 std::vector<std::string>* args, size_t start, size_t len) {
359 if (args == nullptr) {
360 return false;
361 }
362 if (start + len > argv.size()) {
363 return false;
364 }
365 for (size_t i = start; i < start + len; ++i) {
366 std::string var;
367 if (!Evaluate(state, argv[i], &var)) {
368 args->clear();
369 return false;
370 }
371 args->push_back(var);
372 }
373 return true;
374 }
375
376 // Evaluate the expressions in argv, and put the results of Value* in args. If any expression
377 // evaluate to nullptr, return false. Return true on success.
ReadValueArgs(State * state,const std::vector<std::unique_ptr<Expr>> & argv,std::vector<std::unique_ptr<Value>> * args)378 bool ReadValueArgs(State* state, const std::vector<std::unique_ptr<Expr>>& argv,
379 std::vector<std::unique_ptr<Value>>* args) {
380 return ReadValueArgs(state, argv, args, 0, argv.size());
381 }
382
ReadValueArgs(State * state,const std::vector<std::unique_ptr<Expr>> & argv,std::vector<std::unique_ptr<Value>> * args,size_t start,size_t len)383 bool ReadValueArgs(State* state, const std::vector<std::unique_ptr<Expr>>& argv,
384 std::vector<std::unique_ptr<Value>>* args, size_t start, size_t len) {
385 if (args == nullptr) {
386 return false;
387 }
388 if (len == 0 || start + len > argv.size()) {
389 return false;
390 }
391 for (size_t i = start; i < start + len; ++i) {
392 std::unique_ptr<Value> v(EvaluateValue(state, argv[i]));
393 if (!v) {
394 args->clear();
395 return false;
396 }
397 args->push_back(std::move(v));
398 }
399 return true;
400 }
401
402 // Use printf-style arguments to compose an error message to put into
403 // *state. Returns nullptr.
ErrorAbort(State * state,const char * format,...)404 Value* ErrorAbort(State* state, const char* format, ...) {
405 va_list ap;
406 va_start(ap, format);
407 android::base::StringAppendV(&state->errmsg, format, ap);
408 va_end(ap);
409 return nullptr;
410 }
411
ErrorAbort(State * state,CauseCode cause_code,const char * format,...)412 Value* ErrorAbort(State* state, CauseCode cause_code, const char* format, ...) {
413 std::string err_message;
414 va_list ap;
415 va_start(ap, format);
416 android::base::StringAppendV(&err_message, format, ap);
417 va_end(ap);
418 // Ensure that there's exactly one line break at the end of the error message.
419 state->errmsg = android::base::Trim(err_message) + "\n";
420 state->cause_code = cause_code;
421 return nullptr;
422 }
423
State(const std::string & script,void * cookie)424 State::State(const std::string& script, void* cookie)
425 : script(script), cookie(cookie), error_code(kNoError), cause_code(kNoCause) {}
426