Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file | |
| 2 // for details. All rights reserved. Use of this source code is governed by a | |
| 3 // BSD-style license that can be found in the LICENSE file. | |
| 4 | |
| 5 #include "vm/ast_transformer.h" | |
| 6 | |
| 7 #include "vm/parser.h" | |
| 8 | |
| 9 namespace dart { | |
| 10 | |
| 11 // Quick access to the locally defined isolate() method. | |
| 12 #define I (isolate()) | |
| 13 | |
| 14 // Nodes that are unreachable from already parsed expressions. | |
| 15 #define FOR_EACH_UNREACHABLE_NODE(V) \ | |
| 16 V(Case) \ | |
| 17 V(CatchClause) \ | |
| 18 V(CloneContext) \ | |
| 19 V(ClosureCall) \ | |
| 20 V(DoWhile) \ | |
| 21 V(If) \ | |
| 22 V(InlinedFinally) \ | |
| 23 V(For) \ | |
| 24 V(Jump) \ | |
| 25 V(LoadInstanceField) \ | |
| 26 V(NativeBody) \ | |
| 27 V(Primary) \ | |
| 28 V(Return) \ | |
| 29 V(Sequence) \ | |
| 30 V(StoreInstanceField) \ | |
| 31 V(Switch) \ | |
| 32 V(TryCatch) \ | |
| 33 V(While) | |
| 34 | |
| 35 #define DEFINE_UNREACHABLE(BaseName) \ | |
| 36 void AwaitTransformer::Visit##BaseName##Node(BaseName##Node* node) { \ | |
| 37 UNREACHABLE(); \ | |
| 38 } | |
| 39 | |
| 40 FOR_EACH_UNREACHABLE_NODE(DEFINE_UNREACHABLE) | |
| 41 #undef DEFINE_UNREACHABLE | |
| 42 | |
| 43 | |
| 44 void AwaitTransformer::Transform(AstNode* expr) { | |
|
hausner
2014/08/15 18:09:31
Why not return the result_ field as a function res
Michael Lippautz (Google)
2014/08/18 18:39:30
Done.
I also adjusted the parser to take this ret
| |
| 45 expr->Visit(this); | |
| 46 } | |
| 47 | |
| 48 | |
| 49 LocalVariable* AwaitTransformer::EnsureCurrentTempVar() { | |
| 50 const char* await_temp_prefix = ":await_temp_var_"; | |
| 51 const String& cnt_str = String::ZoneHandle(I, | |
| 52 String::NewFormatted( | |
| 53 "%s%" Pd "", await_temp_prefix, temp_cnt_)); | |
| 54 const String& symbol = String::ZoneHandle(I, Symbols::New(cnt_str)); | |
| 55 ASSERT(!symbol.IsNull()); | |
| 56 LocalVariable* await_tmp = | |
| 57 parsed_function_->await_temps_scope()->LookupVariable(symbol, false); | |
| 58 if (await_tmp == NULL) { | |
| 59 await_tmp = new(I) LocalVariable( | |
| 60 Scanner::kNoSourcePos, | |
| 61 symbol, | |
| 62 Type::ZoneHandle(I, Type::DynamicType())); | |
| 63 parsed_function_->await_temps_scope()->AddVariable(await_tmp); | |
| 64 } | |
| 65 return await_tmp; | |
| 66 } | |
| 67 | |
| 68 | |
| 69 LocalVariable* AwaitTransformer::AddToPreambleNewTempVar(AstNode* node) { | |
| 70 LocalVariable* tmp_var = EnsureCurrentTempVar(); | |
| 71 preamble_->Add(new(I) StoreLocalNode(Scanner::kNoSourcePos, tmp_var, node)); | |
| 72 NextTempVar(); | |
| 73 return tmp_var; | |
| 74 } | |
| 75 | |
| 76 | |
| 77 void AwaitTransformer::VisitLiteralNode(LiteralNode* node) { | |
| 78 result_ = node; | |
| 79 } | |
| 80 | |
| 81 | |
| 82 void AwaitTransformer::VisitTypeNode(TypeNode* node) { | |
| 83 result_ = new(I) TypeNode(node->token_pos(), node->type()); | |
| 84 } | |
| 85 | |
| 86 | |
| 87 | |
| 88 void AwaitTransformer::VisitAwaitNode(AwaitNode* node) { | |
| 89 // Await transformation: | |
| 90 // | |
| 91 // :await_temp_var_X = <expr>; | |
| 92 // :result_param = :await_temp_var_X; | |
| 93 // if (:result_param is Future) { | |
| 94 // // :result_param.then(:async_op); | |
| 95 // } | |
| 96 // :await_temp_var_(X+1) = :result_param; | |
| 97 | |
| 98 LocalVariable* async_op = preamble_->scope()->LookupVariable( | |
| 99 Symbols::AsyncOperation(), false); | |
| 100 ASSERT(async_op != NULL); | |
| 101 LocalVariable* result_param = preamble_->scope()->LookupVariable( | |
| 102 Symbols::AsyncOperationParam(), false); | |
| 103 ASSERT(result_param != NULL); | |
| 104 | |
| 105 node->expr()->Visit(this); | |
| 106 preamble_->Add(new(I) StoreLocalNode(Scanner::kNoSourcePos, | |
| 107 result_param, | |
| 108 result_)); | |
| 109 LoadLocalNode* load_result_param = new(I) LoadLocalNode( | |
| 110 Scanner::kNoSourcePos, result_param); | |
| 111 SequenceNode* is_future_branch = new(I) SequenceNode( | |
| 112 Scanner::kNoSourcePos, preamble_->scope()); | |
| 113 ArgumentListNode* args = new(I) ArgumentListNode(Scanner::kNoSourcePos); | |
| 114 args->Add(new(I) LoadLocalNode(Scanner::kNoSourcePos, async_op)); | |
| 115 // TODO(mlippautz): Once continuations are supported, just call .then(). | |
| 116 // is_future_branch->Add(new(I) InstanceCallNode( | |
| 117 // Scanner::kNoSourcePos, load_result_param, Symbols::FutureThen(), args)); | |
| 118 // | |
| 119 // For now, throw an exception. | |
| 120 const String& exception = String::ZoneHandle( | |
| 121 I, String::New("awaitable futures not yet supported", Heap::kOld)); | |
| 122 is_future_branch->Add(new(I) ThrowNode( | |
| 123 Scanner::kNoSourcePos, | |
| 124 new(I) LiteralNode( | |
| 125 Scanner::kNoSourcePos, | |
| 126 String::ZoneHandle(I, Symbols::New(exception))), | |
| 127 NULL)); | |
| 128 const Class& cls = Class::ZoneHandle( | |
| 129 I, library_.LookupClass(Symbols::Future())); | |
| 130 const AbstractType& future_type = AbstractType::ZoneHandle(I, | |
| 131 cls.RareType()); | |
| 132 ASSERT(!future_type.IsNull()); | |
| 133 TypeNode* future_type_node = new(I) TypeNode( | |
| 134 Scanner::kNoSourcePos, future_type); | |
| 135 IfNode* is_future_if = new(I) IfNode( | |
| 136 Scanner::kNoSourcePos, | |
| 137 new(I) ComparisonNode(Scanner::kNoSourcePos, | |
| 138 Token::kIS, | |
| 139 load_result_param, | |
| 140 future_type_node), | |
| 141 is_future_branch, | |
| 142 NULL); | |
| 143 preamble_->Add(is_future_if); | |
| 144 | |
| 145 // TODO(mlippautz): Join for await needs to happen here. | |
| 146 | |
| 147 LocalVariable* result = AddToPreambleNewTempVar(new(I) LoadLocalNode( | |
| 148 Scanner::kNoSourcePos, result_param)); | |
| 149 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 150 } | |
| 151 | |
| 152 | |
| 153 // Transforms boolean expressions into a sequence of evaluatons that only lazily | |
| 154 // evaluate subexpressions. | |
| 155 // | |
| 156 // Example: | |
| 157 // | |
| 158 // (a || b) only evaluates b if a is true | |
|
hausner
2014/08/15 18:09:31
only evaluates b if a is *false*
Michael Lippautz (Google)
2014/08/18 18:39:30
Done.
| |
| 159 // | |
| 160 // Transformation (roughly): | |
| 161 // | |
| 162 // t_1 = a; | |
| 163 // if (t_1) { | |
|
hausner
2014/08/15 18:09:31
!t_1
Michael Lippautz (Google)
2014/08/18 18:39:30
Done.
| |
| 164 // t_2 = b; | |
| 165 // } | |
| 166 // t_3 = t_1 || t_2; // Compiler takes care that lazy evaluation takes place | |
| 167 // on this level. | |
| 168 AstNode* AwaitTransformer::LazyTransform(const Token::Kind logical_op, | |
| 169 AstNode* new_left, | |
| 170 AstNode* right) { | |
| 171 ASSERT(logical_op == Token::kAND || logical_op == Token::kOR); | |
| 172 AstNode* result = NULL; | |
| 173 const Token::Kind compare_logical_op = (logical_op == Token::kAND) ? | |
| 174 Token::kEQ : Token::kNE; | |
| 175 SequenceNode* eval = new(I) SequenceNode( | |
| 176 Scanner::kNoSourcePos, preamble_->scope()); | |
| 177 SequenceNode* saved_preamble = preamble_; | |
| 178 preamble_ = eval; | |
| 179 right->Visit(this); | |
| 180 result = result_; | |
| 181 preamble_ = saved_preamble; | |
| 182 IfNode* right_body = new(I) IfNode( | |
| 183 Scanner::kNoSourcePos, | |
| 184 new(I) ComparisonNode( | |
| 185 Scanner::kNoSourcePos, | |
| 186 compare_logical_op, | |
| 187 new_left, | |
| 188 new(I) LiteralNode(Scanner::kNoSourcePos, Bool::True())), | |
| 189 eval, | |
| 190 NULL); | |
| 191 preamble_->Add(right_body); | |
| 192 return result; | |
| 193 } | |
| 194 | |
| 195 | |
| 196 void AwaitTransformer::VisitBinaryOpNode(BinaryOpNode* node) { | |
| 197 node->left()->Visit(this); | |
| 198 AstNode* new_left = result_; | |
| 199 AstNode* new_right = NULL; | |
| 200 // Preserve lazy evaluaton. | |
| 201 if ((node->kind() == Token::kAND) || (node->kind() == Token::kOR)) { | |
| 202 new_right = LazyTransform(node->kind(), new_left, node->right()); | |
| 203 } else { | |
| 204 node->right()->Visit(this); | |
| 205 new_right = result_; | |
|
hausner
2014/08/15 18:09:31
Would it not make sense to group these two stateme
Michael Lippautz (Google)
2014/08/18 18:39:30
Done.
| |
| 206 } | |
| 207 LocalVariable* result = AddToPreambleNewTempVar( | |
| 208 new(I) BinaryOpNode(node->token_pos(), | |
| 209 node->kind(), | |
| 210 new_left, | |
| 211 new_right)); | |
| 212 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 213 } | |
| 214 | |
| 215 | |
| 216 void AwaitTransformer::VisitBinaryOpWithMask32Node( | |
| 217 BinaryOpWithMask32Node* node) { | |
| 218 node->left()->Visit(this); | |
| 219 AstNode* new_left = result_; | |
| 220 AstNode* new_right = NULL; | |
| 221 // Preserve lazy evaluaton. | |
| 222 if ((node->kind() == Token::kAND) || (node->kind() == Token::kOR)) { | |
| 223 new_right = LazyTransform(node->kind(), new_left, node->right()); | |
| 224 } else { | |
| 225 node->right()->Visit(this); | |
| 226 new_right = result_; | |
| 227 } | |
| 228 LocalVariable* result = AddToPreambleNewTempVar( | |
| 229 new(I) BinaryOpWithMask32Node(node->token_pos(), | |
| 230 node->kind(), | |
| 231 new_left, | |
| 232 new_right, | |
| 233 node->mask32())); | |
| 234 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 235 } | |
| 236 | |
| 237 | |
| 238 void AwaitTransformer::VisitComparisonNode(ComparisonNode* node) { | |
| 239 node->left()->Visit(this); | |
| 240 AstNode* new_left = result_; | |
| 241 node->right()->Visit(this); | |
| 242 AstNode* new_right = result_; | |
| 243 LocalVariable* result = AddToPreambleNewTempVar( | |
| 244 new(I) ComparisonNode(node->token_pos(), | |
| 245 node->kind(), | |
| 246 new_left, | |
| 247 new_right)); | |
| 248 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 249 } | |
| 250 | |
| 251 | |
| 252 void AwaitTransformer::VisitUnaryOpNode(UnaryOpNode* node) { | |
| 253 node->operand()->Visit(this); | |
| 254 AstNode* new_operand = result_; | |
| 255 | |
| 256 LocalVariable* result = AddToPreambleNewTempVar( | |
| 257 new(I) UnaryOpNode(node->token_pos(), | |
| 258 node->kind(), | |
| 259 new_operand)); | |
| 260 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 261 } | |
| 262 | |
| 263 | |
| 264 // ::= (<condition>) ? <true-branch> : <false-branch> | |
| 265 // | |
| 266 void AwaitTransformer::VisitConditionalExprNode(ConditionalExprNode* node) { | |
| 267 node->condition()->Visit(this); | |
| 268 AstNode* new_condition = result_; | |
| 269 SequenceNode* new_true = new(I) SequenceNode( | |
| 270 Scanner::kNoSourcePos, preamble_->scope()); | |
| 271 SequenceNode* saved_preamble = preamble_; | |
| 272 preamble_ = new_true; | |
| 273 node->true_expr()->Visit(this); | |
| 274 AstNode* new_true_result = result_; | |
| 275 SequenceNode* new_false = new(I) SequenceNode( | |
| 276 Scanner::kNoSourcePos, preamble_->scope()); | |
| 277 preamble_ = new_false; | |
| 278 node->false_expr()->Visit(this); | |
| 279 AstNode* new_false_result = result_; | |
| 280 preamble_ = saved_preamble; | |
| 281 IfNode* new_if = new(I) IfNode(Scanner::kNoSourcePos, | |
| 282 new_condition, | |
| 283 new_true, | |
| 284 new_false); | |
| 285 preamble_->Add(new_if); | |
| 286 LocalVariable* result = AddToPreambleNewTempVar( | |
| 287 new(I) ConditionalExprNode(Scanner::kNoSourcePos, | |
| 288 new_condition, | |
| 289 new_true_result, | |
| 290 new_false_result)); | |
| 291 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 292 } | |
| 293 | |
| 294 | |
| 295 void AwaitTransformer::VisitArgumentListNode(ArgumentListNode* node) { | |
| 296 ArgumentListNode* new_args = new(I) ArgumentListNode(node->token_pos()); | |
| 297 for (intptr_t i = 0; i < node->length(); i++) { | |
| 298 node->NodeAt(i)->Visit(this); | |
| 299 new_args->Add(result_); | |
| 300 } | |
| 301 result_ = new_args; | |
| 302 } | |
| 303 | |
| 304 | |
| 305 void AwaitTransformer::VisitArrayNode(ArrayNode* node) { | |
| 306 GrowableArray<AstNode*> new_elements; | |
| 307 for (intptr_t i = 0; i < node->length(); i++) { | |
| 308 node->ElementAt(i)->Visit(this); | |
| 309 new_elements.Add(result_); | |
| 310 } | |
| 311 result_ = new(I) ArrayNode(node->token_pos(), node->type(), new_elements); | |
| 312 } | |
| 313 | |
| 314 | |
| 315 void AwaitTransformer::VisitStringInterpolateNode(StringInterpolateNode* node) { | |
| 316 node->value()->Visit(this); | |
| 317 ArrayNode* new_value = result_->AsArrayNode(); | |
| 318 LocalVariable* result = AddToPreambleNewTempVar( | |
| 319 new(I) StringInterpolateNode(node->token_pos(), | |
| 320 new_value)); | |
| 321 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 322 } | |
| 323 | |
| 324 | |
| 325 void AwaitTransformer::VisitClosureNode(ClosureNode* node) { | |
| 326 AstNode* new_receiver = node->receiver(); | |
| 327 if (new_receiver != NULL) { | |
| 328 new_receiver->Visit(this); | |
| 329 new_receiver = result_; | |
| 330 } | |
| 331 LocalVariable* result = AddToPreambleNewTempVar( | |
| 332 new(I) ClosureNode(node->token_pos(), | |
| 333 node->function(), | |
| 334 new_receiver, | |
| 335 node->scope())); | |
| 336 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 337 } | |
| 338 | |
| 339 | |
| 340 void AwaitTransformer::VisitInstanceCallNode(InstanceCallNode* node) { | |
| 341 node->receiver()->Visit(this); | |
| 342 AstNode* new_receiver = result_; | |
| 343 node->arguments()->Visit(this); | |
| 344 ArgumentListNode* new_args = result_->AsArgumentListNode(); | |
| 345 | |
| 346 LocalVariable* result = AddToPreambleNewTempVar( | |
| 347 new(I) InstanceCallNode(node->token_pos(), | |
| 348 new_receiver, | |
| 349 node->function_name(), | |
| 350 new_args)); | |
| 351 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 352 } | |
| 353 | |
| 354 | |
| 355 void AwaitTransformer::VisitStaticCallNode(StaticCallNode* node) { | |
| 356 node->arguments()->Visit(this); | |
| 357 ArgumentListNode* new_args = result_->AsArgumentListNode(); | |
| 358 | |
| 359 LocalVariable* result = AddToPreambleNewTempVar( | |
| 360 new(I) StaticCallNode(node->token_pos(), | |
| 361 node->function(), | |
| 362 new_args)); | |
| 363 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 364 } | |
| 365 | |
| 366 | |
| 367 void AwaitTransformer::VisitConstructorCallNode(ConstructorCallNode* node) { | |
| 368 node->arguments()->Visit(this); | |
| 369 ArgumentListNode* new_args = result_->AsArgumentListNode(); | |
| 370 | |
| 371 LocalVariable* result = AddToPreambleNewTempVar( | |
| 372 new(I) ConstructorCallNode(node->token_pos(), | |
| 373 node->type_arguments(), | |
| 374 node->constructor(), | |
| 375 new_args)); | |
| 376 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 377 } | |
| 378 | |
| 379 | |
| 380 void AwaitTransformer::VisitInstanceGetterNode(InstanceGetterNode* node) { | |
| 381 node->receiver()->Visit(this); | |
| 382 AstNode* new_receiver = result_; | |
| 383 | |
| 384 LocalVariable* result = AddToPreambleNewTempVar( | |
| 385 new(I) InstanceGetterNode(node->token_pos(), | |
| 386 new_receiver, | |
| 387 node->field_name())); | |
| 388 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 389 } | |
| 390 | |
| 391 | |
| 392 void AwaitTransformer::VisitInstanceSetterNode(InstanceSetterNode* node) { | |
| 393 AstNode* new_receiver = node->receiver(); | |
| 394 if (new_receiver != NULL) { | |
| 395 new_receiver->Visit(this); | |
| 396 new_receiver = result_; | |
| 397 } | |
| 398 | |
| 399 node->value()->Visit(this); | |
| 400 AstNode* new_value = result_; | |
| 401 | |
| 402 LocalVariable* result = AddToPreambleNewTempVar( | |
| 403 new(I) InstanceSetterNode(node->token_pos(), | |
| 404 new_receiver, | |
| 405 node->field_name(), | |
| 406 new_value)); | |
| 407 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 408 } | |
| 409 | |
| 410 | |
| 411 void AwaitTransformer::VisitStaticGetterNode(StaticGetterNode* node) { | |
| 412 AstNode* new_receiver = node->receiver(); | |
| 413 if (new_receiver != NULL) { | |
| 414 new_receiver->Visit(this); | |
| 415 new_receiver = result_; | |
| 416 } | |
| 417 | |
| 418 LocalVariable* result = AddToPreambleNewTempVar( | |
| 419 new(I) StaticGetterNode(node->token_pos(), | |
| 420 new_receiver, | |
| 421 node->is_super_getter(), | |
| 422 node->cls(), | |
| 423 node->field_name())); | |
| 424 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 425 } | |
| 426 | |
| 427 | |
| 428 void AwaitTransformer::VisitStaticSetterNode(StaticSetterNode* node) { | |
| 429 AstNode* new_receiver = node->receiver(); | |
| 430 if (new_receiver != NULL) { | |
| 431 new_receiver->Visit(this); | |
| 432 new_receiver = result_; | |
| 433 } | |
| 434 node->value()->Visit(this); | |
| 435 AstNode* new_value = result_; | |
| 436 | |
| 437 LocalVariable* result = AddToPreambleNewTempVar( | |
| 438 new(I) StaticSetterNode(node->token_pos(), | |
| 439 new_receiver, | |
| 440 node->cls(), | |
| 441 node->field_name(), | |
| 442 new_value)); | |
| 443 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 444 } | |
| 445 | |
| 446 | |
| 447 void AwaitTransformer::VisitLoadLocalNode(LoadLocalNode* node) { | |
| 448 LocalVariable* result = AddToPreambleNewTempVar( | |
| 449 new(I) LoadLocalNode(node->token_pos(), &node->local())); | |
| 450 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 451 } | |
| 452 | |
| 453 | |
| 454 void AwaitTransformer::VisitStoreLocalNode(StoreLocalNode* node) { | |
| 455 node->value()->Visit(this); | |
| 456 AstNode* new_value = result_; | |
| 457 LocalVariable* result = AddToPreambleNewTempVar( | |
| 458 new(I) StoreLocalNode(node->token_pos(), | |
| 459 &node->local(), | |
| 460 new_value)); | |
| 461 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 462 } | |
| 463 | |
| 464 | |
| 465 void AwaitTransformer::VisitLoadStaticFieldNode(LoadStaticFieldNode* node) { | |
| 466 LocalVariable* result = AddToPreambleNewTempVar( | |
| 467 new(I) LoadStaticFieldNode(node->token_pos(), | |
| 468 node->field())); | |
| 469 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 470 } | |
| 471 | |
| 472 | |
| 473 void AwaitTransformer::VisitStoreStaticFieldNode(StoreStaticFieldNode* node) { | |
| 474 node->value()->Visit(this); | |
| 475 AstNode* new_value = result_; | |
| 476 LocalVariable* result = AddToPreambleNewTempVar( | |
| 477 new(I) StoreStaticFieldNode(node->token_pos(), | |
| 478 node->field(), | |
| 479 new_value)); | |
| 480 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 481 } | |
| 482 | |
| 483 | |
| 484 void AwaitTransformer::VisitLoadIndexedNode(LoadIndexedNode* node) { | |
| 485 node->array()->Visit(this); | |
| 486 AstNode* new_array = result_; | |
| 487 node->index_expr()->Visit(this); | |
| 488 AstNode* new_index = result_; | |
| 489 LocalVariable* result = AddToPreambleNewTempVar( | |
| 490 new(I) LoadIndexedNode(node->token_pos(), | |
| 491 new_array, | |
| 492 new_index, | |
| 493 node->super_class())); | |
| 494 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 495 } | |
| 496 | |
| 497 | |
| 498 void AwaitTransformer::VisitStoreIndexedNode(StoreIndexedNode* node) { | |
| 499 node->array()->Visit(this); | |
| 500 AstNode* new_array = result_; | |
| 501 node->index_expr()->Visit(this); | |
| 502 AstNode* new_index = result_; | |
| 503 node->value()->Visit(this); | |
| 504 AstNode* new_value = result_; | |
| 505 LocalVariable* result = AddToPreambleNewTempVar( | |
| 506 new(I) StoreIndexedNode(node->token_pos(), | |
| 507 new_array, | |
| 508 new_index, | |
| 509 new_value, | |
| 510 node->super_class())); | |
| 511 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 512 } | |
| 513 | |
| 514 | |
| 515 void AwaitTransformer::VisitAssignableNode(AssignableNode* node) { | |
| 516 node->expr()->Visit(this); | |
| 517 AstNode* new_expr = result_; | |
| 518 LocalVariable* result = AddToPreambleNewTempVar( | |
| 519 new(I) AssignableNode(node->token_pos(), | |
| 520 new_expr, | |
| 521 node->type(), | |
| 522 node->dst_name())); | |
| 523 result_ = new(I) LoadLocalNode(Scanner::kNoSourcePos, result); | |
| 524 } | |
| 525 | |
| 526 | |
| 527 void AwaitTransformer::VisitLetNode(LetNode* node) { | |
| 528 // TODO(mlippautz): Check initializers and their temps. | |
| 529 LetNode* result = new(I) LetNode(node->token_pos()); | |
| 530 for (intptr_t i = 0; i < node->nodes().length(); i++) { | |
| 531 node->nodes()[i]->Visit(this); | |
| 532 result->AddNode(result_); | |
| 533 } | |
| 534 result_ = result; | |
| 535 } | |
| 536 | |
| 537 | |
| 538 void AwaitTransformer::VisitThrowNode(ThrowNode* node) { | |
| 539 // TODO(mlippautz): Check if relevant. | |
| 540 node->exception()->Visit(this); | |
| 541 AstNode* new_exception = result_; | |
| 542 node->stacktrace()->Visit(this); | |
| 543 AstNode* new_stacktrace = result_; | |
| 544 result_ = new(I) ThrowNode(node->token_pos(), | |
| 545 new_exception, | |
| 546 new_stacktrace); | |
| 547 } | |
| 548 | |
| 549 } // namespace dart | |
| OLD | NEW |