blob: 17f1e879a26206fc0ab6eca55ffcf31cffbc3f71 [file] [log] [blame]
reed@android.com8a1c16f2008-12-17 15:59:43 +00001#include "SkScript2.h"
reed@google.com8a85d0c2011-06-24 19:12:12 +00002#include "SkData.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +00003#include "SkFloatingPoint.h"
4#include "SkMath.h"
5#include "SkParse.h"
6#include "SkScriptCallBack.h"
7#include "SkScriptRuntime.h"
8#include "SkString.h"
9#include "SkOpArray.h"
10
11const SkScriptEngine2::OperatorAttributes SkScriptEngine2::gOpAttributes[] = {
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000012{ SkOperand2::kNoType, SkOperand2::kNoType, kNoBias, kResultIsNotBoolean },
reed@android.com8a1c16f2008-12-17 15:59:43 +000013{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar | SkOperand2::kString),
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000014 SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar | SkOperand2::kString), kTowardsString, kResultIsNotBoolean }, // kAdd
15{ SkOperand2::kS32, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kBitAnd
16{ SkOperand2::kNoType, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kBitNot
17{ SkOperand2::kS32, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kBitOr
reed@android.com8a1c16f2008-12-17 15:59:43 +000018{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar),
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000019 SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar), kNoBias, kResultIsNotBoolean }, // kDivide
reed@android.com8a1c16f2008-12-17 15:59:43 +000020{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar | SkOperand2::kString),
21 SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar |SkOperand2:: kString), kTowardsNumber,
22 kResultIsBoolean }, // kEqual
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000023{ SkOperand2::kS32, SkOperand2::kNoType, kNoBias, kResultIsNotBoolean }, // kFlipOps
reed@android.com8a1c16f2008-12-17 15:59:43 +000024{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar | SkOperand2::kString),
25 SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar | SkOperand2::kString), kTowardsNumber,
26 kResultIsBoolean }, // kGreaterEqual
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000027{ SkOperand2::kNoType, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kLogicalAnd (really, ToBool)
28{ SkOperand2::kNoType, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kLogicalNot
29{ SkOperand2::kS32, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kLogicalOr
30{ SkOperand2::kNoType, SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar), kNoBias, kResultIsNotBoolean }, // kMinus
reed@android.com8a1c16f2008-12-17 15:59:43 +000031{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar),
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000032 SkOperand2::OpType(SkOperand2::kS32 |SkOperand2:: kScalar), kNoBias, kResultIsNotBoolean }, // kModulo
reed@android.com8a1c16f2008-12-17 15:59:43 +000033{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar),
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000034 SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar), kNoBias, kResultIsNotBoolean }, // kMultiply
35{ SkOperand2::kS32, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kShiftLeft
36{ SkOperand2::kS32, SkOperand2::kS32, kNoBias, kResultIsNotBoolean }, // kShiftRight
reed@android.com8a1c16f2008-12-17 15:59:43 +000037{ SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar),
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +000038 SkOperand2::OpType(SkOperand2::kS32 | SkOperand2::kScalar), kNoBias, kResultIsNotBoolean }, // kSubtract
39{ SkOperand2::kS32, SkOperand2::kS32, kNoBias, kResultIsNotBoolean } // kXor
reed@android.com8a1c16f2008-12-17 15:59:43 +000040};
41
42#define kBracketPrecedence 16
43#define kIfElsePrecedence 15
44
45const signed char SkScriptEngine2::gPrecedence[] = {
46 17, // kUnassigned,
47 6, // kAdd,
48 10, // kBitAnd,
49 4, // kBitNot,
50 12, // kBitOr,
51 5, // kDivide,
52 9, // kEqual,
53 -1, // kFlipOps,
54 8, // kGreaterEqual,
55 13, // kLogicalAnd,
56 4, // kLogicalNot,
57 14, // kLogicalOr,
58 4, // kMinus,
59 5, // kModulo,
60 5, // kMultiply,
61 7, // kShiftLeft,
62 7, // kShiftRight, // signed
63 6, // kSubtract,
64 11, // kXor
65 kBracketPrecedence, // kArrayOp
66 kIfElsePrecedence, // kElse
67 kIfElsePrecedence, // kIf
68 kBracketPrecedence, // kParen
69};
70
71const SkScriptEngine2::TypeOp SkScriptEngine2::gTokens[] = {
72 kNop, // unassigned
73 kAddInt, // kAdd,
74 kBitAndInt, // kBitAnd,
75 kBitNotInt, // kBitNot,
76 kBitOrInt, // kBitOr,
77 kDivideInt, // kDivide,
78 kEqualInt, // kEqual,
79 kFlipOpsOp, // kFlipOps,
80 kGreaterEqualInt, // kGreaterEqual,
81 kLogicalAndInt, // kLogicalAnd,
82 kLogicalNotInt, // kLogicalNot,
83 kLogicalOrInt, // kLogicalOr,
84 kMinusInt, // kMinus,
85 kModuloInt, // kModulo,
86 kMultiplyInt, // kMultiply,
87 kShiftLeftInt, // kShiftLeft,
88 kShiftRightInt, // kShiftRight, // signed
89 kSubtractInt, // kSubtract,
90 kXorInt // kXor
91};
92
93static inline bool is_between(int c, int min, int max)
94{
95 return (unsigned)(c - min) <= (unsigned)(max - min);
96}
97
98static inline bool is_ws(int c)
99{
100 return is_between(c, 1, 32);
101}
102
103static int token_length(const char* start) {
104 char ch = start[0];
105 if (! is_between(ch, 'a' , 'z') && ! is_between(ch, 'A', 'Z') && ch != '_' && ch != '$')
106 return -1;
107 int length = 0;
108 do
109 ch = start[++length];
110 while (is_between(ch, 'a' , 'z') || is_between(ch, 'A', 'Z') || is_between(ch, '0', '9') ||
111 ch == '_' || ch == '$');
112 return length;
113}
114
115SkScriptEngine2::SkScriptEngine2(SkOperand2::OpType returnType) : fActiveStream(&fStream),
116fTokenLength(0), fReturnType(returnType), fError(kNoError),
117fAccumulatorType(SkOperand2::kNoType),
118fBranchPopAllowed(true), fConstExpression(true), fOperandInUse(false)
119{
120 Branch branch(kUnassigned, 0, 0);
121 fBranchStack.push(branch);
122 *fOpStack.push() = (Op) kParen;
123}
124
125SkScriptEngine2::~SkScriptEngine2() {
126 for (SkString** stringPtr = fTrackString.begin(); stringPtr < fTrackString.end(); stringPtr++)
127 delete *stringPtr;
128 for (SkOpArray** arrayPtr = fTrackArray.begin(); arrayPtr < fTrackArray.end(); arrayPtr++)
129 delete *arrayPtr;
130}
131
132void SkScriptEngine2::addToken(SkScriptEngine2::TypeOp op) {
133 int limit = fBranchStack.count() - 1;
134 for (int index = 0; index < limit; index++) {
135 Branch& branch = fBranchStack.index(index);
136 if (branch.fPrimed == Branch::kIsPrimed)
137 resolveBranch(branch);
138 }
139 if (fBranchPopAllowed) {
140 while (fBranchStack.top().fDone == Branch::kIsDone)
141 fBranchStack.pop();
142 }
143 unsigned char charOp = (unsigned char) op;
144 fActiveStream->write(&charOp, sizeof(charOp));
145}
146
147void SkScriptEngine2::addTokenConst(SkScriptValue2* value, AddTokenRegister reg,
148 SkOperand2::OpType toType, SkScriptEngine2::TypeOp op) {
149 if (value->fIsConstant == SkScriptValue2::kConstant && convertTo(toType, value))
150 return;
151 addTokenValue(*value, reg);
152 addToken(op);
153 value->fIsWritten = SkScriptValue2::kWritten;
154 value->fType = toType;
155}
156
157void SkScriptEngine2::addTokenInt(int integer) {
158 fActiveStream->write(&integer, sizeof(integer));
159}
160
161void SkScriptEngine2::addTokenScalar(SkScalar scalar) {
162 fActiveStream->write(&scalar, sizeof(scalar));
163}
164
165void SkScriptEngine2::addTokenString(const SkString& string) {
166 int size = string.size();
167 addTokenInt(size);
168 fActiveStream->write(string.c_str(), size);
169}
170
171void SkScriptEngine2::addTokenValue(const SkScriptValue2& value, AddTokenRegister reg) {
172 if (value.isConstant() == false) {
173 if (reg == kAccumulator) {
174 if (fAccumulatorType == SkOperand2::kNoType)
175 addToken(kAccumulatorPop);
176 } else {
177 ; // !!! incomplete?
178 }
179 return;
180 }
181 if (reg == kAccumulator && fAccumulatorType != SkOperand2::kNoType)
182 addToken(kAccumulatorPush);
183 switch (value.fType) {
184 case SkOperand2::kS32:
185 addToken(reg == kAccumulator ? kIntegerAccumulator : kIntegerOperand);
186 addTokenInt(value.fOperand.fS32);
187 if (reg == kAccumulator)
188 fAccumulatorType = SkOperand2::kS32;
189 else
190 fOperandInUse = true;
191 break;
192 case SkOperand2::kScalar:
193 addToken(reg == kAccumulator ? kScalarAccumulator : kScalarOperand);
194 addTokenScalar(value.fOperand.fScalar);
195 if (reg == kAccumulator)
196 fAccumulatorType = SkOperand2::kScalar;
197 else
198 fOperandInUse = true;
199 break;
200 case SkOperand2::kString:
201 addToken(reg == kAccumulator ? kStringAccumulator : kStringOperand);
202 addTokenString(*value.fOperand.fString);
203 if (reg == kAccumulator)
204 fAccumulatorType = SkOperand2::kString;
205 else
206 fOperandInUse = true;
207 break;
208 default:
209 SkASSERT(0); //!!! not implemented yet
210 }
211}
212
213int SkScriptEngine2::arithmeticOp(char ch, char nextChar, bool lastPush) {
214 Op op = kUnassigned;
215 bool reverseOperands = false;
216 bool negateResult = false;
217 int advance = 1;
218 switch (ch) {
219 case '+':
220 // !!! ignoring unary plus as implemented here has the side effect of
221 // suppressing errors like +"hi"
222 if (lastPush == false) // unary plus, don't push an operator
reed@android.com800046e2009-10-14 09:36:25 +0000223 return advance;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000224 op = kAdd;
225 break;
226 case '-':
227 op = lastPush ? kSubtract : kMinus;
228 break;
229 case '*':
230 op = kMultiply;
231 break;
232 case '/':
233 op = kDivide;
234 break;
235 case '>':
236 if (nextChar == '>') {
237 op = kShiftRight;
238 goto twoChar;
239 }
240 op = kGreaterEqual;
241 if (nextChar == '=')
242 goto twoChar;
243 reverseOperands = negateResult = true;
244 break;
245 case '<':
246 if (nextChar == '<') {
247 op = kShiftLeft;
248 goto twoChar;
249 }
250 op = kGreaterEqual;
251 reverseOperands = nextChar == '=';
252 negateResult = ! reverseOperands;
253 advance += reverseOperands;
254 break;
255 case '=':
256 if (nextChar == '=') {
257 op = kEqual;
258 goto twoChar;
259 }
260 break;
261 case '!':
262 if (nextChar == '=') {
263 op = kEqual;
264 negateResult = true;
265twoChar:
266 advance++;
267 break;
268 }
269 op = kLogicalNot;
270 break;
271 case '?':
272 op =(Op) kIf;
273 break;
274 case ':':
275 op = (Op) kElse;
276 break;
277 case '^':
278 op = kXor;
279 break;
280 case '(':
281 *fOpStack.push() = (Op) kParen;
reed@android.com800046e2009-10-14 09:36:25 +0000282 return advance;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000283 case '&':
284 SkASSERT(nextChar != '&');
285 op = kBitAnd;
286 break;
287 case '|':
288 SkASSERT(nextChar != '|');
289 op = kBitOr;
290 break;
291 case '%':
292 op = kModulo;
293 break;
294 case '~':
295 op = kBitNot;
296 break;
297 }
298 if (op == kUnassigned)
299 return 0;
300 signed char precedence = gPrecedence[op];
301 do {
302 int idx = 0;
303 Op compare;
304 do {
305 compare = fOpStack.index(idx);
306 if ((compare & kArtificialOp) == 0)
307 break;
308 idx++;
309 } while (true);
310 signed char topPrecedence = gPrecedence[compare];
311 SkASSERT(topPrecedence != -1);
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +0000312 if (topPrecedence > precedence || (topPrecedence == precedence &&
313 gOpAttributes[op].fLeftType == SkOperand2::kNoType)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000314 break;
315 }
316 processOp();
317 } while (true);
318 if (negateResult)
319 *fOpStack.push() = (Op) (kLogicalNot | kArtificialOp);
320 fOpStack.push(op);
321 if (reverseOperands)
322 *fOpStack.push() = (Op) (kFlipOps | kArtificialOp);
reed@android.com800046e2009-10-14 09:36:25 +0000323
324 return advance;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000325}
326
327bool SkScriptEngine2::convertParams(SkTDArray<SkScriptValue2>* params,
328 const SkOperand2::OpType* paramTypes, int paramCount) {
329 int count = params->count();
330 if (count > paramCount) {
331 SkASSERT(0);
332 return false; // too many parameters passed
333 }
334 for (int index = 0; index < count; index++)
335 convertTo(paramTypes[index], &(*params)[index]);
336 return true;
337}
338
339bool SkScriptEngine2::convertTo(SkOperand2::OpType toType, SkScriptValue2* value ) {
340 SkOperand2::OpType type = value->fType;
341 if (type == toType)
342 return true;
343 if (type == SkOperand2::kObject) {
344 if (handleUnbox(value) == false)
345 return false;
346 return convertTo(toType, value);
347 }
348 return ConvertTo(this, toType, value);
349}
350
351bool SkScriptEngine2::evaluateDot(const char*& script) {
352 size_t fieldLength = token_length(++script); // skip dot
353 SkASSERT(fieldLength > 0); // !!! add error handling
354 const char* field = script;
355 script += fieldLength;
356 bool success = handleProperty();
357 if (success == false) {
358 fError = kCouldNotFindReferencedID;
359 goto error;
360 }
361 return evaluateDotParam(script, field, fieldLength);
362error:
363 return false;
364}
365
366bool SkScriptEngine2::evaluateDotParam(const char*& script, const char* field, size_t fieldLength) {
367 SkScriptValue2& top = fValueStack.top();
368 if (top.fType != SkOperand2::kObject)
369 return false;
370 void* object = top.fOperand.fObject;
371 fValueStack.pop();
372 char ch; // see if it is a simple member or a function
373 while (is_ws(ch = script[0]))
374 script++;
375 bool success = true;
376 if (ch != '(')
377 success = handleMember(field, fieldLength, object);
378 else {
379 SkTDArray<SkScriptValue2> params;
380 *fBraceStack.push() = kFunctionBrace;
381 success = functionParams(&script, &params);
382 if (success)
383 success = handleMemberFunction(field, fieldLength, object, &params);
384 }
385 return success;
386}
387
388bool SkScriptEngine2::evaluateScript(const char** scriptPtr, SkScriptValue2* value) {
389 // fArrayOffset = 0; // no support for structures for now
390 bool success;
391 const char* inner;
392 if (strncmp(*scriptPtr, "#script:", sizeof("#script:") - 1) == 0) {
393 *scriptPtr += sizeof("#script:") - 1;
394 if (fReturnType == SkOperand2::kNoType || fReturnType == SkOperand2::kString) {
395 success = innerScript(scriptPtr, value);
396 SkASSERT(success);
397 inner = value->fOperand.fString->c_str();
398 scriptPtr = &inner;
399 }
400 }
401 success = innerScript(scriptPtr, value);
402 const char* script = *scriptPtr;
403 char ch;
404 while (is_ws(ch = script[0]))
405 script++;
406 if (ch != '\0') {
407 // error may trigger on scripts like "50,0" that were intended to be written as "[50, 0]"
408 return false;
409 }
410 return success;
411}
412
413void SkScriptEngine2::forget(SkOpArray* array) {
414 if (array->getType() == SkOperand2::kString) {
415 for (int index = 0; index < array->count(); index++) {
416 SkString* string = (*array)[index].fString;
417 int found = fTrackString.find(string);
418 if (found >= 0)
419 fTrackString.remove(found);
420 }
421 return;
422 }
423 if (array->getType() == SkOperand2::kArray) {
424 for (int index = 0; index < array->count(); index++) {
425 SkOpArray* child = (*array)[index].fArray;
426 forget(child); // forgets children of child
427 int found = fTrackArray.find(child);
428 if (found >= 0)
429 fTrackArray.remove(found);
430 }
431 }
432}
433
434bool SkScriptEngine2::functionParams(const char** scriptPtr, SkTDArray<SkScriptValue2>* params) {
435 (*scriptPtr)++; // skip open paren
436 *fOpStack.push() = (Op) kParen;
437 *fBraceStack.push() = kFunctionBrace;
438 do {
439 SkScriptValue2 value;
440 bool success = innerScript(scriptPtr, &value);
441 SkASSERT(success);
442 if (success == false)
443 return false;
444 *params->append() = value;
445 } while ((*scriptPtr)[-1] == ',');
446 fBraceStack.pop();
447 fOpStack.pop(); // pop paren
448 (*scriptPtr)++; // advance beyond close paren
449 return true;
450}
451
452size_t SkScriptEngine2::getTokenOffset() {
453 return fActiveStream->getOffset();
454}
455
456SkOperand2::OpType SkScriptEngine2::getUnboxType(SkOperand2 scriptValue) {
457 for (SkScriptCallBack** callBack = fCallBackArray.begin(); callBack < fCallBackArray.end(); callBack++) {
458 if ((*callBack)->getType() != SkScriptCallBack::kUnbox)
459 continue;
460 return (*callBack)->getReturnType(0, &scriptValue);
461 }
462 return SkOperand2::kObject;
463}
464
465bool SkScriptEngine2::innerScript(const char** scriptPtr, SkScriptValue2* value) {
466 const char* script = *scriptPtr;
467 char ch;
468 bool lastPush = false;
469 bool success = true;
470 int opBalance = fOpStack.count();
471 int baseBrace = fBraceStack.count();
472 int branchBalance = fBranchStack.count();
473 while ((ch = script[0]) != '\0') {
474 if (is_ws(ch)) {
475 script++;
476 continue;
477 }
478 SkScriptValue2 operand;
479 const char* dotCheck;
480 if (fBraceStack.count() > baseBrace) {
481 if (fBraceStack.top() == kArrayBrace) {
482 SkScriptValue2 tokenValue;
483 success = innerScript(&script, &tokenValue); // terminate and return on comma, close brace
484 SkASSERT(success);
485 {
486 SkOperand2::OpType type = fReturnType;
487 if (fReturnType == SkOperand2::kNoType) {
488 // !!! short sighted; in the future, allow each returned array component to carry
489 // its own type, and let caller do any needed conversions
490 if (value->fOperand.fArray->count() == 0)
491 value->fOperand.fArray->setType(type = tokenValue.fType);
492 else
493 type = value->fOperand.fArray->getType();
494 }
495 if (tokenValue.fType != type)
496 convertTo(type, &tokenValue);
497 *value->fOperand.fArray->append() = tokenValue.fOperand;
498 }
499 lastPush = false;
500 continue;
501 } else
502 SkASSERT(token_length(script) > 0);
503 }
504 if (lastPush != false && fTokenLength > 0) {
505 if (ch == '(') {
506 *fBraceStack.push() = kFunctionBrace;
507 SkString functionName(fToken, fTokenLength);
508
509 if (handleFunction(&script) == false)
510 return false;
511 lastPush = true;
512 continue;
513 } else if (ch == '[') {
514 if (handleProperty() == false) {
515 SkASSERT(0);
516 return false;
517 }
518 if (handleArrayIndexer(&script) == false)
519 return false;
520 lastPush = true;
521 continue;
522 } else if (ch != '.') {
523 if (handleProperty() == false) {
524 SkASSERT(0);
525 return false;
526 }
527 lastPush = true;
528 continue;
529 }
530 }
531 if (ch == '0' && (script[1] & ~0x20) == 'X') {
532 SkASSERT(lastPush == false);
533 script += 2;
534 script = SkParse::FindHex(script, (uint32_t*) &operand.fOperand.fS32);
535 SkASSERT(script);
536 goto intCommon;
537 }
538 if (lastPush == false && ch == '.')
539 goto scalarCommon;
540 if (ch >= '0' && ch <= '9') {
541 SkASSERT(lastPush == false);
542 dotCheck = SkParse::FindS32(script, &operand.fOperand.fS32);
543 if (dotCheck[0] != '.') {
544 script = dotCheck;
545intCommon:
546 operand.fType = SkOperand2::kS32;
547 } else {
548scalarCommon:
549 script = SkParse::FindScalar(script, &operand.fOperand.fScalar);
550 operand.fType = SkOperand2::kScalar;
551 }
552 operand.fIsConstant = SkScriptValue2::kConstant;
553 fValueStack.push(operand);
554 lastPush = true;
555 continue;
556 }
557 int length = token_length(script);
558 if (length > 0) {
559 SkASSERT(lastPush == false);
560 fToken = script;
561 fTokenLength = length;
562 script += length;
563 lastPush = true;
564 continue;
565 }
566 char startQuote = ch;
567 if (startQuote == '\'' || startQuote == '\"') {
568 SkASSERT(lastPush == false);
569 operand.fOperand.fString = new SkString();
570 ++script;
571 const char* stringStart = script;
572 do { // measure string
573 if (script[0] == '\\')
574 ++script;
575 ++script;
576 SkASSERT(script[0]); // !!! throw an error
577 } while (script[0] != startQuote);
578 operand.fOperand.fString->set(stringStart, script - stringStart);
579 script = stringStart;
580 char* stringWrite = operand.fOperand.fString->writable_str();
581 do { // copy string
582 if (script[0] == '\\')
583 ++script;
584 *stringWrite++ = script[0];
585 ++script;
586 SkASSERT(script[0]); // !!! throw an error
587 } while (script[0] != startQuote);
588 ++script;
589 track(operand.fOperand.fString);
590 operand.fType = SkOperand2::kString;
591 operand.fIsConstant = SkScriptValue2::kConstant;
592 fValueStack.push(operand);
593 lastPush = true;
594 continue;
595 }
596 if (ch == '.') {
597 if (fTokenLength == 0) {
598 SkScriptValue2 scriptValue;
599 SkDEBUGCODE(scriptValue.fOperand.fObject = NULL);
600 int tokenLength = token_length(++script);
601 const char* token = script;
602 script += tokenLength;
603 SkASSERT(fValueStack.count() > 0); // !!! add error handling
604 SkScriptValue2 top;
605 fValueStack.pop(&top);
606
607 addTokenInt(top.fType);
608 addToken(kBoxToken);
609 top.fType = SkOperand2::kObject;
610 top.fIsConstant = SkScriptValue2::kVariable;
611 fConstExpression = false;
612 fValueStack.push(top);
613 success = evaluateDotParam(script, token, tokenLength);
614 SkASSERT(success);
615 lastPush = true;
616 continue;
617 }
618 // get next token, and evaluate immediately
619 success = evaluateDot(script);
620 if (success == false) {
621 // SkASSERT(0);
622 return false;
623 }
624 lastPush = true;
625 continue;
626 }
627 if (ch == '[') {
628 if (lastPush == false) {
629 script++;
630 *fBraceStack.push() = kArrayBrace;
631 operand.fOperand.fArray = value->fOperand.fArray = new SkOpArray(fReturnType);
632 track(value->fOperand.fArray);
633
634 operand.fType = SkOperand2::kArray;
635 operand.fIsConstant = SkScriptValue2::kVariable;
636 fValueStack.push(operand);
637 continue;
638 }
639 if (handleArrayIndexer(&script) == false)
640 return false;
641 lastPush = true;
642 continue;
643 }
644#if 0 // structs not supported for now
645 if (ch == '{') {
646 if (lastPush == false) {
647 script++;
648 *fBraceStack.push() = kStructBrace;
649 operand.fS32 = 0;
650 *fTypeStack.push() = (SkOpType) kStruct;
651 fOperandStack.push(operand);
652 continue;
653 }
654 SkASSERT(0); // braces in other contexts aren't supported yet
655 }
656#endif
657 if (ch == ')' && fBraceStack.count() > 0) {
658 BraceStyle braceStyle = fBraceStack.top();
659 if (braceStyle == kFunctionBrace) {
660 fBraceStack.pop();
661 break;
662 }
663 }
664 if (ch == ',' || ch == ']') {
665 if (ch != ',') {
666 BraceStyle match;
667 fBraceStack.pop(&match);
668 SkASSERT(match == kArrayBrace);
669 }
670 script++;
671 // !!! see if brace or bracket is correct closer
672 break;
673 }
674 char nextChar = script[1];
675 int advance = logicalOp(ch, nextChar);
676 if (advance == 0)
677 advance = arithmeticOp(ch, nextChar, lastPush);
678 if (advance == 0) // unknown token
679 return false;
680 if (advance > 0)
681 script += advance;
682 lastPush = ch == ']' || ch == ')';
683 }
684 if (fTokenLength > 0) {
685 success = handleProperty();
686 SkASSERT(success);
687 }
688 int branchIndex = 0;
689 branchBalance = fBranchStack.count() - branchBalance;
690 fBranchPopAllowed = false;
691 while (branchIndex < branchBalance) {
692 Branch& branch = fBranchStack.index(branchIndex++);
693 if (branch.fPrimed == Branch::kIsPrimed)
694 break;
695 Op branchOp = branch.fOperator;
696 SkOperand2::OpType lastType = fValueStack.top().fType;
697 addTokenValue(fValueStack.top(), kAccumulator);
698 fValueStack.pop();
699 if (branchOp == kLogicalAnd || branchOp == kLogicalOr) {
700 if (branch.fOperator == kLogicalAnd)
701 branch.prime();
702 addToken(kToBool);
703 } else {
704 resolveBranch(branch);
705 SkScriptValue2 operand;
706 operand.fType = lastType;
707 // !!! note that many branching expressions could be constant
708 // today, we always evaluate branches as returning variables
709 operand.fIsConstant = SkScriptValue2::kVariable;
710 fValueStack.push(operand);
711 }
712 if (branch.fDone == Branch::kIsNotDone)
713 branch.prime();
714 }
715 fBranchPopAllowed = true;
716 while (fBranchStack.top().fDone == Branch::kIsDone)
717 fBranchStack.pop();
718 while (fOpStack.count() > opBalance) { // leave open paren
719 if (processOp() == false)
720 return false;
721 }
722 SkOperand2::OpType topType = fValueStack.count() > 0 ? fValueStack.top().fType : SkOperand2::kNoType;
723 if (topType != fReturnType &&
724 topType == SkOperand2::kString && fReturnType != SkOperand2::kNoType) { // if result is a string, give handle property a chance to convert it to the property value
725 SkString* string = fValueStack.top().fOperand.fString;
726 fToken = string->c_str();
727 fTokenLength = string->size();
728 fValueStack.pop();
729 success = handleProperty();
730 if (success == false) { // if it couldn't convert, return string (error?)
731 SkScriptValue2 operand;
732 operand.fType = SkOperand2::kString;
733 operand.fOperand.fString = string;
734 operand.fIsConstant = SkScriptValue2::kVariable; // !!! ?
735 fValueStack.push(operand);
736 }
737 }
738 if (fStream.getOffset() > 0) {
739 addToken(kEnd);
reed@google.com8a85d0c2011-06-24 19:12:12 +0000740 SkAutoDataUnref data(fStream.copyToData());
reed@android.com8a1c16f2008-12-17 15:59:43 +0000741#ifdef SK_DEBUG
reed@google.com8a85d0c2011-06-24 19:12:12 +0000742 decompile(data.bytes(), data.size());
reed@android.com8a1c16f2008-12-17 15:59:43 +0000743#endif
744 SkScriptRuntime runtime(fCallBackArray);
reed@google.com8a85d0c2011-06-24 19:12:12 +0000745 runtime.executeTokens((unsigned char*) data.bytes());
reed@android.com8a1c16f2008-12-17 15:59:43 +0000746 SkScriptValue2 value1;
747 runtime.getResult(&value1.fOperand);
748 value1.fType = fReturnType;
749 fValueStack.push(value1);
750 }
751 if (value) {
752 if (fValueStack.count() == 0)
753 return false;
754 fValueStack.pop(value);
755 if (value->fType != fReturnType && value->fType == SkOperand2::kObject &&
756 fReturnType != SkOperand2::kNoType)
757 convertTo(fReturnType, value);
758 }
759 // if (fBranchStack.top().fOpStackDepth > fOpStack.count())
760 // resolveBranch();
761 *scriptPtr = script;
762 return true; // no error
763}
764
765bool SkScriptEngine2::handleArrayIndexer(const char** scriptPtr) {
766 SkScriptValue2 scriptValue;
767 (*scriptPtr)++;
768 *fOpStack.push() = (Op) kParen;
769 *fBraceStack.push() = kArrayBrace;
770 SkOperand2::OpType saveType = fReturnType;
771 fReturnType = SkOperand2::kS32;
772 bool success = innerScript(scriptPtr, &scriptValue);
773 fReturnType = saveType;
774 SkASSERT(success);
775 success = convertTo(SkOperand2::kS32, &scriptValue);
776 SkASSERT(success);
777 int index = scriptValue.fOperand.fS32;
778 fValueStack.pop(&scriptValue);
779 if (scriptValue.fType == SkOperand2::kObject) {
780 success = handleUnbox(&scriptValue);
781 SkASSERT(success);
782 SkASSERT(scriptValue.fType == SkOperand2::kArray);
783 }
784 scriptValue.fType = scriptValue.fOperand.fArray->getType();
785 // SkASSERT(index >= 0);
786 if ((unsigned) index >= (unsigned) scriptValue.fOperand.fArray->count()) {
787 fError = kArrayIndexOutOfBounds;
788 return false;
789 }
790 scriptValue.fOperand = scriptValue.fOperand.fArray->begin()[index];
791 scriptValue.fIsConstant = SkScriptValue2::kVariable;
792 fValueStack.push(scriptValue);
793 fOpStack.pop(); // pop paren
794 return success;
795}
796
797bool SkScriptEngine2::handleFunction(const char** scriptPtr) {
798 const char* functionName = fToken;
799 size_t functionNameLen = fTokenLength;
800 fTokenLength = 0;
801 SkTDArray<SkScriptValue2> params;
802 bool success = functionParams(scriptPtr, &params);
803 if (success == false)
804 goto done;
805 {
806 for (SkScriptCallBack** callBack = fCallBackArray.begin(); callBack < fCallBackArray.end(); callBack++) {
807 if ((*callBack)->getType() != SkScriptCallBack::kFunction)
808 continue;
809 SkScriptValue2 callbackResult;
810 success = (*callBack)->getReference(functionName, functionNameLen, &callbackResult);
811 if (success) {
812 callbackResult.fType = (*callBack)->getReturnType(callbackResult.fOperand.fReference, NULL);
813 callbackResult.fIsConstant = SkScriptValue2::kVariable;
814 fValueStack.push(callbackResult);
815 goto done;
816 }
817 }
818 }
819 return false;
820done:
821 fOpStack.pop();
822 return success;
823}
824
825bool SkScriptEngine2::handleMember(const char* field, size_t len, void* object) {
826 bool success = true;
827 for (SkScriptCallBack** callBack = fCallBackArray.begin(); callBack < fCallBackArray.end(); callBack++) {
828 if ((*callBack)->getType() != SkScriptCallBack::kMember)
829 continue;
830 SkScriptValue2 callbackResult;
831 success = (*callBack)->getReference(field, len, &callbackResult);
832 if (success) {
833 if (callbackResult.fType == SkOperand2::kString)
834 track(callbackResult.fOperand.fString);
835 callbackResult.fIsConstant = SkScriptValue2::kVariable;
836 fValueStack.push(callbackResult);
837 goto done;
838 }
839 }
840 return false;
841done:
842 return success;
843}
844
845bool SkScriptEngine2::handleMemberFunction(const char* field, size_t len, void* object,
846 SkTDArray<SkScriptValue2>* params) {
847 bool success = true;
848 for (SkScriptCallBack** callBack = fCallBackArray.begin(); callBack < fCallBackArray.end(); callBack++) {
849 if ((*callBack)->getType() != SkScriptCallBack::kMemberFunction)
850 continue;
851 SkScriptValue2 callbackResult;
852 success = (*callBack)->getReference(field, len, &callbackResult);
853 if (success) {
854 if (callbackResult.fType == SkOperand2::kString)
855 track(callbackResult.fOperand.fString);
856 callbackResult.fIsConstant = SkScriptValue2::kVariable;
857 fValueStack.push(callbackResult);
858 goto done;
859 }
860 }
861 return false;
862done:
863 return success;
864}
865
866bool SkScriptEngine2::handleProperty() {
867 bool success = true;
868 for (SkScriptCallBack** callBack = fCallBackArray.begin(); callBack < fCallBackArray.end(); callBack++) {
869 if ((*callBack)->getType() != SkScriptCallBack::kProperty)
870 continue;
871 SkScriptValue2 callbackResult;
872 success = (*callBack)->getReference(fToken, fTokenLength, &callbackResult);
873 if (success) {
874 if (callbackResult.fType == SkOperand2::kString && callbackResult.fOperand.fString == NULL) {
875 callbackResult.fOperand.fString = new SkString(fToken, fTokenLength);
876 track(callbackResult.fOperand.fString);
877 }
878 callbackResult.fIsConstant = SkScriptValue2::kVariable;
879 fValueStack.push(callbackResult);
880 goto done;
881 }
882 }
883done:
884 fTokenLength = 0;
885 return success;
886}
887
888bool SkScriptEngine2::handleUnbox(SkScriptValue2* scriptValue) {
889 bool success = true;
890 for (SkScriptCallBack** callBack = fCallBackArray.begin(); callBack < fCallBackArray.end(); callBack++) {
891 if ((*callBack)->getType() != SkScriptCallBack::kUnbox)
892 continue;
893 SkScriptCallBackConvert* callBackConvert = (SkScriptCallBackConvert*) *callBack;
894 success = callBackConvert->convert(scriptValue->fType, &scriptValue->fOperand);
895 if (success) {
896 if (scriptValue->fType == SkOperand2::kString)
897 track(scriptValue->fOperand.fString);
898 goto done;
899 }
900 }
901 return false;
902done:
903 return success;
904}
905
906// note that entire expression is treated as if it were enclosed in parens
907// an open paren is always the first thing in the op stack
908
909int SkScriptEngine2::logicalOp(char ch, char nextChar) {
910 int advance = 1;
911 Op op;
912 signed char precedence;
913 switch (ch) {
914 case ')':
915 op = (Op) kParen;
916 break;
917 case ']':
918 op = (Op) kArrayOp;
919 break;
920 case '?':
921 op = (Op) kIf;
922 break;
923 case ':':
924 op = (Op) kElse;
925 break;
926 case '&':
927 if (nextChar != '&')
928 goto noMatch;
929 op = kLogicalAnd;
930 advance = 2;
931 break;
932 case '|':
933 if (nextChar != '|')
934 goto noMatch;
935 op = kLogicalOr;
936 advance = 2;
937 break;
938 default:
939 noMatch:
940 return 0;
941 }
942 precedence = gPrecedence[op];
943 int branchIndex = 0;
944 fBranchPopAllowed = false;
945 do {
946 while (gPrecedence[fOpStack.top() & ~kArtificialOp] < precedence)
947 processOp();
948 Branch& branch = fBranchStack.index(branchIndex++);
949 Op branchOp = branch.fOperator;
950 if (gPrecedence[branchOp] >= precedence)
951 break;
952 addTokenValue(fValueStack.top(), kAccumulator);
953 fValueStack.pop();
954 if (branchOp == kLogicalAnd || branchOp == kLogicalOr) {
955 if (branch.fOperator == kLogicalAnd)
956 branch.prime();
957 addToken(kToBool);
958 } else
959 resolveBranch(branch);
960 if (branch.fDone == Branch::kIsNotDone)
961 branch.prime();
962 } while (true);
963 fBranchPopAllowed = true;
964 while (fBranchStack.top().fDone == Branch::kIsDone)
965 fBranchStack.pop();
966 processLogicalOp(op);
967 return advance;
968}
969
970void SkScriptEngine2::processLogicalOp(Op op) {
971 switch (op) {
972 case kParen:
973 case kArrayOp:
974 SkASSERT(fOpStack.count() > 1 && fOpStack.top() == op); // !!! add error handling
975 if (op == kParen)
976 fOpStack.pop();
977 else {
978 SkScriptValue2 value;
979 fValueStack.pop(&value);
980 SkASSERT(value.fType == SkOperand2::kS32 || value.fType == SkOperand2::kScalar); // !!! add error handling (although, could permit strings eventually)
981 int index = value.fType == SkOperand2::kScalar ? SkScalarFloor(value.fOperand.fScalar) :
982 value.fOperand.fS32;
983 SkScriptValue2 arrayValue;
984 fValueStack.pop(&arrayValue);
985 SkASSERT(arrayValue.fType == SkOperand2::kArray); // !!! add error handling
986 SkOpArray* array = arrayValue.fOperand.fArray;
987 SkOperand2 operand;
988 bool success = array->getIndex(index, &operand);
989 SkASSERT(success); // !!! add error handling
990 SkScriptValue2 resultValue;
991 resultValue.fType = array->getType();
992 resultValue.fOperand = operand;
993 resultValue.fIsConstant = SkScriptValue2::kVariable;
994 fValueStack.push(resultValue);
995 }
996 break;
997 case kIf: {
998 if (fAccumulatorType == SkOperand2::kNoType) {
999 addTokenValue(fValueStack.top(), kAccumulator);
1000 fValueStack.pop();
1001 }
1002 SkASSERT(fAccumulatorType != SkOperand2::kString); // !!! add error handling
1003 addToken(kIfOp);
1004 Branch branch(op, fOpStack.count(), getTokenOffset());
1005 *fBranchStack.push() = branch;
1006 addTokenInt(0); // placeholder for future branch
1007 fAccumulatorType = SkOperand2::kNoType;
1008 } break;
1009 case kElse: {
1010 addTokenValue(fValueStack.top(), kAccumulator);
1011 fValueStack.pop();
1012 addToken(kElseOp);
1013 size_t newOffset = getTokenOffset();
1014 addTokenInt(0); // placeholder for future branch
1015 Branch& branch = fBranchStack.top();
1016 resolveBranch(branch);
1017 branch.fOperator = op;
1018 branch.fDone = Branch::kIsNotDone;
1019 SkASSERT(branch.fOpStackDepth == fOpStack.count());
1020 branch.fOffset = newOffset;
1021 fAccumulatorType = SkOperand2::kNoType;
1022 } break;
1023 case kLogicalAnd:
1024 case kLogicalOr: {
1025 Branch& oldTop = fBranchStack.top();
1026 Branch::Primed wasPrime = oldTop.fPrimed;
1027 Branch::Done wasDone = oldTop.fDone;
1028 oldTop.fPrimed = Branch::kIsNotPrimed;
1029 oldTop.fDone = Branch::kIsNotDone;
1030 if (fAccumulatorType == SkOperand2::kNoType) {
1031 SkASSERT(fValueStack.top().fType == SkOperand2::kS32); // !!! add error handling, and conversion to int?
1032 addTokenValue(fValueStack.top(), kAccumulator);
1033 fValueStack.pop();
1034 } else
1035 SkASSERT(fAccumulatorType == SkOperand2::kS32);
1036 // if 'and', write beq goto opcode after end of predicate (after to bool)
1037 // if 'or', write bne goto to bool
1038 addToken(op == kLogicalAnd ? kLogicalAndInt : kLogicalOrInt);
1039 Branch branch(op, fOpStack.count(), getTokenOffset());
1040 addTokenInt(0); // placeholder for future branch
1041 oldTop.fPrimed = wasPrime;
1042 oldTop.fDone = wasDone;
1043 *fBranchStack.push() = branch;
1044 fAccumulatorType = SkOperand2::kNoType;
1045 } break;
1046 default:
1047 SkASSERT(0);
1048 }
1049}
1050
1051bool SkScriptEngine2::processOp() {
1052 Op op;
1053 fOpStack.pop(&op);
1054 op = (Op) (op & ~kArtificialOp);
1055 const OperatorAttributes* attributes = &gOpAttributes[op];
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001056 SkScriptValue2 value1;
1057 memset(&value1, 0, sizeof(SkScriptValue2));
reed@android.com8a1c16f2008-12-17 15:59:43 +00001058 SkScriptValue2 value2;
1059 fValueStack.pop(&value2);
1060 value2.fIsWritten = SkScriptValue2::kUnwritten;
1061 // SkScriptEngine2::SkTypeOp convert1[3];
1062 // SkScriptEngine2::SkTypeOp convert2[3];
1063 // SkScriptEngine2::SkTypeOp* convert2Ptr = convert2;
1064 bool constantOperands = value2.fIsConstant == SkScriptValue2::kConstant;
1065 if (attributes->fLeftType != SkOperand2::kNoType) {
1066 fValueStack.pop(&value1);
1067 constantOperands &= value1.fIsConstant == SkScriptValue2::kConstant;
1068 value1.fIsWritten = SkScriptValue2::kUnwritten;
1069 if (op == kFlipOps) {
1070 SkTSwap(value1, value2);
1071 fOpStack.pop(&op);
1072 op = (Op) (op & ~kArtificialOp);
1073 attributes = &gOpAttributes[op];
1074 if (constantOperands == false)
1075 addToken(kFlipOpsOp);
1076 }
1077 if (value1.fType == SkOperand2::kObject && (value1.fType & attributes->fLeftType) == 0) {
1078 value1.fType = getUnboxType(value1.fOperand);
1079 addToken(kUnboxToken);
1080 }
1081 }
1082 if (value2.fType == SkOperand2::kObject && (value2.fType & attributes->fLeftType) == 0) {
1083 value1.fType = getUnboxType(value2.fOperand);
1084 addToken(kUnboxToken2);
1085 }
1086 if (attributes->fLeftType != SkOperand2::kNoType) {
1087 if (value1.fType != value2.fType) {
1088 if ((attributes->fLeftType & SkOperand2::kString) && attributes->fBias & kTowardsString &&
1089 ((value1.fType | value2.fType) & SkOperand2::kString)) {
1090 if (value1.fType == SkOperand2::kS32 || value1.fType == SkOperand2::kScalar) {
1091 addTokenConst(&value1, kAccumulator, SkOperand2::kString,
1092 value1.fType == SkOperand2::kS32 ? kIntToString : kScalarToString);
1093 }
1094 if (value2.fType == SkOperand2::kS32 || value2.fType == SkOperand2::kScalar) {
1095 addTokenConst(&value2, kOperand, SkOperand2::kString,
1096 value2.fType == SkOperand2::kS32 ? kIntToString2 : kScalarToString2);
1097 }
1098 } else if (attributes->fLeftType & SkOperand2::kScalar && ((value1.fType | value2.fType) &
1099 SkOperand2::kScalar)) {
1100 if (value1.fType == SkOperand2::kS32)
1101 addTokenConst(&value1, kAccumulator, SkOperand2::kScalar, kIntToScalar);
1102 if (value2.fType == SkOperand2::kS32)
1103 addTokenConst(&value2, kOperand, SkOperand2::kScalar, kIntToScalar2);
1104 }
1105 }
1106 if ((value1.fType & attributes->fLeftType) == 0 || value1.fType != value2.fType) {
1107 if (value1.fType == SkOperand2::kString)
1108 addTokenConst(&value1, kAccumulator, SkOperand2::kScalar, kStringToScalar);
1109 if (value1.fType == SkOperand2::kScalar && (attributes->fLeftType == SkOperand2::kS32 ||
1110 value2.fType == SkOperand2::kS32))
1111 addTokenConst(&value1, kAccumulator, SkOperand2::kS32, kScalarToInt);
1112 }
1113 }
1114 AddTokenRegister rhRegister = attributes->fLeftType != SkOperand2::kNoType ?
1115 kOperand : kAccumulator;
1116 if ((value2.fType & attributes->fRightType) == 0 || value1.fType != value2.fType) {
1117 if (value2.fType == SkOperand2::kString)
1118 addTokenConst(&value2, rhRegister, SkOperand2::kScalar, kStringToScalar2);
1119 if (value2.fType == SkOperand2::kScalar && (attributes->fRightType == SkOperand2::kS32 ||
1120 value1.fType == SkOperand2::kS32))
1121 addTokenConst(&value2, rhRegister, SkOperand2::kS32, kScalarToInt2);
1122 }
1123 TypeOp typeOp = gTokens[op];
1124 if (value2.fType == SkOperand2::kScalar)
1125 typeOp = (TypeOp) (typeOp + 1);
1126 else if (value2.fType == SkOperand2::kString)
1127 typeOp = (TypeOp) (typeOp + 2);
1128 SkDynamicMemoryWStream stream;
1129 SkOperand2::OpType saveType;
1130 SkBool saveOperand;
1131 if (constantOperands) {
1132 fActiveStream = &stream;
1133 saveType = fAccumulatorType;
1134 saveOperand = fOperandInUse;
1135 fAccumulatorType = SkOperand2::kNoType;
1136 fOperandInUse = false;
1137 }
1138 if (attributes->fLeftType != SkOperand2::kNoType) { // two operands
1139 if (value1.fIsWritten == SkScriptValue2::kUnwritten)
1140 addTokenValue(value1, kAccumulator);
1141 }
1142 if (value2.fIsWritten == SkScriptValue2::kUnwritten)
1143 addTokenValue(value2, rhRegister);
1144 addToken(typeOp);
1145 if (constantOperands) {
1146 addToken(kEnd);
reed@google.com8a85d0c2011-06-24 19:12:12 +00001147 SkAutoDataUnref data(fStream.copyToData());
reed@android.com8a1c16f2008-12-17 15:59:43 +00001148#ifdef SK_DEBUG
reed@google.com8a85d0c2011-06-24 19:12:12 +00001149 decompile(data.bytes(), data.size());
reed@android.com8a1c16f2008-12-17 15:59:43 +00001150#endif
1151 SkScriptRuntime runtime(fCallBackArray);
reed@google.com8a85d0c2011-06-24 19:12:12 +00001152 runtime.executeTokens((unsigned char*)data.bytes());
reed@android.com8a1c16f2008-12-17 15:59:43 +00001153 runtime.getResult(&value1.fOperand);
1154 if (attributes->fResultIsBoolean == kResultIsBoolean)
1155 value1.fType = SkOperand2::kS32;
1156 else if (attributes->fLeftType == SkOperand2::kNoType) // unary operand
1157 value1.fType = value2.fType;
1158 fValueStack.push(value1);
1159 if (value1.fType == SkOperand2::kString)
1160 runtime.untrack(value1.fOperand.fString);
1161 else if (value1.fType == SkOperand2::kArray)
1162 runtime.untrack(value1.fOperand.fArray);
1163 fActiveStream = &fStream;
1164 fAccumulatorType = saveType;
1165 fOperandInUse = saveOperand;
1166 return true;
1167 }
1168 value2.fIsConstant = SkScriptValue2::kVariable;
1169 fValueStack.push(value2);
1170 return true;
1171}
1172
1173void SkScriptEngine2::Branch::resolve(SkDynamicMemoryWStream* stream, size_t off) {
1174 SkASSERT(fDone == kIsNotDone);
1175 fPrimed = kIsNotPrimed;
1176 fDone = kIsDone;
1177 SkASSERT(off > fOffset + sizeof(size_t));
1178 size_t offset = off - fOffset - sizeof(offset);
1179 stream->write(&offset, fOffset, sizeof(offset));
1180}
1181
1182void SkScriptEngine2::resolveBranch(SkScriptEngine2::Branch& branch) {
1183 branch.resolve(fActiveStream, getTokenOffset());
1184}
1185
1186bool SkScriptEngine2::ConvertTo(SkScriptEngine2* engine, SkOperand2::OpType toType, SkScriptValue2* value ) {
1187 SkASSERT(value);
1188 SkOperand2::OpType type = value->fType;
1189 if (type == toType)
1190 return true;
1191 SkOperand2& operand = value->fOperand;
1192 bool success = true;
1193 switch (toType) {
1194 case SkOperand2::kS32:
1195 if (type == SkOperand2::kScalar)
1196 operand.fS32 = SkScalarFloor(operand.fScalar);
1197 else {
1198 SkASSERT(type == SkOperand2::kString);
1199 success = SkParse::FindS32(operand.fString->c_str(), &operand.fS32) != NULL;
1200 }
1201 break;
1202 case SkOperand2::kScalar:
1203 if (type == SkOperand2::kS32)
1204 operand.fScalar = IntToScalar(operand.fS32);
1205 else {
1206 SkASSERT(type == SkOperand2::kString);
1207 success = SkParse::FindScalar(operand.fString->c_str(), &operand.fScalar) != NULL;
1208 }
1209 break;
1210 case SkOperand2::kString: {
1211 SkString* strPtr = new SkString();
1212 SkASSERT(engine);
1213 engine->track(strPtr);
1214 if (type == SkOperand2::kS32)
1215 strPtr->appendS32(operand.fS32);
1216 else {
1217 SkASSERT(type == SkOperand2::kScalar);
1218 strPtr->appendScalar(operand.fScalar);
1219 }
1220 operand.fString = strPtr;
1221 } break;
1222 case SkOperand2::kArray: {
1223 SkOpArray* array = new SkOpArray(type);
1224 *array->append() = operand;
1225 engine->track(array);
1226 operand.fArray = array;
1227 } break;
1228 default:
1229 SkASSERT(0);
1230 }
1231 value->fType = toType;
1232 return success;
1233}
1234
1235SkScalar SkScriptEngine2::IntToScalar(int32_t s32) {
1236 SkScalar scalar;
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001237 if (s32 == (int32_t) SK_NaN32)
reed@android.com8a1c16f2008-12-17 15:59:43 +00001238 scalar = SK_ScalarNaN;
1239 else if (SkAbs32(s32) == SK_MaxS32)
1240 scalar = SkSign32(s32) * SK_ScalarMax;
1241 else
1242 scalar = SkIntToScalar(s32);
1243 return scalar;
1244}
1245
1246bool SkScriptEngine2::ValueToString(const SkScriptValue2& value, SkString* string) {
1247 switch (value.fType) {
1248 case SkOperand2::kS32:
1249 string->reset();
1250 string->appendS32(value.fOperand.fS32);
1251 break;
1252 case SkOperand2::kScalar:
1253 string->reset();
1254 string->appendScalar(value.fOperand.fScalar);
1255 break;
1256 case SkOperand2::kString:
1257 string->set(*value.fOperand.fString);
1258 break;
1259 default:
1260 SkASSERT(0);
1261 return false;
1262 }
1263 return true; // no error
1264}
1265
1266#ifdef SK_DEBUG
1267
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001268#define testInt(expression) { #expression, SkOperand2::kS32, expression, 0, NULL }
reed@android.com8a1c16f2008-12-17 15:59:43 +00001269#ifdef SK_SCALAR_IS_FLOAT
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001270#define testScalar(expression) { #expression, SkOperand2::kScalar, 0, (float) expression, NULL }
1271#define testRemainder(exp1, exp2) { #exp1 "%" #exp2, SkOperand2::kScalar, 0, fmodf(exp1, exp2), NULL }
reed@android.com8a1c16f2008-12-17 15:59:43 +00001272#else
1273#ifdef SK_CAN_USE_FLOAT
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001274#define testScalar(expression) { #expression, SkOperand2::kScalar, 0, (int) ((expression) * 65536.0f), NULL }
1275#define testRemainder(exp1, exp2) { #exp1 "%" #exp2, SkOperand2::kScalar, 0, (int) (fmod(exp1, exp2) * 65536.0f), NULL }
reed@android.com8a1c16f2008-12-17 15:59:43 +00001276#endif
1277#endif
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001278#define testTrue(expression) { #expression, SkOperand2::kS32, 1, 0, NULL }
1279#define testFalse(expression) { #expression, SkOperand2::kS32, 0, 0, NULL }
reed@android.com8a1c16f2008-12-17 15:59:43 +00001280
reed@android.com8a1c16f2008-12-17 15:59:43 +00001281static const SkScriptNAnswer2 scriptTests[] = {
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001282 testInt(1||(0&&3)),
reed@android.com8a1c16f2008-12-17 15:59:43 +00001283#ifdef SK_CAN_USE_FLOAT
1284 testScalar(- -5.5- -1.5),
1285 testScalar(1.0+5),
1286#endif
1287 testInt((6+7)*8),
1288 testInt(3*(4+5)),
1289#ifdef SK_CAN_USE_FLOAT
1290 testScalar(1.0+2.0),
1291 testScalar(3.0-1.0),
1292 testScalar(6-1.0),
1293 testScalar(2.5*6.),
1294 testScalar(0.5*4),
1295 testScalar(4.5/.5),
1296 testScalar(9.5/19),
1297 testRemainder(9.5, 0.5),
1298 testRemainder(9.,2),
1299 testRemainder(9,2.5),
1300 testRemainder(-9,2.5),
1301 testTrue(-9==-9.0),
1302 testTrue(-9.==-4.0-5),
1303 testTrue(-9.*1==-4-5),
1304 testFalse(-9!=-9.0),
1305 testFalse(-9.!=-4.0-5),
1306 testFalse(-9.*1!=-4-5),
1307#endif
1308 testInt(0x123),
1309 testInt(0XABC),
1310 testInt(0xdeadBEEF),
1311 { "'123'+\"456\"", SkOperand2::kString, 0, 0, "123456" },
1312 { "123+\"456\"", SkOperand2::kString, 0, 0, "123456" },
1313 { "'123'+456", SkOperand2::kString, 0, 0, "123456" },
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001314 { "'123'|\"456\"", SkOperand2::kS32, 123|456, 0, NULL },
1315 { "123|\"456\"", SkOperand2::kS32, 123|456, 0, NULL },
1316 { "'123'|456", SkOperand2::kS32, 123|456, 0, NULL },
1317 { "'2'<11", SkOperand2::kS32, 1, 0, NULL },
1318 { "2<'11'", SkOperand2::kS32, 1, 0, NULL },
1319 { "'2'<'11'", SkOperand2::kS32, 0, 0, NULL },
reed@android.com8a1c16f2008-12-17 15:59:43 +00001320 testInt(123),
1321 testInt(-345),
1322 testInt(+678),
1323 testInt(1+2+3),
1324 testInt(3*4+5),
1325 testInt(6+7*8),
1326 testInt(-1-2-8/4),
1327 testInt(-9%4),
1328 testInt(9%-4),
1329 testInt(-9%-4),
1330 testInt(123|978),
1331 testInt(123&978),
1332 testInt(123^978),
1333 testInt(2<<4),
1334 testInt(99>>3),
1335 testInt(~55),
1336 testInt(~~55),
1337 testInt(!55),
1338 testInt(!!55),
1339 // both int
1340 testInt(2<2),
1341 testInt(2<11),
1342 testInt(20<11),
1343 testInt(2<=2),
1344 testInt(2<=11),
1345 testInt(20<=11),
1346 testInt(2>2),
1347 testInt(2>11),
1348 testInt(20>11),
1349 testInt(2>=2),
1350 testInt(2>=11),
1351 testInt(20>=11),
1352 testInt(2==2),
1353 testInt(2==11),
1354 testInt(20==11),
1355 testInt(2!=2),
1356 testInt(2!=11),
1357 testInt(20!=11),
1358#ifdef SK_CAN_USE_FLOAT
1359 // left int, right scalar
1360 testInt(2<2.),
1361 testInt(2<11.),
1362 testInt(20<11.),
1363 testInt(2<=2.),
1364 testInt(2<=11.),
1365 testInt(20<=11.),
1366 testInt(2>2.),
1367 testInt(2>11.),
1368 testInt(20>11.),
1369 testInt(2>=2.),
1370 testInt(2>=11.),
1371 testInt(20>=11.),
1372 testInt(2==2.),
1373 testInt(2==11.),
1374 testInt(20==11.),
1375 testInt(2!=2.),
1376 testInt(2!=11.),
1377 testInt(20!=11.),
1378 // left scalar, right int
1379 testInt(2.<2),
1380 testInt(2.<11),
1381 testInt(20.<11),
1382 testInt(2.<=2),
1383 testInt(2.<=11),
1384 testInt(20.<=11),
1385 testInt(2.>2),
1386 testInt(2.>11),
1387 testInt(20.>11),
1388 testInt(2.>=2),
1389 testInt(2.>=11),
1390 testInt(20.>=11),
1391 testInt(2.==2),
1392 testInt(2.==11),
1393 testInt(20.==11),
1394 testInt(2.!=2),
1395 testInt(2.!=11),
1396 testInt(20.!=11),
1397 // both scalar
1398 testInt(2.<11.),
1399 testInt(20.<11.),
1400 testInt(2.<=2.),
1401 testInt(2.<=11.),
1402 testInt(20.<=11.),
1403 testInt(2.>2.),
1404 testInt(2.>11.),
1405 testInt(20.>11.),
1406 testInt(2.>=2.),
1407 testInt(2.>=11.),
1408 testInt(20.>=11.),
1409 testInt(2.==2.),
1410 testInt(2.==11.),
1411 testInt(20.==11.),
1412 testInt(2.!=2.),
1413 testInt(2.!=11.),
1414 testInt(20.!=11.),
1415#endif
1416 // int, string (string is int)
1417 testFalse(2<'2'),
1418 testTrue(2<'11'),
1419 testFalse(20<'11'),
1420 testTrue(2<='2'),
1421 testTrue(2<='11'),
1422 testFalse(20<='11'),
1423 testFalse(2>'2'),
1424 testFalse(2>'11'),
1425 testTrue(20>'11'),
1426 testTrue(2>='2'),
1427 testFalse(2>='11'),
1428 testTrue(20>='11'),
1429 testTrue(2=='2'),
1430 testFalse(2=='11'),
1431 testFalse(2!='2'),
1432 testTrue(2!='11'),
1433 // int, string (string is scalar)
1434 testFalse(2<'2.'),
1435 testTrue(2<'11.'),
1436 testFalse(20<'11.'),
1437 testTrue(2=='2.'),
1438 testFalse(2=='11.'),
1439#ifdef SK_CAN_USE_FLOAT
1440 // scalar, string
1441 testFalse(2.<'2.'),
1442 testTrue(2.<'11.'),
1443 testFalse(20.<'11.'),
1444 testTrue(2.=='2.'),
1445 testFalse(2.=='11.'),
1446 // string, int
1447 testFalse('2'<2),
1448 testTrue('2'<11),
1449 testFalse('20'<11),
1450 testTrue('2'==2),
1451 testFalse('2'==11),
1452 // string, scalar
1453 testFalse('2'<2.),
1454 testTrue('2'<11.),
1455 testFalse('20'<11.),
1456 testTrue('2'==2.),
1457 testFalse('2'==11.),
1458#endif
1459 // string, string
1460 testFalse('2'<'2'),
1461 testFalse('2'<'11'),
1462 testFalse('20'<'11'),
1463 testTrue('2'=='2'),
1464 testFalse('2'=='11'),
1465 // logic
1466 testInt(1?2:3),
1467 testInt(0?2:3),
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001468 testInt((1&&2)||3),
1469 testInt((1&&0)||3),
1470 testInt((1&&0)||0),
1471 testInt(1||(0&&3)),
1472 testInt(0||(0&&3)),
1473 testInt(0||(1&&3)),
reed@android.com8a1c16f2008-12-17 15:59:43 +00001474 testInt(0&&1?2:3)
1475#ifdef SK_CAN_USE_FLOAT
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +00001476 , { "123.5", SkOperand2::kScalar, 0, SkIntToScalar(123) + SK_Scalar1/2, NULL }
reed@android.com8a1c16f2008-12-17 15:59:43 +00001477#endif
1478};
reed@android.com8a1c16f2008-12-17 15:59:43 +00001479
1480#define SkScriptNAnswer_testCount SK_ARRAY_COUNT(scriptTests)
1481
1482void SkScriptEngine2::UnitTest() {
reed@android.comec10d212010-01-12 22:58:35 +00001483#if defined(SK_SUPPORT_UNITTEST)
reed@android.com8a1c16f2008-12-17 15:59:43 +00001484 ValidateDecompileTable();
1485 for (int index = 0; index < SkScriptNAnswer_testCount; index++) {
1486 SkScriptEngine2 engine(scriptTests[index].fType);
1487 SkScriptValue2 value;
1488 const char* script = scriptTests[index].fScript;
1489 const char* scriptPtr = script;
1490 SkASSERT(engine.evaluateScript(&scriptPtr, &value) == true);
1491 SkASSERT(value.fType == scriptTests[index].fType);
1492 SkScalar error;
1493 switch (value.fType) {
1494 case SkOperand2::kS32:
1495 if (value.fOperand.fS32 != scriptTests[index].fIntAnswer)
1496 SkDEBUGF(("script '%s' == value %d != expected answer %d\n", script, value.fOperand.fS32, scriptTests[index].fIntAnswer));
1497 SkASSERT(value.fOperand.fS32 == scriptTests[index].fIntAnswer);
1498 break;
1499 case SkOperand2::kScalar:
1500 error = SkScalarAbs(value.fOperand.fScalar - scriptTests[index].fScalarAnswer);
1501#ifdef SK_CAN_USE_FLOAT
1502 if (error >= SK_Scalar1 / 10000)
1503 SkDEBUGF(("script '%s' == value %g != expected answer %g\n", script, value.fOperand.fScalar / (1.0f * SK_Scalar1), scriptTests[index].fScalarAnswer / (1.0f * SK_Scalar1)));
1504#endif
1505 SkASSERT(error < SK_Scalar1 / 10000);
1506 break;
1507 case SkOperand2::kString:
1508 SkASSERT(value.fOperand.fString->equals(scriptTests[index].fStringAnswer));
1509 break;
1510 default:
1511 SkASSERT(0);
1512 }
1513 }
1514#endif
1515}
1516#endif