• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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