1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "src/ast/prettyprinter.h"
6
7 #include <stdarg.h>
8
9 #include "src/ast/ast-value-factory.h"
10 #include "src/ast/scopes.h"
11 #include "src/base/platform/platform.h"
12 #include "src/common/globals.h"
13 #include "src/objects/objects-inl.h"
14 #include "src/strings/string-builder-inl.h"
15 #include "src/utils/vector.h"
16
17 namespace v8 {
18 namespace internal {
19
CallPrinter(Isolate * isolate,bool is_user_js,SpreadErrorInArgsHint error_in_spread_args)20 CallPrinter::CallPrinter(Isolate* isolate, bool is_user_js,
21 SpreadErrorInArgsHint error_in_spread_args)
22 : builder_(new IncrementalStringBuilder(isolate)) {
23 isolate_ = isolate;
24 position_ = 0;
25 num_prints_ = 0;
26 found_ = false;
27 done_ = false;
28 is_call_error_ = false;
29 is_iterator_error_ = false;
30 is_async_iterator_error_ = false;
31 destructuring_prop_ = nullptr;
32 destructuring_assignment_ = nullptr;
33 is_user_js_ = is_user_js;
34 error_in_spread_args_ = error_in_spread_args;
35 spread_arg_ = nullptr;
36 function_kind_ = kNormalFunction;
37 InitializeAstVisitor(isolate);
38 }
39
40 CallPrinter::~CallPrinter() = default;
41
GetErrorHint() const42 CallPrinter::ErrorHint CallPrinter::GetErrorHint() const {
43 if (is_call_error_) {
44 if (is_iterator_error_) return ErrorHint::kCallAndNormalIterator;
45 if (is_async_iterator_error_) return ErrorHint::kCallAndAsyncIterator;
46 } else {
47 if (is_iterator_error_) return ErrorHint::kNormalIterator;
48 if (is_async_iterator_error_) return ErrorHint::kAsyncIterator;
49 }
50 return ErrorHint::kNone;
51 }
52
Print(FunctionLiteral * program,int position)53 Handle<String> CallPrinter::Print(FunctionLiteral* program, int position) {
54 num_prints_ = 0;
55 position_ = position;
56 Find(program);
57 return builder_->Finish().ToHandleChecked();
58 }
59
60
Find(AstNode * node,bool print)61 void CallPrinter::Find(AstNode* node, bool print) {
62 if (found_) {
63 if (print) {
64 int prev_num_prints = num_prints_;
65 Visit(node);
66 if (prev_num_prints != num_prints_) return;
67 }
68 Print("(intermediate value)");
69 } else {
70 Visit(node);
71 }
72 }
73
Print(const char * str)74 void CallPrinter::Print(const char* str) {
75 if (!found_ || done_) return;
76 num_prints_++;
77 builder_->AppendCString(str);
78 }
79
Print(Handle<String> str)80 void CallPrinter::Print(Handle<String> str) {
81 if (!found_ || done_) return;
82 num_prints_++;
83 builder_->AppendString(str);
84 }
85
VisitBlock(Block * node)86 void CallPrinter::VisitBlock(Block* node) {
87 FindStatements(node->statements());
88 }
89
90
VisitVariableDeclaration(VariableDeclaration * node)91 void CallPrinter::VisitVariableDeclaration(VariableDeclaration* node) {}
92
93
VisitFunctionDeclaration(FunctionDeclaration * node)94 void CallPrinter::VisitFunctionDeclaration(FunctionDeclaration* node) {}
95
96
VisitExpressionStatement(ExpressionStatement * node)97 void CallPrinter::VisitExpressionStatement(ExpressionStatement* node) {
98 Find(node->expression());
99 }
100
101
VisitEmptyStatement(EmptyStatement * node)102 void CallPrinter::VisitEmptyStatement(EmptyStatement* node) {}
103
104
VisitSloppyBlockFunctionStatement(SloppyBlockFunctionStatement * node)105 void CallPrinter::VisitSloppyBlockFunctionStatement(
106 SloppyBlockFunctionStatement* node) {
107 Find(node->statement());
108 }
109
110
VisitIfStatement(IfStatement * node)111 void CallPrinter::VisitIfStatement(IfStatement* node) {
112 Find(node->condition());
113 Find(node->then_statement());
114 if (node->HasElseStatement()) {
115 Find(node->else_statement());
116 }
117 }
118
119
VisitContinueStatement(ContinueStatement * node)120 void CallPrinter::VisitContinueStatement(ContinueStatement* node) {}
121
122
VisitBreakStatement(BreakStatement * node)123 void CallPrinter::VisitBreakStatement(BreakStatement* node) {}
124
125
VisitReturnStatement(ReturnStatement * node)126 void CallPrinter::VisitReturnStatement(ReturnStatement* node) {
127 Find(node->expression());
128 }
129
130
VisitWithStatement(WithStatement * node)131 void CallPrinter::VisitWithStatement(WithStatement* node) {
132 Find(node->expression());
133 Find(node->statement());
134 }
135
136
VisitSwitchStatement(SwitchStatement * node)137 void CallPrinter::VisitSwitchStatement(SwitchStatement* node) {
138 Find(node->tag());
139 for (CaseClause* clause : *node->cases()) {
140 if (!clause->is_default()) Find(clause->label());
141 FindStatements(clause->statements());
142 }
143 }
144
145
VisitDoWhileStatement(DoWhileStatement * node)146 void CallPrinter::VisitDoWhileStatement(DoWhileStatement* node) {
147 Find(node->body());
148 Find(node->cond());
149 }
150
151
VisitWhileStatement(WhileStatement * node)152 void CallPrinter::VisitWhileStatement(WhileStatement* node) {
153 Find(node->cond());
154 Find(node->body());
155 }
156
157
VisitForStatement(ForStatement * node)158 void CallPrinter::VisitForStatement(ForStatement* node) {
159 if (node->init() != nullptr) {
160 Find(node->init());
161 }
162 if (node->cond() != nullptr) Find(node->cond());
163 if (node->next() != nullptr) Find(node->next());
164 Find(node->body());
165 }
166
167
VisitForInStatement(ForInStatement * node)168 void CallPrinter::VisitForInStatement(ForInStatement* node) {
169 Find(node->each());
170 Find(node->subject());
171 Find(node->body());
172 }
173
174
VisitForOfStatement(ForOfStatement * node)175 void CallPrinter::VisitForOfStatement(ForOfStatement* node) {
176 Find(node->each());
177
178 // Check the subject's position in case there was a GetIterator error.
179 bool was_found = false;
180 if (node->subject()->position() == position_) {
181 is_async_iterator_error_ = node->type() == IteratorType::kAsync;
182 is_iterator_error_ = !is_async_iterator_error_;
183 was_found = !found_;
184 if (was_found) {
185 found_ = true;
186 }
187 }
188 Find(node->subject(), true);
189 if (was_found) {
190 done_ = true;
191 found_ = false;
192 }
193
194 Find(node->body());
195 }
196
197
VisitTryCatchStatement(TryCatchStatement * node)198 void CallPrinter::VisitTryCatchStatement(TryCatchStatement* node) {
199 Find(node->try_block());
200 Find(node->catch_block());
201 }
202
203
VisitTryFinallyStatement(TryFinallyStatement * node)204 void CallPrinter::VisitTryFinallyStatement(TryFinallyStatement* node) {
205 Find(node->try_block());
206 Find(node->finally_block());
207 }
208
209
VisitDebuggerStatement(DebuggerStatement * node)210 void CallPrinter::VisitDebuggerStatement(DebuggerStatement* node) {}
211
212
VisitFunctionLiteral(FunctionLiteral * node)213 void CallPrinter::VisitFunctionLiteral(FunctionLiteral* node) {
214 FunctionKind last_function_kind = function_kind_;
215 function_kind_ = node->kind();
216 FindStatements(node->body());
217 function_kind_ = last_function_kind;
218 }
219
220
VisitClassLiteral(ClassLiteral * node)221 void CallPrinter::VisitClassLiteral(ClassLiteral* node) {
222 if (node->extends()) Find(node->extends());
223 for (int i = 0; i < node->public_members()->length(); i++) {
224 Find(node->public_members()->at(i)->value());
225 }
226 for (int i = 0; i < node->private_members()->length(); i++) {
227 Find(node->private_members()->at(i)->value());
228 }
229 }
230
VisitInitializeClassMembersStatement(InitializeClassMembersStatement * node)231 void CallPrinter::VisitInitializeClassMembersStatement(
232 InitializeClassMembersStatement* node) {
233 for (int i = 0; i < node->fields()->length(); i++) {
234 Find(node->fields()->at(i)->value());
235 }
236 }
237
VisitNativeFunctionLiteral(NativeFunctionLiteral * node)238 void CallPrinter::VisitNativeFunctionLiteral(NativeFunctionLiteral* node) {}
239
240
VisitConditional(Conditional * node)241 void CallPrinter::VisitConditional(Conditional* node) {
242 Find(node->condition());
243 Find(node->then_expression());
244 Find(node->else_expression());
245 }
246
247
VisitLiteral(Literal * node)248 void CallPrinter::VisitLiteral(Literal* node) {
249 // TODO(adamk): Teach Literal how to print its values without
250 // allocating on the heap.
251 PrintLiteral(node->BuildValue(isolate_), true);
252 }
253
254
VisitRegExpLiteral(RegExpLiteral * node)255 void CallPrinter::VisitRegExpLiteral(RegExpLiteral* node) {
256 Print("/");
257 PrintLiteral(node->pattern(), false);
258 Print("/");
259 if (node->flags() & RegExp::kGlobal) Print("g");
260 if (node->flags() & RegExp::kIgnoreCase) Print("i");
261 if (node->flags() & RegExp::kLinear) Print("l");
262 if (node->flags() & RegExp::kMultiline) Print("m");
263 if (node->flags() & RegExp::kUnicode) Print("u");
264 if (node->flags() & RegExp::kSticky) Print("y");
265 }
266
267
VisitObjectLiteral(ObjectLiteral * node)268 void CallPrinter::VisitObjectLiteral(ObjectLiteral* node) {
269 Print("{");
270 for (int i = 0; i < node->properties()->length(); i++) {
271 Find(node->properties()->at(i)->value());
272 }
273 Print("}");
274 }
275
276
VisitArrayLiteral(ArrayLiteral * node)277 void CallPrinter::VisitArrayLiteral(ArrayLiteral* node) {
278 Print("[");
279 for (int i = 0; i < node->values()->length(); i++) {
280 if (i != 0) Print(",");
281 Expression* subexpr = node->values()->at(i);
282 Spread* spread = subexpr->AsSpread();
283 if (spread != nullptr && !found_ &&
284 position_ == spread->expression()->position()) {
285 found_ = true;
286 is_iterator_error_ = true;
287 Find(spread->expression(), true);
288 done_ = true;
289 return;
290 }
291 Find(subexpr, true);
292 }
293 Print("]");
294 }
295
296
VisitVariableProxy(VariableProxy * node)297 void CallPrinter::VisitVariableProxy(VariableProxy* node) {
298 if (is_user_js_) {
299 PrintLiteral(node->name(), false);
300 } else {
301 // Variable names of non-user code are meaningless due to minification.
302 Print("(var)");
303 }
304 }
305
306
VisitAssignment(Assignment * node)307 void CallPrinter::VisitAssignment(Assignment* node) {
308 bool was_found = false;
309 if (node->target()->IsObjectLiteral()) {
310 ObjectLiteral* target = node->target()->AsObjectLiteral();
311 if (target->position() == position_) {
312 was_found = !found_;
313 found_ = true;
314 destructuring_assignment_ = node;
315 } else {
316 for (ObjectLiteralProperty* prop : *target->properties()) {
317 if (prop->value()->position() == position_) {
318 was_found = !found_;
319 found_ = true;
320 destructuring_prop_ = prop;
321 destructuring_assignment_ = node;
322 break;
323 }
324 }
325 }
326 }
327 if (!was_found) {
328 Find(node->target());
329 if (node->target()->IsArrayLiteral()) {
330 // Special case the visit for destructuring array assignment.
331 bool was_found = false;
332 if (node->value()->position() == position_) {
333 is_iterator_error_ = true;
334 was_found = !found_;
335 found_ = true;
336 }
337 Find(node->value(), true);
338 if (was_found) {
339 done_ = true;
340 found_ = false;
341 }
342 } else {
343 Find(node->value());
344 }
345 } else {
346 Find(node->value(), true);
347 }
348
349 if (was_found) {
350 done_ = true;
351 found_ = false;
352 }
353 }
354
VisitCompoundAssignment(CompoundAssignment * node)355 void CallPrinter::VisitCompoundAssignment(CompoundAssignment* node) {
356 VisitAssignment(node);
357 }
358
VisitYield(Yield * node)359 void CallPrinter::VisitYield(Yield* node) { Find(node->expression()); }
360
VisitYieldStar(YieldStar * node)361 void CallPrinter::VisitYieldStar(YieldStar* node) {
362 if (!found_ && position_ == node->expression()->position()) {
363 found_ = true;
364 if (IsAsyncFunction(function_kind_))
365 is_async_iterator_error_ = true;
366 else
367 is_iterator_error_ = true;
368 Print("yield* ");
369 }
370 Find(node->expression());
371 }
372
VisitAwait(Await * node)373 void CallPrinter::VisitAwait(Await* node) { Find(node->expression()); }
374
VisitThrow(Throw * node)375 void CallPrinter::VisitThrow(Throw* node) { Find(node->exception()); }
376
VisitOptionalChain(OptionalChain * node)377 void CallPrinter::VisitOptionalChain(OptionalChain* node) {
378 Find(node->expression());
379 }
380
VisitProperty(Property * node)381 void CallPrinter::VisitProperty(Property* node) {
382 Expression* key = node->key();
383 Literal* literal = key->AsLiteral();
384 if (literal != nullptr &&
385 literal->BuildValue(isolate_)->IsInternalizedString()) {
386 Find(node->obj(), true);
387 if (node->is_optional_chain_link()) {
388 Print("?");
389 }
390 Print(".");
391 // TODO(adamk): Teach Literal how to print its values without
392 // allocating on the heap.
393 PrintLiteral(literal->BuildValue(isolate_), false);
394 } else {
395 Find(node->obj(), true);
396 if (node->is_optional_chain_link()) {
397 Print("?.");
398 }
399 Print("[");
400 Find(key, true);
401 Print("]");
402 }
403 }
404
VisitCall(Call * node)405 void CallPrinter::VisitCall(Call* node) {
406 bool was_found = false;
407 if (node->position() == position_) {
408 if (error_in_spread_args_ == SpreadErrorInArgsHint::kErrorInArgs) {
409 found_ = true;
410 spread_arg_ = node->arguments()->last()->AsSpread()->expression();
411 Find(spread_arg_, true);
412
413 done_ = true;
414 found_ = false;
415 return;
416 }
417
418 is_call_error_ = true;
419 was_found = !found_;
420 }
421
422 if (was_found) {
423 // Bail out if the error is caused by a direct call to a variable in
424 // non-user JS code. The variable name is meaningless due to minification.
425 if (!is_user_js_ && node->expression()->IsVariableProxy()) {
426 done_ = true;
427 return;
428 }
429 found_ = true;
430 }
431 Find(node->expression(), true);
432 if (!was_found && !is_iterator_error_) Print("(...)");
433 FindArguments(node->arguments());
434 if (was_found) {
435 done_ = true;
436 found_ = false;
437 }
438 }
439
440
VisitCallNew(CallNew * node)441 void CallPrinter::VisitCallNew(CallNew* node) {
442 bool was_found = false;
443 if (node->position() == position_) {
444 if (error_in_spread_args_ == SpreadErrorInArgsHint::kErrorInArgs) {
445 found_ = true;
446 spread_arg_ = node->arguments()->last()->AsSpread()->expression();
447 Find(spread_arg_, true);
448
449 done_ = true;
450 found_ = false;
451 return;
452 }
453
454 is_call_error_ = true;
455 was_found = !found_;
456 }
457 if (was_found) {
458 // Bail out if the error is caused by a direct call to a variable in
459 // non-user JS code. The variable name is meaningless due to minification.
460 if (!is_user_js_ && node->expression()->IsVariableProxy()) {
461 done_ = true;
462 return;
463 }
464 found_ = true;
465 }
466 Find(node->expression(), was_found || is_iterator_error_);
467 FindArguments(node->arguments());
468 if (was_found) {
469 done_ = true;
470 found_ = false;
471 }
472 }
473
474
VisitCallRuntime(CallRuntime * node)475 void CallPrinter::VisitCallRuntime(CallRuntime* node) {
476 FindArguments(node->arguments());
477 }
478
479
VisitUnaryOperation(UnaryOperation * node)480 void CallPrinter::VisitUnaryOperation(UnaryOperation* node) {
481 Token::Value op = node->op();
482 bool needsSpace =
483 op == Token::DELETE || op == Token::TYPEOF || op == Token::VOID;
484 Print("(");
485 Print(Token::String(op));
486 if (needsSpace) Print(" ");
487 Find(node->expression(), true);
488 Print(")");
489 }
490
491
VisitCountOperation(CountOperation * node)492 void CallPrinter::VisitCountOperation(CountOperation* node) {
493 Print("(");
494 if (node->is_prefix()) Print(Token::String(node->op()));
495 Find(node->expression(), true);
496 if (node->is_postfix()) Print(Token::String(node->op()));
497 Print(")");
498 }
499
500
VisitBinaryOperation(BinaryOperation * node)501 void CallPrinter::VisitBinaryOperation(BinaryOperation* node) {
502 Print("(");
503 Find(node->left(), true);
504 Print(" ");
505 Print(Token::String(node->op()));
506 Print(" ");
507 Find(node->right(), true);
508 Print(")");
509 }
510
VisitNaryOperation(NaryOperation * node)511 void CallPrinter::VisitNaryOperation(NaryOperation* node) {
512 Print("(");
513 Find(node->first(), true);
514 for (size_t i = 0; i < node->subsequent_length(); ++i) {
515 Print(" ");
516 Print(Token::String(node->op()));
517 Print(" ");
518 Find(node->subsequent(i), true);
519 }
520 Print(")");
521 }
522
VisitCompareOperation(CompareOperation * node)523 void CallPrinter::VisitCompareOperation(CompareOperation* node) {
524 Print("(");
525 Find(node->left(), true);
526 Print(" ");
527 Print(Token::String(node->op()));
528 Print(" ");
529 Find(node->right(), true);
530 Print(")");
531 }
532
533
VisitSpread(Spread * node)534 void CallPrinter::VisitSpread(Spread* node) {
535 Print("(...");
536 Find(node->expression(), true);
537 Print(")");
538 }
539
VisitEmptyParentheses(EmptyParentheses * node)540 void CallPrinter::VisitEmptyParentheses(EmptyParentheses* node) {
541 UNREACHABLE();
542 }
543
VisitGetTemplateObject(GetTemplateObject * node)544 void CallPrinter::VisitGetTemplateObject(GetTemplateObject* node) {}
545
VisitTemplateLiteral(TemplateLiteral * node)546 void CallPrinter::VisitTemplateLiteral(TemplateLiteral* node) {
547 for (Expression* substitution : *node->substitutions()) {
548 Find(substitution, true);
549 }
550 }
551
VisitImportCallExpression(ImportCallExpression * node)552 void CallPrinter::VisitImportCallExpression(ImportCallExpression* node) {
553 Print("ImportCall(");
554 Find(node->argument(), true);
555 Print(")");
556 }
557
VisitThisExpression(ThisExpression * node)558 void CallPrinter::VisitThisExpression(ThisExpression* node) { Print("this"); }
559
VisitSuperPropertyReference(SuperPropertyReference * node)560 void CallPrinter::VisitSuperPropertyReference(SuperPropertyReference* node) {}
561
562
VisitSuperCallReference(SuperCallReference * node)563 void CallPrinter::VisitSuperCallReference(SuperCallReference* node) {
564 Print("super");
565 }
566
567
FindStatements(const ZonePtrList<Statement> * statements)568 void CallPrinter::FindStatements(const ZonePtrList<Statement>* statements) {
569 if (statements == nullptr) return;
570 for (int i = 0; i < statements->length(); i++) {
571 Find(statements->at(i));
572 }
573 }
574
FindArguments(const ZonePtrList<Expression> * arguments)575 void CallPrinter::FindArguments(const ZonePtrList<Expression>* arguments) {
576 if (found_) return;
577 for (int i = 0; i < arguments->length(); i++) {
578 Find(arguments->at(i));
579 }
580 }
581
PrintLiteral(Handle<Object> value,bool quote)582 void CallPrinter::PrintLiteral(Handle<Object> value, bool quote) {
583 if (value->IsString()) {
584 if (quote) Print("\"");
585 Print(Handle<String>::cast(value));
586 if (quote) Print("\"");
587 } else if (value->IsNull(isolate_)) {
588 Print("null");
589 } else if (value->IsTrue(isolate_)) {
590 Print("true");
591 } else if (value->IsFalse(isolate_)) {
592 Print("false");
593 } else if (value->IsUndefined(isolate_)) {
594 Print("undefined");
595 } else if (value->IsNumber()) {
596 Print(isolate_->factory()->NumberToString(value));
597 } else if (value->IsSymbol()) {
598 // Symbols can only occur as literals if they were inserted by the parser.
599 PrintLiteral(handle(Handle<Symbol>::cast(value)->description(), isolate_),
600 false);
601 }
602 }
603
604
PrintLiteral(const AstRawString * value,bool quote)605 void CallPrinter::PrintLiteral(const AstRawString* value, bool quote) {
606 PrintLiteral(value->string(), quote);
607 }
608
609 //-----------------------------------------------------------------------------
610
611
612 #ifdef DEBUG
613
Print(AstNode * node)614 const char* AstPrinter::Print(AstNode* node) {
615 Init();
616 Visit(node);
617 return output_;
618 }
619
Init()620 void AstPrinter::Init() {
621 if (size_ == 0) {
622 DCHECK_NULL(output_);
623 const int initial_size = 256;
624 output_ = NewArray<char>(initial_size);
625 size_ = initial_size;
626 }
627 output_[0] = '\0';
628 pos_ = 0;
629 }
630
Print(const char * format,...)631 void AstPrinter::Print(const char* format, ...) {
632 for (;;) {
633 va_list arguments;
634 va_start(arguments, format);
635 int n = VSNPrintF(Vector<char>(output_, size_) + pos_,
636 format,
637 arguments);
638 va_end(arguments);
639
640 if (n >= 0) {
641 // there was enough space - we are done
642 pos_ += n;
643 return;
644 } else {
645 // there was not enough space - allocate more and try again
646 const int slack = 32;
647 int new_size = size_ + (size_ >> 1) + slack;
648 char* new_output = NewArray<char>(new_size);
649 MemCopy(new_output, output_, pos_);
650 DeleteArray(output_);
651 output_ = new_output;
652 size_ = new_size;
653 }
654 }
655 }
656
PrintLiteral(Literal * literal,bool quote)657 void AstPrinter::PrintLiteral(Literal* literal, bool quote) {
658 switch (literal->type()) {
659 case Literal::kString:
660 PrintLiteral(literal->AsRawString(), quote);
661 break;
662 case Literal::kSymbol:
663 const char* symbol;
664 switch (literal->AsSymbol()) {
665 case AstSymbol::kHomeObjectSymbol:
666 symbol = "HomeObjectSymbol";
667 }
668 Print("%s", symbol);
669 break;
670 case Literal::kSmi:
671 Print("%d", Smi::ToInt(literal->AsSmiLiteral()));
672 break;
673 case Literal::kHeapNumber:
674 Print("%g", literal->AsNumber());
675 break;
676 case Literal::kBigInt:
677 Print("%sn", literal->AsBigInt().c_str());
678 break;
679 case Literal::kNull:
680 Print("null");
681 break;
682 case Literal::kUndefined:
683 Print("undefined");
684 break;
685 case Literal::kTheHole:
686 Print("the hole");
687 break;
688 case Literal::kBoolean:
689 if (literal->ToBooleanIsTrue()) {
690 Print("true");
691 } else {
692 Print("false");
693 }
694 break;
695 }
696 }
697
PrintLiteral(const AstRawString * value,bool quote)698 void AstPrinter::PrintLiteral(const AstRawString* value, bool quote) {
699 if (quote) Print("\"");
700 if (value != nullptr) {
701 const char* format = value->is_one_byte() ? "%c" : "%lc";
702 const int increment = value->is_one_byte() ? 1 : 2;
703 const unsigned char* raw_bytes = value->raw_data();
704 for (int i = 0; i < value->length(); i += increment) {
705 Print(format, raw_bytes[i]);
706 }
707 }
708 if (quote) Print("\"");
709 }
710
PrintLiteral(const AstConsString * value,bool quote)711 void AstPrinter::PrintLiteral(const AstConsString* value, bool quote) {
712 if (quote) Print("\"");
713 if (value != nullptr) {
714 std::forward_list<const AstRawString*> strings = value->ToRawStrings();
715 for (const AstRawString* string : strings) {
716 PrintLiteral(string, false);
717 }
718 }
719 if (quote) Print("\"");
720 }
721
722 //-----------------------------------------------------------------------------
723
724 class IndentedScope {
725 public:
IndentedScope(AstPrinter * printer,const char * txt)726 IndentedScope(AstPrinter* printer, const char* txt)
727 : ast_printer_(printer) {
728 ast_printer_->PrintIndented(txt);
729 ast_printer_->Print("\n");
730 ast_printer_->inc_indent();
731 }
732
IndentedScope(AstPrinter * printer,const char * txt,int pos)733 IndentedScope(AstPrinter* printer, const char* txt, int pos)
734 : ast_printer_(printer) {
735 ast_printer_->PrintIndented(txt);
736 ast_printer_->Print(" at %d\n", pos);
737 ast_printer_->inc_indent();
738 }
739
~IndentedScope()740 virtual ~IndentedScope() {
741 ast_printer_->dec_indent();
742 }
743
744 private:
745 AstPrinter* ast_printer_;
746 };
747
748
749 //-----------------------------------------------------------------------------
750
AstPrinter(uintptr_t stack_limit)751 AstPrinter::AstPrinter(uintptr_t stack_limit)
752 : output_(nullptr), size_(0), pos_(0), indent_(0) {
753 InitializeAstVisitor(stack_limit);
754 }
755
~AstPrinter()756 AstPrinter::~AstPrinter() {
757 DCHECK_EQ(indent_, 0);
758 DeleteArray(output_);
759 }
760
761
PrintIndented(const char * txt)762 void AstPrinter::PrintIndented(const char* txt) {
763 for (int i = 0; i < indent_; i++) {
764 Print(". ");
765 }
766 Print("%s", txt);
767 }
768
PrintLiteralIndented(const char * info,Literal * literal,bool quote)769 void AstPrinter::PrintLiteralIndented(const char* info, Literal* literal,
770 bool quote) {
771 PrintIndented(info);
772 Print(" ");
773 PrintLiteral(literal, quote);
774 Print("\n");
775 }
776
PrintLiteralIndented(const char * info,const AstRawString * value,bool quote)777 void AstPrinter::PrintLiteralIndented(const char* info,
778 const AstRawString* value, bool quote) {
779 PrintIndented(info);
780 Print(" ");
781 PrintLiteral(value, quote);
782 Print("\n");
783 }
784
PrintLiteralIndented(const char * info,const AstConsString * value,bool quote)785 void AstPrinter::PrintLiteralIndented(const char* info,
786 const AstConsString* value, bool quote) {
787 PrintIndented(info);
788 Print(" ");
789 PrintLiteral(value, quote);
790 Print("\n");
791 }
792
PrintLiteralWithModeIndented(const char * info,Variable * var,const AstRawString * value)793 void AstPrinter::PrintLiteralWithModeIndented(const char* info, Variable* var,
794 const AstRawString* value) {
795 if (var == nullptr) {
796 PrintLiteralIndented(info, value, true);
797 } else {
798 EmbeddedVector<char, 256> buf;
799 int pos =
800 SNPrintF(buf, "%s (%p) (mode = %s, assigned = %s", info,
801 reinterpret_cast<void*>(var), VariableMode2String(var->mode()),
802 var->maybe_assigned() == kMaybeAssigned ? "true" : "false");
803 SNPrintF(buf + pos, ")");
804 PrintLiteralIndented(buf.begin(), value, true);
805 }
806 }
807
PrintIndentedVisit(const char * s,AstNode * node)808 void AstPrinter::PrintIndentedVisit(const char* s, AstNode* node) {
809 if (node != nullptr) {
810 IndentedScope indent(this, s, node->position());
811 Visit(node);
812 }
813 }
814
815
PrintProgram(FunctionLiteral * program)816 const char* AstPrinter::PrintProgram(FunctionLiteral* program) {
817 Init();
818 { IndentedScope indent(this, "FUNC", program->position());
819 PrintIndented("KIND");
820 Print(" %d\n", program->kind());
821 PrintIndented("LITERAL ID");
822 Print(" %d\n", program->function_literal_id());
823 PrintIndented("SUSPEND COUNT");
824 Print(" %d\n", program->suspend_count());
825 PrintLiteralIndented("NAME", program->raw_name(), true);
826 if (program->raw_inferred_name()) {
827 PrintLiteralIndented("INFERRED NAME", program->raw_inferred_name(), true);
828 }
829 if (program->requires_instance_members_initializer()) {
830 Print(" REQUIRES INSTANCE FIELDS INITIALIZER\n");
831 }
832 if (program->class_scope_has_private_brand()) {
833 Print(" CLASS SCOPE HAS PRIVATE BRAND\n");
834 }
835 if (program->has_static_private_methods_or_accessors()) {
836 Print(" HAS STATIC PRIVATE METHODS\n");
837 }
838 PrintParameters(program->scope());
839 PrintDeclarations(program->scope()->declarations());
840 PrintStatements(program->body());
841 }
842 return output_;
843 }
844
845
PrintOut(Isolate * isolate,AstNode * node)846 void AstPrinter::PrintOut(Isolate* isolate, AstNode* node) {
847 AstPrinter printer(isolate->stack_guard()->real_climit());
848 printer.Init();
849 printer.Visit(node);
850 PrintF("%s", printer.output_);
851 }
852
PrintDeclarations(Declaration::List * declarations)853 void AstPrinter::PrintDeclarations(Declaration::List* declarations) {
854 if (!declarations->is_empty()) {
855 IndentedScope indent(this, "DECLS");
856 for (Declaration* decl : *declarations) Visit(decl);
857 }
858 }
859
PrintParameters(DeclarationScope * scope)860 void AstPrinter::PrintParameters(DeclarationScope* scope) {
861 if (scope->num_parameters() > 0) {
862 IndentedScope indent(this, "PARAMS");
863 for (int i = 0; i < scope->num_parameters(); i++) {
864 PrintLiteralWithModeIndented("VAR", scope->parameter(i),
865 scope->parameter(i)->raw_name());
866 }
867 }
868 }
869
PrintStatements(const ZonePtrList<Statement> * statements)870 void AstPrinter::PrintStatements(const ZonePtrList<Statement>* statements) {
871 for (int i = 0; i < statements->length(); i++) {
872 Visit(statements->at(i));
873 }
874 }
875
PrintArguments(const ZonePtrList<Expression> * arguments)876 void AstPrinter::PrintArguments(const ZonePtrList<Expression>* arguments) {
877 for (int i = 0; i < arguments->length(); i++) {
878 Visit(arguments->at(i));
879 }
880 }
881
882
VisitBlock(Block * node)883 void AstPrinter::VisitBlock(Block* node) {
884 const char* block_txt =
885 node->ignore_completion_value() ? "BLOCK NOCOMPLETIONS" : "BLOCK";
886 IndentedScope indent(this, block_txt, node->position());
887 PrintStatements(node->statements());
888 }
889
890
891 // TODO(svenpanne) Start with IndentedScope.
VisitVariableDeclaration(VariableDeclaration * node)892 void AstPrinter::VisitVariableDeclaration(VariableDeclaration* node) {
893 PrintLiteralWithModeIndented("VARIABLE", node->var(),
894 node->var()->raw_name());
895 }
896
897
898 // TODO(svenpanne) Start with IndentedScope.
VisitFunctionDeclaration(FunctionDeclaration * node)899 void AstPrinter::VisitFunctionDeclaration(FunctionDeclaration* node) {
900 PrintIndented("FUNCTION ");
901 PrintLiteral(node->var()->raw_name(), true);
902 Print(" = function ");
903 PrintLiteral(node->fun()->raw_name(), false);
904 Print("\n");
905 }
906
907
VisitExpressionStatement(ExpressionStatement * node)908 void AstPrinter::VisitExpressionStatement(ExpressionStatement* node) {
909 IndentedScope indent(this, "EXPRESSION STATEMENT", node->position());
910 Visit(node->expression());
911 }
912
913
VisitEmptyStatement(EmptyStatement * node)914 void AstPrinter::VisitEmptyStatement(EmptyStatement* node) {
915 IndentedScope indent(this, "EMPTY", node->position());
916 }
917
918
VisitSloppyBlockFunctionStatement(SloppyBlockFunctionStatement * node)919 void AstPrinter::VisitSloppyBlockFunctionStatement(
920 SloppyBlockFunctionStatement* node) {
921 Visit(node->statement());
922 }
923
924
VisitIfStatement(IfStatement * node)925 void AstPrinter::VisitIfStatement(IfStatement* node) {
926 IndentedScope indent(this, "IF", node->position());
927 PrintIndentedVisit("CONDITION", node->condition());
928 PrintIndentedVisit("THEN", node->then_statement());
929 if (node->HasElseStatement()) {
930 PrintIndentedVisit("ELSE", node->else_statement());
931 }
932 }
933
934
VisitContinueStatement(ContinueStatement * node)935 void AstPrinter::VisitContinueStatement(ContinueStatement* node) {
936 IndentedScope indent(this, "CONTINUE", node->position());
937 }
938
939
VisitBreakStatement(BreakStatement * node)940 void AstPrinter::VisitBreakStatement(BreakStatement* node) {
941 IndentedScope indent(this, "BREAK", node->position());
942 }
943
944
VisitReturnStatement(ReturnStatement * node)945 void AstPrinter::VisitReturnStatement(ReturnStatement* node) {
946 IndentedScope indent(this, "RETURN", node->position());
947 Visit(node->expression());
948 }
949
950
VisitWithStatement(WithStatement * node)951 void AstPrinter::VisitWithStatement(WithStatement* node) {
952 IndentedScope indent(this, "WITH", node->position());
953 PrintIndentedVisit("OBJECT", node->expression());
954 PrintIndentedVisit("BODY", node->statement());
955 }
956
957
VisitSwitchStatement(SwitchStatement * node)958 void AstPrinter::VisitSwitchStatement(SwitchStatement* node) {
959 IndentedScope indent(this, "SWITCH", node->position());
960 PrintIndentedVisit("TAG", node->tag());
961 for (CaseClause* clause : *node->cases()) {
962 if (clause->is_default()) {
963 IndentedScope indent(this, "DEFAULT");
964 PrintStatements(clause->statements());
965 } else {
966 IndentedScope indent(this, "CASE");
967 Visit(clause->label());
968 PrintStatements(clause->statements());
969 }
970 }
971 }
972
973
VisitDoWhileStatement(DoWhileStatement * node)974 void AstPrinter::VisitDoWhileStatement(DoWhileStatement* node) {
975 IndentedScope indent(this, "DO", node->position());
976 PrintIndentedVisit("BODY", node->body());
977 PrintIndentedVisit("COND", node->cond());
978 }
979
980
VisitWhileStatement(WhileStatement * node)981 void AstPrinter::VisitWhileStatement(WhileStatement* node) {
982 IndentedScope indent(this, "WHILE", node->position());
983 PrintIndentedVisit("COND", node->cond());
984 PrintIndentedVisit("BODY", node->body());
985 }
986
987
VisitForStatement(ForStatement * node)988 void AstPrinter::VisitForStatement(ForStatement* node) {
989 IndentedScope indent(this, "FOR", node->position());
990 if (node->init()) PrintIndentedVisit("INIT", node->init());
991 if (node->cond()) PrintIndentedVisit("COND", node->cond());
992 PrintIndentedVisit("BODY", node->body());
993 if (node->next()) PrintIndentedVisit("NEXT", node->next());
994 }
995
996
VisitForInStatement(ForInStatement * node)997 void AstPrinter::VisitForInStatement(ForInStatement* node) {
998 IndentedScope indent(this, "FOR IN", node->position());
999 PrintIndentedVisit("FOR", node->each());
1000 PrintIndentedVisit("IN", node->subject());
1001 PrintIndentedVisit("BODY", node->body());
1002 }
1003
1004
VisitForOfStatement(ForOfStatement * node)1005 void AstPrinter::VisitForOfStatement(ForOfStatement* node) {
1006 IndentedScope indent(this, "FOR OF", node->position());
1007 const char* for_type;
1008 switch (node->type()) {
1009 case IteratorType::kNormal:
1010 for_type = "FOR";
1011 break;
1012 case IteratorType::kAsync:
1013 for_type = "FOR AWAIT";
1014 break;
1015 }
1016 PrintIndentedVisit(for_type, node->each());
1017 PrintIndentedVisit("OF", node->subject());
1018 PrintIndentedVisit("BODY", node->body());
1019 }
1020
1021
VisitTryCatchStatement(TryCatchStatement * node)1022 void AstPrinter::VisitTryCatchStatement(TryCatchStatement* node) {
1023 IndentedScope indent(this, "TRY CATCH", node->position());
1024 PrintIndentedVisit("TRY", node->try_block());
1025 PrintIndented("CATCH PREDICTION");
1026 const char* prediction = "";
1027 switch (node->GetCatchPrediction(HandlerTable::UNCAUGHT)) {
1028 case HandlerTable::UNCAUGHT:
1029 prediction = "UNCAUGHT";
1030 break;
1031 case HandlerTable::CAUGHT:
1032 prediction = "CAUGHT";
1033 break;
1034 case HandlerTable::DESUGARING:
1035 prediction = "DESUGARING";
1036 break;
1037 case HandlerTable::ASYNC_AWAIT:
1038 prediction = "ASYNC_AWAIT";
1039 break;
1040 case HandlerTable::UNCAUGHT_ASYNC_AWAIT:
1041 prediction = "UNCAUGHT_ASYNC_AWAIT";
1042 break;
1043 case HandlerTable::PROMISE:
1044 // Catch prediction resulting in promise rejections aren't
1045 // parsed by the parser.
1046 UNREACHABLE();
1047 }
1048 Print(" %s\n", prediction);
1049 if (node->scope()) {
1050 PrintLiteralWithModeIndented("CATCHVAR", node->scope()->catch_variable(),
1051 node->scope()->catch_variable()->raw_name());
1052 }
1053 PrintIndentedVisit("CATCH", node->catch_block());
1054 }
1055
VisitTryFinallyStatement(TryFinallyStatement * node)1056 void AstPrinter::VisitTryFinallyStatement(TryFinallyStatement* node) {
1057 IndentedScope indent(this, "TRY FINALLY", node->position());
1058 PrintIndentedVisit("TRY", node->try_block());
1059 PrintIndentedVisit("FINALLY", node->finally_block());
1060 }
1061
VisitDebuggerStatement(DebuggerStatement * node)1062 void AstPrinter::VisitDebuggerStatement(DebuggerStatement* node) {
1063 IndentedScope indent(this, "DEBUGGER", node->position());
1064 }
1065
1066
VisitFunctionLiteral(FunctionLiteral * node)1067 void AstPrinter::VisitFunctionLiteral(FunctionLiteral* node) {
1068 IndentedScope indent(this, "FUNC LITERAL", node->position());
1069 PrintIndented("LITERAL ID");
1070 Print(" %d\n", node->function_literal_id());
1071 PrintLiteralIndented("NAME", node->raw_name(), false);
1072 PrintLiteralIndented("INFERRED NAME", node->raw_inferred_name(), false);
1073 // We don't want to see the function literal in this case: it
1074 // will be printed via PrintProgram when the code for it is
1075 // generated.
1076 // PrintParameters(node->scope());
1077 // PrintStatements(node->body());
1078 }
1079
1080
VisitClassLiteral(ClassLiteral * node)1081 void AstPrinter::VisitClassLiteral(ClassLiteral* node) {
1082 IndentedScope indent(this, "CLASS LITERAL", node->position());
1083 PrintLiteralIndented("NAME", node->constructor()->raw_name(), false);
1084 if (node->extends() != nullptr) {
1085 PrintIndentedVisit("EXTENDS", node->extends());
1086 }
1087 Scope* outer = node->constructor()->scope()->outer_scope();
1088 if (outer->is_class_scope()) {
1089 Variable* brand = outer->AsClassScope()->brand();
1090 if (brand != nullptr) {
1091 PrintLiteralWithModeIndented("BRAND", brand, brand->raw_name());
1092 }
1093 }
1094 if (node->static_fields_initializer() != nullptr) {
1095 PrintIndentedVisit("STATIC FIELDS INITIALIZER",
1096 node->static_fields_initializer());
1097 }
1098 if (node->instance_members_initializer_function() != nullptr) {
1099 PrintIndentedVisit("INSTANCE MEMBERS INITIALIZER",
1100 node->instance_members_initializer_function());
1101 }
1102 PrintClassProperties(node->private_members());
1103 PrintClassProperties(node->public_members());
1104 }
1105
VisitInitializeClassMembersStatement(InitializeClassMembersStatement * node)1106 void AstPrinter::VisitInitializeClassMembersStatement(
1107 InitializeClassMembersStatement* node) {
1108 IndentedScope indent(this, "INITIALIZE CLASS MEMBERS", node->position());
1109 PrintClassProperties(node->fields());
1110 }
1111
PrintClassProperties(const ZonePtrList<ClassLiteral::Property> * properties)1112 void AstPrinter::PrintClassProperties(
1113 const ZonePtrList<ClassLiteral::Property>* properties) {
1114 for (int i = 0; i < properties->length(); i++) {
1115 ClassLiteral::Property* property = properties->at(i);
1116 const char* prop_kind = nullptr;
1117 switch (property->kind()) {
1118 case ClassLiteral::Property::METHOD:
1119 prop_kind = "METHOD";
1120 break;
1121 case ClassLiteral::Property::GETTER:
1122 prop_kind = "GETTER";
1123 break;
1124 case ClassLiteral::Property::SETTER:
1125 prop_kind = "SETTER";
1126 break;
1127 case ClassLiteral::Property::FIELD:
1128 prop_kind = "FIELD";
1129 break;
1130 }
1131 EmbeddedVector<char, 128> buf;
1132 SNPrintF(buf, "PROPERTY%s%s - %s", property->is_static() ? " - STATIC" : "",
1133 property->is_private() ? " - PRIVATE" : " - PUBLIC", prop_kind);
1134 IndentedScope prop(this, buf.begin());
1135 PrintIndentedVisit("KEY", properties->at(i)->key());
1136 PrintIndentedVisit("VALUE", properties->at(i)->value());
1137 }
1138 }
1139
1140
VisitNativeFunctionLiteral(NativeFunctionLiteral * node)1141 void AstPrinter::VisitNativeFunctionLiteral(NativeFunctionLiteral* node) {
1142 IndentedScope indent(this, "NATIVE FUNC LITERAL", node->position());
1143 PrintLiteralIndented("NAME", node->raw_name(), false);
1144 }
1145
1146
VisitConditional(Conditional * node)1147 void AstPrinter::VisitConditional(Conditional* node) {
1148 IndentedScope indent(this, "CONDITIONAL", node->position());
1149 PrintIndentedVisit("CONDITION", node->condition());
1150 PrintIndentedVisit("THEN", node->then_expression());
1151 PrintIndentedVisit("ELSE", node->else_expression());
1152 }
1153
1154
VisitLiteral(Literal * node)1155 void AstPrinter::VisitLiteral(Literal* node) {
1156 PrintLiteralIndented("LITERAL", node, true);
1157 }
1158
1159
VisitRegExpLiteral(RegExpLiteral * node)1160 void AstPrinter::VisitRegExpLiteral(RegExpLiteral* node) {
1161 IndentedScope indent(this, "REGEXP LITERAL", node->position());
1162 PrintLiteralIndented("PATTERN", node->raw_pattern(), false);
1163 int i = 0;
1164 EmbeddedVector<char, 128> buf;
1165 if (node->flags() & RegExp::kGlobal) buf[i++] = 'g';
1166 if (node->flags() & RegExp::kIgnoreCase) buf[i++] = 'i';
1167 if (node->flags() & RegExp::kLinear) buf[i++] = 'l';
1168 if (node->flags() & RegExp::kMultiline) buf[i++] = 'm';
1169 if (node->flags() & RegExp::kUnicode) buf[i++] = 'u';
1170 if (node->flags() & RegExp::kSticky) buf[i++] = 'y';
1171 buf[i] = '\0';
1172 PrintIndented("FLAGS ");
1173 Print("%s", buf.begin());
1174 Print("\n");
1175 }
1176
1177
VisitObjectLiteral(ObjectLiteral * node)1178 void AstPrinter::VisitObjectLiteral(ObjectLiteral* node) {
1179 IndentedScope indent(this, "OBJ LITERAL", node->position());
1180 PrintObjectProperties(node->properties());
1181 }
1182
PrintObjectProperties(const ZonePtrList<ObjectLiteral::Property> * properties)1183 void AstPrinter::PrintObjectProperties(
1184 const ZonePtrList<ObjectLiteral::Property>* properties) {
1185 for (int i = 0; i < properties->length(); i++) {
1186 ObjectLiteral::Property* property = properties->at(i);
1187 const char* prop_kind = nullptr;
1188 switch (property->kind()) {
1189 case ObjectLiteral::Property::CONSTANT:
1190 prop_kind = "CONSTANT";
1191 break;
1192 case ObjectLiteral::Property::COMPUTED:
1193 prop_kind = "COMPUTED";
1194 break;
1195 case ObjectLiteral::Property::MATERIALIZED_LITERAL:
1196 prop_kind = "MATERIALIZED_LITERAL";
1197 break;
1198 case ObjectLiteral::Property::PROTOTYPE:
1199 prop_kind = "PROTOTYPE";
1200 break;
1201 case ObjectLiteral::Property::GETTER:
1202 prop_kind = "GETTER";
1203 break;
1204 case ObjectLiteral::Property::SETTER:
1205 prop_kind = "SETTER";
1206 break;
1207 case ObjectLiteral::Property::SPREAD:
1208 prop_kind = "SPREAD";
1209 break;
1210 }
1211 EmbeddedVector<char, 128> buf;
1212 SNPrintF(buf, "PROPERTY - %s", prop_kind);
1213 IndentedScope prop(this, buf.begin());
1214 PrintIndentedVisit("KEY", properties->at(i)->key());
1215 PrintIndentedVisit("VALUE", properties->at(i)->value());
1216 }
1217 }
1218
1219
VisitArrayLiteral(ArrayLiteral * node)1220 void AstPrinter::VisitArrayLiteral(ArrayLiteral* node) {
1221 IndentedScope indent(this, "ARRAY LITERAL", node->position());
1222 if (node->values()->length() > 0) {
1223 IndentedScope indent(this, "VALUES", node->position());
1224 for (int i = 0; i < node->values()->length(); i++) {
1225 Visit(node->values()->at(i));
1226 }
1227 }
1228 }
1229
1230
VisitVariableProxy(VariableProxy * node)1231 void AstPrinter::VisitVariableProxy(VariableProxy* node) {
1232 EmbeddedVector<char, 128> buf;
1233 int pos = SNPrintF(buf, "VAR PROXY");
1234
1235 if (!node->is_resolved()) {
1236 SNPrintF(buf + pos, " unresolved");
1237 PrintLiteralWithModeIndented(buf.begin(), nullptr, node->raw_name());
1238 } else {
1239 Variable* var = node->var();
1240 switch (var->location()) {
1241 case VariableLocation::UNALLOCATED:
1242 SNPrintF(buf + pos, " unallocated");
1243 break;
1244 case VariableLocation::PARAMETER:
1245 SNPrintF(buf + pos, " parameter[%d]", var->index());
1246 break;
1247 case VariableLocation::LOCAL:
1248 SNPrintF(buf + pos, " local[%d]", var->index());
1249 break;
1250 case VariableLocation::CONTEXT:
1251 SNPrintF(buf + pos, " context[%d]", var->index());
1252 break;
1253 case VariableLocation::LOOKUP:
1254 SNPrintF(buf + pos, " lookup");
1255 break;
1256 case VariableLocation::MODULE:
1257 SNPrintF(buf + pos, " module");
1258 break;
1259 case VariableLocation::REPL_GLOBAL:
1260 SNPrintF(buf + pos, " repl global[%d]", var->index());
1261 break;
1262 }
1263 PrintLiteralWithModeIndented(buf.begin(), var, node->raw_name());
1264 }
1265 }
1266
1267
VisitAssignment(Assignment * node)1268 void AstPrinter::VisitAssignment(Assignment* node) {
1269 IndentedScope indent(this, Token::Name(node->op()), node->position());
1270 Visit(node->target());
1271 Visit(node->value());
1272 }
1273
VisitCompoundAssignment(CompoundAssignment * node)1274 void AstPrinter::VisitCompoundAssignment(CompoundAssignment* node) {
1275 VisitAssignment(node);
1276 }
1277
VisitYield(Yield * node)1278 void AstPrinter::VisitYield(Yield* node) {
1279 EmbeddedVector<char, 128> buf;
1280 SNPrintF(buf, "YIELD");
1281 IndentedScope indent(this, buf.begin(), node->position());
1282 Visit(node->expression());
1283 }
1284
VisitYieldStar(YieldStar * node)1285 void AstPrinter::VisitYieldStar(YieldStar* node) {
1286 EmbeddedVector<char, 128> buf;
1287 SNPrintF(buf, "YIELD_STAR");
1288 IndentedScope indent(this, buf.begin(), node->position());
1289 Visit(node->expression());
1290 }
1291
VisitAwait(Await * node)1292 void AstPrinter::VisitAwait(Await* node) {
1293 EmbeddedVector<char, 128> buf;
1294 SNPrintF(buf, "AWAIT");
1295 IndentedScope indent(this, buf.begin(), node->position());
1296 Visit(node->expression());
1297 }
1298
VisitThrow(Throw * node)1299 void AstPrinter::VisitThrow(Throw* node) {
1300 IndentedScope indent(this, "THROW", node->position());
1301 Visit(node->exception());
1302 }
1303
VisitOptionalChain(OptionalChain * node)1304 void AstPrinter::VisitOptionalChain(OptionalChain* node) {
1305 IndentedScope indent(this, "OPTIONAL_CHAIN", node->position());
1306 Visit(node->expression());
1307 }
1308
VisitProperty(Property * node)1309 void AstPrinter::VisitProperty(Property* node) {
1310 EmbeddedVector<char, 128> buf;
1311 SNPrintF(buf, "PROPERTY");
1312 IndentedScope indent(this, buf.begin(), node->position());
1313
1314 Visit(node->obj());
1315 AssignType type = Property::GetAssignType(node);
1316 switch (type) {
1317 case NAMED_PROPERTY:
1318 case NAMED_SUPER_PROPERTY: {
1319 PrintLiteralIndented("NAME", node->key()->AsLiteral(), false);
1320 break;
1321 }
1322 case PRIVATE_METHOD: {
1323 PrintIndentedVisit("PRIVATE_METHOD", node->key());
1324 break;
1325 }
1326 case PRIVATE_GETTER_ONLY: {
1327 PrintIndentedVisit("PRIVATE_GETTER_ONLY", node->key());
1328 break;
1329 }
1330 case PRIVATE_SETTER_ONLY: {
1331 PrintIndentedVisit("PRIVATE_SETTER_ONLY", node->key());
1332 break;
1333 }
1334 case PRIVATE_GETTER_AND_SETTER: {
1335 PrintIndentedVisit("PRIVATE_GETTER_AND_SETTER", node->key());
1336 break;
1337 }
1338 case KEYED_PROPERTY:
1339 case KEYED_SUPER_PROPERTY: {
1340 PrintIndentedVisit("KEY", node->key());
1341 break;
1342 }
1343 case NON_PROPERTY:
1344 UNREACHABLE();
1345 }
1346 }
1347
VisitCall(Call * node)1348 void AstPrinter::VisitCall(Call* node) {
1349 EmbeddedVector<char, 128> buf;
1350 SNPrintF(buf, "CALL");
1351 IndentedScope indent(this, buf.begin());
1352
1353 Visit(node->expression());
1354 PrintArguments(node->arguments());
1355 }
1356
1357
VisitCallNew(CallNew * node)1358 void AstPrinter::VisitCallNew(CallNew* node) {
1359 IndentedScope indent(this, "CALL NEW", node->position());
1360 Visit(node->expression());
1361 PrintArguments(node->arguments());
1362 }
1363
1364
VisitCallRuntime(CallRuntime * node)1365 void AstPrinter::VisitCallRuntime(CallRuntime* node) {
1366 EmbeddedVector<char, 128> buf;
1367 SNPrintF(buf, "CALL RUNTIME %s%s", node->debug_name(),
1368 node->is_jsruntime() ? " (JS function)" : "");
1369 IndentedScope indent(this, buf.begin(), node->position());
1370 PrintArguments(node->arguments());
1371 }
1372
1373
VisitUnaryOperation(UnaryOperation * node)1374 void AstPrinter::VisitUnaryOperation(UnaryOperation* node) {
1375 IndentedScope indent(this, Token::Name(node->op()), node->position());
1376 Visit(node->expression());
1377 }
1378
1379
VisitCountOperation(CountOperation * node)1380 void AstPrinter::VisitCountOperation(CountOperation* node) {
1381 EmbeddedVector<char, 128> buf;
1382 SNPrintF(buf, "%s %s", (node->is_prefix() ? "PRE" : "POST"),
1383 Token::Name(node->op()));
1384 IndentedScope indent(this, buf.begin(), node->position());
1385 Visit(node->expression());
1386 }
1387
1388
VisitBinaryOperation(BinaryOperation * node)1389 void AstPrinter::VisitBinaryOperation(BinaryOperation* node) {
1390 IndentedScope indent(this, Token::Name(node->op()), node->position());
1391 Visit(node->left());
1392 Visit(node->right());
1393 }
1394
VisitNaryOperation(NaryOperation * node)1395 void AstPrinter::VisitNaryOperation(NaryOperation* node) {
1396 IndentedScope indent(this, Token::Name(node->op()), node->position());
1397 Visit(node->first());
1398 for (size_t i = 0; i < node->subsequent_length(); ++i) {
1399 Visit(node->subsequent(i));
1400 }
1401 }
1402
VisitCompareOperation(CompareOperation * node)1403 void AstPrinter::VisitCompareOperation(CompareOperation* node) {
1404 IndentedScope indent(this, Token::Name(node->op()), node->position());
1405 Visit(node->left());
1406 Visit(node->right());
1407 }
1408
1409
VisitSpread(Spread * node)1410 void AstPrinter::VisitSpread(Spread* node) {
1411 IndentedScope indent(this, "SPREAD", node->position());
1412 Visit(node->expression());
1413 }
1414
VisitEmptyParentheses(EmptyParentheses * node)1415 void AstPrinter::VisitEmptyParentheses(EmptyParentheses* node) {
1416 IndentedScope indent(this, "()", node->position());
1417 }
1418
VisitGetTemplateObject(GetTemplateObject * node)1419 void AstPrinter::VisitGetTemplateObject(GetTemplateObject* node) {
1420 IndentedScope indent(this, "GET-TEMPLATE-OBJECT", node->position());
1421 }
1422
VisitTemplateLiteral(TemplateLiteral * node)1423 void AstPrinter::VisitTemplateLiteral(TemplateLiteral* node) {
1424 IndentedScope indent(this, "TEMPLATE-LITERAL", node->position());
1425 const AstRawString* string = node->string_parts()->first();
1426 if (!string->IsEmpty()) PrintLiteralIndented("SPAN", string, true);
1427 for (int i = 0; i < node->substitutions()->length();) {
1428 PrintIndentedVisit("EXPR", node->substitutions()->at(i++));
1429 if (i < node->string_parts()->length()) {
1430 string = node->string_parts()->at(i);
1431 if (!string->IsEmpty()) PrintLiteralIndented("SPAN", string, true);
1432 }
1433 }
1434 }
1435
VisitImportCallExpression(ImportCallExpression * node)1436 void AstPrinter::VisitImportCallExpression(ImportCallExpression* node) {
1437 IndentedScope indent(this, "IMPORT-CALL", node->position());
1438 Visit(node->argument());
1439 }
1440
VisitThisExpression(ThisExpression * node)1441 void AstPrinter::VisitThisExpression(ThisExpression* node) {
1442 IndentedScope indent(this, "THIS-EXPRESSION", node->position());
1443 }
1444
VisitSuperPropertyReference(SuperPropertyReference * node)1445 void AstPrinter::VisitSuperPropertyReference(SuperPropertyReference* node) {
1446 IndentedScope indent(this, "SUPER-PROPERTY-REFERENCE", node->position());
1447 }
1448
1449
VisitSuperCallReference(SuperCallReference * node)1450 void AstPrinter::VisitSuperCallReference(SuperCallReference* node) {
1451 IndentedScope indent(this, "SUPER-CALL-REFERENCE", node->position());
1452 }
1453
1454
1455 #endif // DEBUG
1456
1457 } // namespace internal
1458 } // namespace v8
1459