1 // Copyright 2011 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/messages.h"
6 
7 #include <memory>
8 
9 #include "src/api-inl.h"
10 #include "src/execution.h"
11 #include "src/isolate-inl.h"
12 #include "src/keys.h"
13 #include "src/objects/frame-array-inl.h"
14 #include "src/objects/js-array-inl.h"
15 #include "src/string-builder-inl.h"
16 #include "src/wasm/wasm-code-manager.h"
17 #include "src/wasm/wasm-objects.h"
18 
19 namespace v8 {
20 namespace internal {
21 
MessageLocation(Handle<Script> script,int start_pos,int end_pos)22 MessageLocation::MessageLocation(Handle<Script> script, int start_pos,
23                                  int end_pos)
24     : script_(script), start_pos_(start_pos), end_pos_(end_pos) {}
MessageLocation(Handle<Script> script,int start_pos,int end_pos,Handle<SharedFunctionInfo> shared)25 MessageLocation::MessageLocation(Handle<Script> script, int start_pos,
26                                  int end_pos, Handle<SharedFunctionInfo> shared)
27     : script_(script),
28       start_pos_(start_pos),
29       end_pos_(end_pos),
30       shared_(shared) {}
MessageLocation()31 MessageLocation::MessageLocation() : start_pos_(-1), end_pos_(-1) {}
32 
33 // If no message listeners have been registered this one is called
34 // by default.
DefaultMessageReport(Isolate * isolate,const MessageLocation * loc,Handle<Object> message_obj)35 void MessageHandler::DefaultMessageReport(Isolate* isolate,
36                                           const MessageLocation* loc,
37                                           Handle<Object> message_obj) {
38   std::unique_ptr<char[]> str = GetLocalizedMessage(isolate, message_obj);
39   if (loc == nullptr) {
40     PrintF("%s\n", str.get());
41   } else {
42     HandleScope scope(isolate);
43     Handle<Object> data(loc->script()->name(), isolate);
44     std::unique_ptr<char[]> data_str;
45     if (data->IsString())
46       data_str = Handle<String>::cast(data)->ToCString(DISALLOW_NULLS);
47     PrintF("%s:%i: %s\n", data_str.get() ? data_str.get() : "<unknown>",
48            loc->start_pos(), str.get());
49   }
50 }
51 
MakeMessageObject(Isolate * isolate,MessageTemplate::Template message,const MessageLocation * location,Handle<Object> argument,Handle<FixedArray> stack_frames)52 Handle<JSMessageObject> MessageHandler::MakeMessageObject(
53     Isolate* isolate, MessageTemplate::Template message,
54     const MessageLocation* location, Handle<Object> argument,
55     Handle<FixedArray> stack_frames) {
56   Factory* factory = isolate->factory();
57 
58   int start = -1;
59   int end = -1;
60   Handle<Script> script_handle = isolate->factory()->empty_script();
61   if (location != nullptr) {
62     start = location->start_pos();
63     end = location->end_pos();
64     script_handle = location->script();
65   }
66 
67   Handle<Object> stack_frames_handle = stack_frames.is_null()
68       ? Handle<Object>::cast(factory->undefined_value())
69       : Handle<Object>::cast(stack_frames);
70 
71   Handle<JSMessageObject> message_obj = factory->NewJSMessageObject(
72       message, argument, start, end, script_handle, stack_frames_handle);
73 
74   return message_obj;
75 }
76 
ReportMessage(Isolate * isolate,const MessageLocation * loc,Handle<JSMessageObject> message)77 void MessageHandler::ReportMessage(Isolate* isolate, const MessageLocation* loc,
78                                    Handle<JSMessageObject> message) {
79   v8::Local<v8::Message> api_message_obj = v8::Utils::MessageToLocal(message);
80 
81   if (api_message_obj->ErrorLevel() == v8::Isolate::kMessageError) {
82     // We are calling into embedder's code which can throw exceptions.
83     // Thus we need to save current exception state, reset it to the clean one
84     // and ignore scheduled exceptions callbacks can throw.
85 
86     // We pass the exception object into the message handler callback though.
87     Object* exception_object = ReadOnlyRoots(isolate).undefined_value();
88     if (isolate->has_pending_exception()) {
89       exception_object = isolate->pending_exception();
90     }
91     Handle<Object> exception(exception_object, isolate);
92 
93     Isolate::ExceptionScope exception_scope(isolate);
94     isolate->clear_pending_exception();
95     isolate->set_external_caught_exception(false);
96 
97     // Turn the exception on the message into a string if it is an object.
98     if (message->argument()->IsJSObject()) {
99       HandleScope scope(isolate);
100       Handle<Object> argument(message->argument(), isolate);
101 
102       MaybeHandle<Object> maybe_stringified;
103       Handle<Object> stringified;
104       // Make sure we don't leak uncaught internally generated Error objects.
105       if (argument->IsJSError()) {
106         maybe_stringified = Object::NoSideEffectsToString(isolate, argument);
107       } else {
108         v8::TryCatch catcher(reinterpret_cast<v8::Isolate*>(isolate));
109         catcher.SetVerbose(false);
110         catcher.SetCaptureMessage(false);
111 
112         maybe_stringified = Object::ToString(isolate, argument);
113       }
114 
115       if (!maybe_stringified.ToHandle(&stringified)) {
116         DCHECK(isolate->has_pending_exception());
117         isolate->clear_pending_exception();
118         isolate->set_external_caught_exception(false);
119         stringified =
120             isolate->factory()->NewStringFromAsciiChecked("exception");
121       }
122       message->set_argument(*stringified);
123     }
124 
125     v8::Local<v8::Value> api_exception_obj = v8::Utils::ToLocal(exception);
126     ReportMessageNoExceptions(isolate, loc, message, api_exception_obj);
127   } else {
128     ReportMessageNoExceptions(isolate, loc, message, v8::Local<v8::Value>());
129   }
130 }
131 
ReportMessageNoExceptions(Isolate * isolate,const MessageLocation * loc,Handle<Object> message,v8::Local<v8::Value> api_exception_obj)132 void MessageHandler::ReportMessageNoExceptions(
133     Isolate* isolate, const MessageLocation* loc, Handle<Object> message,
134     v8::Local<v8::Value> api_exception_obj) {
135   v8::Local<v8::Message> api_message_obj = v8::Utils::MessageToLocal(message);
136   int error_level = api_message_obj->ErrorLevel();
137 
138   Handle<TemplateList> global_listeners =
139       isolate->factory()->message_listeners();
140   int global_length = global_listeners->length();
141   if (global_length == 0) {
142     DefaultMessageReport(isolate, loc, message);
143     if (isolate->has_scheduled_exception()) {
144       isolate->clear_scheduled_exception();
145     }
146   } else {
147     for (int i = 0; i < global_length; i++) {
148       HandleScope scope(isolate);
149       if (global_listeners->get(i)->IsUndefined(isolate)) continue;
150       FixedArray* listener = FixedArray::cast(global_listeners->get(i));
151       Foreign* callback_obj = Foreign::cast(listener->get(0));
152       int32_t message_levels =
153           static_cast<int32_t>(Smi::ToInt(listener->get(2)));
154       if (!(message_levels & error_level)) {
155         continue;
156       }
157       v8::MessageCallback callback =
158           FUNCTION_CAST<v8::MessageCallback>(callback_obj->foreign_address());
159       Handle<Object> callback_data(listener->get(1), isolate);
160       {
161         // Do not allow exceptions to propagate.
162         v8::TryCatch try_catch(reinterpret_cast<v8::Isolate*>(isolate));
163         callback(api_message_obj, callback_data->IsUndefined(isolate)
164                                       ? api_exception_obj
165                                       : v8::Utils::ToLocal(callback_data));
166       }
167       if (isolate->has_scheduled_exception()) {
168         isolate->clear_scheduled_exception();
169       }
170     }
171   }
172 }
173 
174 
GetMessage(Isolate * isolate,Handle<Object> data)175 Handle<String> MessageHandler::GetMessage(Isolate* isolate,
176                                           Handle<Object> data) {
177   Handle<JSMessageObject> message = Handle<JSMessageObject>::cast(data);
178   Handle<Object> arg = Handle<Object>(message->argument(), isolate);
179   return MessageTemplate::FormatMessage(isolate, message->type(), arg);
180 }
181 
GetLocalizedMessage(Isolate * isolate,Handle<Object> data)182 std::unique_ptr<char[]> MessageHandler::GetLocalizedMessage(
183     Isolate* isolate, Handle<Object> data) {
184   HandleScope scope(isolate);
185   return GetMessage(isolate, data)->ToCString(DISALLOW_NULLS);
186 }
187 
188 namespace {
189 
EvalFromFunctionName(Isolate * isolate,Handle<Script> script)190 Object* EvalFromFunctionName(Isolate* isolate, Handle<Script> script) {
191   if (!script->has_eval_from_shared())
192     return ReadOnlyRoots(isolate).undefined_value();
193 
194   Handle<SharedFunctionInfo> shared(script->eval_from_shared(), isolate);
195   // Find the name of the function calling eval.
196   if (shared->Name()->BooleanValue(isolate)) {
197     return shared->Name();
198   }
199 
200   return shared->inferred_name();
201 }
202 
EvalFromScript(Isolate * isolate,Handle<Script> script)203 Object* EvalFromScript(Isolate* isolate, Handle<Script> script) {
204   if (!script->has_eval_from_shared())
205     return ReadOnlyRoots(isolate).undefined_value();
206 
207   Handle<SharedFunctionInfo> eval_from_shared(script->eval_from_shared(),
208                                               isolate);
209   return eval_from_shared->script()->IsScript()
210              ? eval_from_shared->script()
211              : ReadOnlyRoots(isolate).undefined_value();
212 }
213 
FormatEvalOrigin(Isolate * isolate,Handle<Script> script)214 MaybeHandle<String> FormatEvalOrigin(Isolate* isolate, Handle<Script> script) {
215   Handle<Object> sourceURL(script->GetNameOrSourceURL(), isolate);
216   if (!sourceURL->IsUndefined(isolate)) {
217     DCHECK(sourceURL->IsString());
218     return Handle<String>::cast(sourceURL);
219   }
220 
221   IncrementalStringBuilder builder(isolate);
222   builder.AppendCString("eval at ");
223 
224   Handle<Object> eval_from_function_name =
225       handle(EvalFromFunctionName(isolate, script), isolate);
226   if (eval_from_function_name->BooleanValue(isolate)) {
227     Handle<String> str;
228     ASSIGN_RETURN_ON_EXCEPTION(
229         isolate, str, Object::ToString(isolate, eval_from_function_name),
230         String);
231     builder.AppendString(str);
232   } else {
233     builder.AppendCString("<anonymous>");
234   }
235 
236   Handle<Object> eval_from_script_obj =
237       handle(EvalFromScript(isolate, script), isolate);
238   if (eval_from_script_obj->IsScript()) {
239     Handle<Script> eval_from_script =
240         Handle<Script>::cast(eval_from_script_obj);
241     builder.AppendCString(" (");
242     if (eval_from_script->compilation_type() == Script::COMPILATION_TYPE_EVAL) {
243       // Eval script originated from another eval.
244       Handle<String> str;
245       ASSIGN_RETURN_ON_EXCEPTION(
246           isolate, str, FormatEvalOrigin(isolate, eval_from_script), String);
247       builder.AppendString(str);
248     } else {
249       DCHECK(eval_from_script->compilation_type() !=
250              Script::COMPILATION_TYPE_EVAL);
251       // eval script originated from "real" source.
252       Handle<Object> name_obj = handle(eval_from_script->name(), isolate);
253       if (eval_from_script->name()->IsString()) {
254         builder.AppendString(Handle<String>::cast(name_obj));
255 
256         Script::PositionInfo info;
257         if (Script::GetPositionInfo(eval_from_script, script->GetEvalPosition(),
258                                     &info, Script::NO_OFFSET)) {
259           builder.AppendCString(":");
260 
261           Handle<String> str = isolate->factory()->NumberToString(
262               handle(Smi::FromInt(info.line + 1), isolate));
263           builder.AppendString(str);
264 
265           builder.AppendCString(":");
266 
267           str = isolate->factory()->NumberToString(
268               handle(Smi::FromInt(info.column + 1), isolate));
269           builder.AppendString(str);
270         }
271       } else {
272         DCHECK(!eval_from_script->name()->IsString());
273         builder.AppendCString("unknown source");
274       }
275     }
276     builder.AppendCString(")");
277   }
278 
279   Handle<String> result;
280   ASSIGN_RETURN_ON_EXCEPTION(isolate, result, builder.Finish(), String);
281   return result;
282 }
283 
284 }  // namespace
285 
GetEvalOrigin()286 Handle<Object> StackFrameBase::GetEvalOrigin() {
287   if (!HasScript()) return isolate_->factory()->undefined_value();
288   return FormatEvalOrigin(isolate_, GetScript()).ToHandleChecked();
289 }
290 
IsEval()291 bool StackFrameBase::IsEval() {
292   return HasScript() &&
293          GetScript()->compilation_type() == Script::COMPILATION_TYPE_EVAL;
294 }
295 
FromFrameArray(Isolate * isolate,Handle<FrameArray> array,int frame_ix)296 void JSStackFrame::FromFrameArray(Isolate* isolate, Handle<FrameArray> array,
297                                   int frame_ix) {
298   DCHECK(!array->IsWasmFrame(frame_ix));
299   isolate_ = isolate;
300   receiver_ = handle(array->Receiver(frame_ix), isolate);
301   function_ = handle(array->Function(frame_ix), isolate);
302   code_ = handle(array->Code(frame_ix), isolate);
303   offset_ = array->Offset(frame_ix)->value();
304 
305   const int flags = array->Flags(frame_ix)->value();
306   is_constructor_ = (flags & FrameArray::kIsConstructor) != 0;
307   is_strict_ = (flags & FrameArray::kIsStrict) != 0;
308 }
309 
JSStackFrame()310 JSStackFrame::JSStackFrame() {}
311 
JSStackFrame(Isolate * isolate,Handle<Object> receiver,Handle<JSFunction> function,Handle<AbstractCode> code,int offset)312 JSStackFrame::JSStackFrame(Isolate* isolate, Handle<Object> receiver,
313                            Handle<JSFunction> function,
314                            Handle<AbstractCode> code, int offset)
315     : StackFrameBase(isolate),
316       receiver_(receiver),
317       function_(function),
318       code_(code),
319       offset_(offset),
320       is_constructor_(false),
321       is_strict_(false) {}
322 
GetFunction() const323 Handle<Object> JSStackFrame::GetFunction() const {
324   return Handle<Object>::cast(function_);
325 }
326 
GetFileName()327 Handle<Object> JSStackFrame::GetFileName() {
328   if (!HasScript()) return isolate_->factory()->null_value();
329   return handle(GetScript()->name(), isolate_);
330 }
331 
GetFunctionName()332 Handle<Object> JSStackFrame::GetFunctionName() {
333   Handle<String> result = JSFunction::GetName(function_);
334   if (result->length() != 0) return result;
335 
336   if (HasScript() &&
337       GetScript()->compilation_type() == Script::COMPILATION_TYPE_EVAL) {
338     return isolate_->factory()->eval_string();
339   }
340   return isolate_->factory()->null_value();
341 }
342 
343 namespace {
344 
CheckMethodName(Isolate * isolate,Handle<JSReceiver> receiver,Handle<Name> name,Handle<JSFunction> fun,LookupIterator::Configuration config)345 bool CheckMethodName(Isolate* isolate, Handle<JSReceiver> receiver,
346                      Handle<Name> name, Handle<JSFunction> fun,
347                      LookupIterator::Configuration config) {
348   LookupIterator iter =
349       LookupIterator::PropertyOrElement(isolate, receiver, name, config);
350   if (iter.state() == LookupIterator::DATA) {
351     return iter.GetDataValue().is_identical_to(fun);
352   } else if (iter.state() == LookupIterator::ACCESSOR) {
353     Handle<Object> accessors = iter.GetAccessors();
354     if (accessors->IsAccessorPair()) {
355       Handle<AccessorPair> pair = Handle<AccessorPair>::cast(accessors);
356       return pair->getter() == *fun || pair->setter() == *fun;
357     }
358   }
359   return false;
360 }
361 
ScriptNameOrSourceUrl(Handle<Script> script,Isolate * isolate)362 Handle<Object> ScriptNameOrSourceUrl(Handle<Script> script, Isolate* isolate) {
363   Object* name_or_url = script->source_url();
364   if (!name_or_url->IsString()) name_or_url = script->name();
365   return handle(name_or_url, isolate);
366 }
367 
368 }  // namespace
369 
GetScriptNameOrSourceUrl()370 Handle<Object> JSStackFrame::GetScriptNameOrSourceUrl() {
371   if (!HasScript()) return isolate_->factory()->null_value();
372   return ScriptNameOrSourceUrl(GetScript(), isolate_);
373 }
374 
GetMethodName()375 Handle<Object> JSStackFrame::GetMethodName() {
376   if (receiver_->IsNullOrUndefined(isolate_)) {
377     return isolate_->factory()->null_value();
378   }
379 
380   Handle<JSReceiver> receiver;
381   if (!Object::ToObject(isolate_, receiver_).ToHandle(&receiver)) {
382     DCHECK(isolate_->has_pending_exception());
383     isolate_->clear_pending_exception();
384     isolate_->set_external_caught_exception(false);
385     return isolate_->factory()->null_value();
386   }
387 
388   Handle<String> name(function_->shared()->Name(), isolate_);
389   // ES2015 gives getters and setters name prefixes which must
390   // be stripped to find the property name.
391   if (name->IsUtf8EqualTo(CStrVector("get "), true) ||
392       name->IsUtf8EqualTo(CStrVector("set "), true)) {
393     name = isolate_->factory()->NewProperSubString(name, 4, name->length());
394   }
395   if (CheckMethodName(isolate_, receiver, name, function_,
396                       LookupIterator::PROTOTYPE_CHAIN_SKIP_INTERCEPTOR)) {
397     return name;
398   }
399 
400   HandleScope outer_scope(isolate_);
401   Handle<Object> result;
402   for (PrototypeIterator iter(isolate_, receiver, kStartAtReceiver);
403        !iter.IsAtEnd(); iter.Advance()) {
404     Handle<Object> current = PrototypeIterator::GetCurrent(iter);
405     if (!current->IsJSObject()) break;
406     Handle<JSObject> current_obj = Handle<JSObject>::cast(current);
407     if (current_obj->IsAccessCheckNeeded()) break;
408     Handle<FixedArray> keys =
409         KeyAccumulator::GetOwnEnumPropertyKeys(isolate_, current_obj);
410     for (int i = 0; i < keys->length(); i++) {
411       HandleScope inner_scope(isolate_);
412       if (!keys->get(i)->IsName()) continue;
413       Handle<Name> name_key(Name::cast(keys->get(i)), isolate_);
414       if (!CheckMethodName(isolate_, current_obj, name_key, function_,
415                            LookupIterator::OWN_SKIP_INTERCEPTOR))
416         continue;
417       // Return null in case of duplicates to avoid confusion.
418       if (!result.is_null()) return isolate_->factory()->null_value();
419       result = inner_scope.CloseAndEscape(name_key);
420     }
421   }
422 
423   if (!result.is_null()) return outer_scope.CloseAndEscape(result);
424   return isolate_->factory()->null_value();
425 }
426 
GetTypeName()427 Handle<Object> JSStackFrame::GetTypeName() {
428   // TODO(jgruber): Check for strict/constructor here as in
429   // CallSitePrototypeGetThis.
430 
431   if (receiver_->IsNullOrUndefined(isolate_)) {
432     return isolate_->factory()->null_value();
433   } else if (receiver_->IsJSProxy()) {
434     return isolate_->factory()->Proxy_string();
435   }
436 
437   Handle<JSReceiver> receiver;
438   if (!Object::ToObject(isolate_, receiver_).ToHandle(&receiver)) {
439     DCHECK(isolate_->has_pending_exception());
440     isolate_->clear_pending_exception();
441     isolate_->set_external_caught_exception(false);
442     return isolate_->factory()->null_value();
443   }
444 
445   return JSReceiver::GetConstructorName(receiver);
446 }
447 
GetLineNumber()448 int JSStackFrame::GetLineNumber() {
449   DCHECK_LE(0, GetPosition());
450   if (HasScript()) return Script::GetLineNumber(GetScript(), GetPosition()) + 1;
451   return -1;
452 }
453 
GetColumnNumber()454 int JSStackFrame::GetColumnNumber() {
455   DCHECK_LE(0, GetPosition());
456   if (HasScript()) {
457     return Script::GetColumnNumber(GetScript(), GetPosition()) + 1;
458   }
459   return -1;
460 }
461 
IsNative()462 bool JSStackFrame::IsNative() {
463   return HasScript() && GetScript()->type() == Script::TYPE_NATIVE;
464 }
465 
IsToplevel()466 bool JSStackFrame::IsToplevel() {
467   return receiver_->IsJSGlobalProxy() || receiver_->IsNullOrUndefined(isolate_);
468 }
469 
470 namespace {
471 
IsNonEmptyString(Handle<Object> object)472 bool IsNonEmptyString(Handle<Object> object) {
473   return (object->IsString() && String::cast(*object)->length() > 0);
474 }
475 
AppendFileLocation(Isolate * isolate,StackFrameBase * call_site,IncrementalStringBuilder * builder)476 void AppendFileLocation(Isolate* isolate, StackFrameBase* call_site,
477                         IncrementalStringBuilder* builder) {
478   if (call_site->IsNative()) {
479     builder->AppendCString("native");
480     return;
481   }
482 
483   Handle<Object> file_name = call_site->GetScriptNameOrSourceUrl();
484   if (!file_name->IsString() && call_site->IsEval()) {
485     Handle<Object> eval_origin = call_site->GetEvalOrigin();
486     DCHECK(eval_origin->IsString());
487     builder->AppendString(Handle<String>::cast(eval_origin));
488     builder->AppendCString(", ");  // Expecting source position to follow.
489   }
490 
491   if (IsNonEmptyString(file_name)) {
492     builder->AppendString(Handle<String>::cast(file_name));
493   } else {
494     // Source code does not originate from a file and is not native, but we
495     // can still get the source position inside the source string, e.g. in
496     // an eval string.
497     builder->AppendCString("<anonymous>");
498   }
499 
500   int line_number = call_site->GetLineNumber();
501   if (line_number != -1) {
502     builder->AppendCharacter(':');
503     Handle<String> line_string = isolate->factory()->NumberToString(
504         handle(Smi::FromInt(line_number), isolate), isolate);
505     builder->AppendString(line_string);
506 
507     int column_number = call_site->GetColumnNumber();
508     if (column_number != -1) {
509       builder->AppendCharacter(':');
510       Handle<String> column_string = isolate->factory()->NumberToString(
511           handle(Smi::FromInt(column_number), isolate), isolate);
512       builder->AppendString(column_string);
513     }
514   }
515 }
516 
StringIndexOf(Isolate * isolate,Handle<String> subject,Handle<String> pattern)517 int StringIndexOf(Isolate* isolate, Handle<String> subject,
518                   Handle<String> pattern) {
519   if (pattern->length() > subject->length()) return -1;
520   return String::IndexOf(isolate, subject, pattern, 0);
521 }
522 
523 // Returns true iff
524 // 1. the subject ends with '.' + pattern, or
525 // 2. subject == pattern.
StringEndsWithMethodName(Isolate * isolate,Handle<String> subject,Handle<String> pattern)526 bool StringEndsWithMethodName(Isolate* isolate, Handle<String> subject,
527                               Handle<String> pattern) {
528   if (String::Equals(isolate, subject, pattern)) return true;
529 
530   FlatStringReader subject_reader(isolate, String::Flatten(isolate, subject));
531   FlatStringReader pattern_reader(isolate, String::Flatten(isolate, pattern));
532 
533   int pattern_index = pattern_reader.length() - 1;
534   int subject_index = subject_reader.length() - 1;
535   for (int i = 0; i <= pattern_reader.length(); i++) {  // Iterate over len + 1.
536     if (subject_index < 0) {
537       return false;
538     }
539 
540     const uc32 subject_char = subject_reader.Get(subject_index);
541     if (i == pattern_reader.length()) {
542       if (subject_char != '.') return false;
543     } else if (subject_char != pattern_reader.Get(pattern_index)) {
544       return false;
545     }
546 
547     pattern_index--;
548     subject_index--;
549   }
550 
551   return true;
552 }
553 
AppendMethodCall(Isolate * isolate,JSStackFrame * call_site,IncrementalStringBuilder * builder)554 void AppendMethodCall(Isolate* isolate, JSStackFrame* call_site,
555                       IncrementalStringBuilder* builder) {
556   Handle<Object> type_name = call_site->GetTypeName();
557   Handle<Object> method_name = call_site->GetMethodName();
558   Handle<Object> function_name = call_site->GetFunctionName();
559 
560   if (IsNonEmptyString(function_name)) {
561     Handle<String> function_string = Handle<String>::cast(function_name);
562     if (IsNonEmptyString(type_name)) {
563       Handle<String> type_string = Handle<String>::cast(type_name);
564       bool starts_with_type_name =
565           (StringIndexOf(isolate, function_string, type_string) == 0);
566       if (!starts_with_type_name) {
567         builder->AppendString(type_string);
568         builder->AppendCharacter('.');
569       }
570     }
571     builder->AppendString(function_string);
572 
573     if (IsNonEmptyString(method_name)) {
574       Handle<String> method_string = Handle<String>::cast(method_name);
575       if (!StringEndsWithMethodName(isolate, function_string, method_string)) {
576         builder->AppendCString(" [as ");
577         builder->AppendString(method_string);
578         builder->AppendCharacter(']');
579       }
580     }
581   } else {
582     if (IsNonEmptyString(type_name)) {
583       builder->AppendString(Handle<String>::cast(type_name));
584       builder->AppendCharacter('.');
585     }
586     if (IsNonEmptyString(method_name)) {
587       builder->AppendString(Handle<String>::cast(method_name));
588     } else {
589       builder->AppendCString("<anonymous>");
590     }
591   }
592 }
593 
594 }  // namespace
595 
ToString()596 MaybeHandle<String> JSStackFrame::ToString() {
597   IncrementalStringBuilder builder(isolate_);
598 
599   Handle<Object> function_name = GetFunctionName();
600 
601   const bool is_toplevel = IsToplevel();
602   const bool is_constructor = IsConstructor();
603   const bool is_method_call = !(is_toplevel || is_constructor);
604 
605   if (is_method_call) {
606     AppendMethodCall(isolate_, this, &builder);
607   } else if (is_constructor) {
608     builder.AppendCString("new ");
609     if (IsNonEmptyString(function_name)) {
610       builder.AppendString(Handle<String>::cast(function_name));
611     } else {
612       builder.AppendCString("<anonymous>");
613     }
614   } else if (IsNonEmptyString(function_name)) {
615     builder.AppendString(Handle<String>::cast(function_name));
616   } else {
617     AppendFileLocation(isolate_, this, &builder);
618     return builder.Finish();
619   }
620 
621   builder.AppendCString(" (");
622   AppendFileLocation(isolate_, this, &builder);
623   builder.AppendCString(")");
624 
625   return builder.Finish();
626 }
627 
GetPosition() const628 int JSStackFrame::GetPosition() const { return code_->SourcePosition(offset_); }
629 
HasScript() const630 bool JSStackFrame::HasScript() const {
631   return function_->shared()->script()->IsScript();
632 }
633 
GetScript() const634 Handle<Script> JSStackFrame::GetScript() const {
635   return handle(Script::cast(function_->shared()->script()), isolate_);
636 }
637 
WasmStackFrame()638 WasmStackFrame::WasmStackFrame() {}
639 
FromFrameArray(Isolate * isolate,Handle<FrameArray> array,int frame_ix)640 void WasmStackFrame::FromFrameArray(Isolate* isolate, Handle<FrameArray> array,
641                                     int frame_ix) {
642   // This function is called for compiled and interpreted wasm frames, and for
643   // asm.js->wasm frames.
644   DCHECK(array->IsWasmFrame(frame_ix) ||
645          array->IsWasmInterpretedFrame(frame_ix) ||
646          array->IsAsmJsWasmFrame(frame_ix));
647   isolate_ = isolate;
648   wasm_instance_ = handle(array->WasmInstance(frame_ix), isolate);
649   wasm_func_index_ = array->WasmFunctionIndex(frame_ix)->value();
650   if (array->IsWasmInterpretedFrame(frame_ix)) {
651     code_ = nullptr;
652   } else {
653     code_ = reinterpret_cast<wasm::WasmCode*>(
654         array->WasmCodeObject(frame_ix)->foreign_address());
655   }
656   offset_ = array->Offset(frame_ix)->value();
657 }
658 
GetReceiver() const659 Handle<Object> WasmStackFrame::GetReceiver() const { return wasm_instance_; }
660 
GetFunction() const661 Handle<Object> WasmStackFrame::GetFunction() const {
662   return handle(Smi::FromInt(wasm_func_index_), isolate_);
663 }
664 
GetFunctionName()665 Handle<Object> WasmStackFrame::GetFunctionName() {
666   Handle<Object> name;
667   Handle<WasmModuleObject> module_object(wasm_instance_->module_object(),
668                                          isolate_);
669   if (!WasmModuleObject::GetFunctionNameOrNull(isolate_, module_object,
670                                                wasm_func_index_)
671            .ToHandle(&name)) {
672     name = isolate_->factory()->null_value();
673   }
674   return name;
675 }
676 
ToString()677 MaybeHandle<String> WasmStackFrame::ToString() {
678   IncrementalStringBuilder builder(isolate_);
679 
680   Handle<WasmModuleObject> module_object(wasm_instance_->module_object(),
681                                          isolate_);
682   MaybeHandle<String> module_name =
683       WasmModuleObject::GetModuleNameOrNull(isolate_, module_object);
684   MaybeHandle<String> function_name = WasmModuleObject::GetFunctionNameOrNull(
685       isolate_, module_object, wasm_func_index_);
686   bool has_name = !module_name.is_null() || !function_name.is_null();
687   if (has_name) {
688     if (module_name.is_null()) {
689       builder.AppendString(function_name.ToHandleChecked());
690     } else {
691       builder.AppendString(module_name.ToHandleChecked());
692       if (!function_name.is_null()) {
693         builder.AppendCString(".");
694         builder.AppendString(function_name.ToHandleChecked());
695       }
696     }
697     builder.AppendCString(" (");
698   }
699 
700   builder.AppendCString("wasm-function[");
701 
702   char buffer[16];
703   SNPrintF(ArrayVector(buffer), "%u]", wasm_func_index_);
704   builder.AppendCString(buffer);
705 
706   SNPrintF(ArrayVector(buffer), ":%d", GetPosition());
707   builder.AppendCString(buffer);
708 
709   if (has_name) builder.AppendCString(")");
710 
711   return builder.Finish();
712 }
713 
GetPosition() const714 int WasmStackFrame::GetPosition() const {
715   return IsInterpreted()
716              ? offset_
717              : FrameSummary::WasmCompiledFrameSummary::GetWasmSourcePosition(
718                    code_, offset_);
719 }
720 
Null() const721 Handle<Object> WasmStackFrame::Null() const {
722   return isolate_->factory()->null_value();
723 }
724 
HasScript() const725 bool WasmStackFrame::HasScript() const { return true; }
726 
GetScript() const727 Handle<Script> WasmStackFrame::GetScript() const {
728   return handle(wasm_instance_->module_object()->script(), isolate_);
729 }
730 
AsmJsWasmStackFrame()731 AsmJsWasmStackFrame::AsmJsWasmStackFrame() {}
732 
FromFrameArray(Isolate * isolate,Handle<FrameArray> array,int frame_ix)733 void AsmJsWasmStackFrame::FromFrameArray(Isolate* isolate,
734                                          Handle<FrameArray> array,
735                                          int frame_ix) {
736   DCHECK(array->IsAsmJsWasmFrame(frame_ix));
737   WasmStackFrame::FromFrameArray(isolate, array, frame_ix);
738   is_at_number_conversion_ =
739       array->Flags(frame_ix)->value() & FrameArray::kAsmJsAtNumberConversion;
740 }
741 
GetReceiver() const742 Handle<Object> AsmJsWasmStackFrame::GetReceiver() const {
743   return isolate_->global_proxy();
744 }
745 
GetFunction() const746 Handle<Object> AsmJsWasmStackFrame::GetFunction() const {
747   // TODO(clemensh): Return lazily created JSFunction.
748   return Null();
749 }
750 
GetFileName()751 Handle<Object> AsmJsWasmStackFrame::GetFileName() {
752   Handle<Script> script(wasm_instance_->module_object()->script(), isolate_);
753   DCHECK(script->IsUserJavaScript());
754   return handle(script->name(), isolate_);
755 }
756 
GetScriptNameOrSourceUrl()757 Handle<Object> AsmJsWasmStackFrame::GetScriptNameOrSourceUrl() {
758   Handle<Script> script(wasm_instance_->module_object()->script(), isolate_);
759   DCHECK_EQ(Script::TYPE_NORMAL, script->type());
760   return ScriptNameOrSourceUrl(script, isolate_);
761 }
762 
GetPosition() const763 int AsmJsWasmStackFrame::GetPosition() const {
764   DCHECK_LE(0, offset_);
765   int byte_offset =
766       FrameSummary::WasmCompiledFrameSummary::GetWasmSourcePosition(code_,
767                                                                     offset_);
768   Handle<WasmModuleObject> module_object(wasm_instance_->module_object(),
769                                          isolate_);
770   DCHECK_LE(0, byte_offset);
771   return WasmModuleObject::GetSourcePosition(module_object, wasm_func_index_,
772                                              static_cast<uint32_t>(byte_offset),
773                                              is_at_number_conversion_);
774 }
775 
GetLineNumber()776 int AsmJsWasmStackFrame::GetLineNumber() {
777   DCHECK_LE(0, GetPosition());
778   Handle<Script> script(wasm_instance_->module_object()->script(), isolate_);
779   DCHECK(script->IsUserJavaScript());
780   return Script::GetLineNumber(script, GetPosition()) + 1;
781 }
782 
GetColumnNumber()783 int AsmJsWasmStackFrame::GetColumnNumber() {
784   DCHECK_LE(0, GetPosition());
785   Handle<Script> script(wasm_instance_->module_object()->script(), isolate_);
786   DCHECK(script->IsUserJavaScript());
787   return Script::GetColumnNumber(script, GetPosition()) + 1;
788 }
789 
ToString()790 MaybeHandle<String> AsmJsWasmStackFrame::ToString() {
791   // The string should look exactly as the respective javascript frame string.
792   // Keep this method in line to JSStackFrame::ToString().
793 
794   IncrementalStringBuilder builder(isolate_);
795 
796   Handle<Object> function_name = GetFunctionName();
797 
798   if (IsNonEmptyString(function_name)) {
799     builder.AppendString(Handle<String>::cast(function_name));
800     builder.AppendCString(" (");
801   }
802 
803   AppendFileLocation(isolate_, this, &builder);
804 
805   if (IsNonEmptyString(function_name)) builder.AppendCString(")");
806 
807   return builder.Finish();
808 }
809 
FrameArrayIterator(Isolate * isolate,Handle<FrameArray> array,int frame_ix)810 FrameArrayIterator::FrameArrayIterator(Isolate* isolate,
811                                        Handle<FrameArray> array, int frame_ix)
812     : isolate_(isolate), array_(array), next_frame_ix_(frame_ix) {}
813 
HasNext() const814 bool FrameArrayIterator::HasNext() const {
815   return (next_frame_ix_ < array_->FrameCount());
816 }
817 
Next()818 void FrameArrayIterator::Next() { next_frame_ix_++; }
819 
Frame()820 StackFrameBase* FrameArrayIterator::Frame() {
821   DCHECK(HasNext());
822   const int flags = array_->Flags(next_frame_ix_)->value();
823   int flag_mask = FrameArray::kIsWasmFrame |
824                   FrameArray::kIsWasmInterpretedFrame |
825                   FrameArray::kIsAsmJsWasmFrame;
826   switch (flags & flag_mask) {
827     case 0:
828       // JavaScript Frame.
829       js_frame_.FromFrameArray(isolate_, array_, next_frame_ix_);
830       return &js_frame_;
831     case FrameArray::kIsWasmFrame:
832     case FrameArray::kIsWasmInterpretedFrame:
833       // Wasm Frame:
834       wasm_frame_.FromFrameArray(isolate_, array_, next_frame_ix_);
835       return &wasm_frame_;
836     case FrameArray::kIsAsmJsWasmFrame:
837       // Asm.js Wasm Frame:
838       asm_wasm_frame_.FromFrameArray(isolate_, array_, next_frame_ix_);
839       return &asm_wasm_frame_;
840     default:
841       UNREACHABLE();
842   }
843 }
844 
845 namespace {
846 
ConstructCallSite(Isolate * isolate,Handle<FrameArray> frame_array,int frame_index)847 MaybeHandle<Object> ConstructCallSite(Isolate* isolate,
848                                       Handle<FrameArray> frame_array,
849                                       int frame_index) {
850   Handle<JSFunction> target =
851       handle(isolate->native_context()->callsite_function(), isolate);
852 
853   Handle<JSObject> obj;
854   ASSIGN_RETURN_ON_EXCEPTION(isolate, obj, JSObject::New(target, target),
855                              Object);
856 
857   Handle<Symbol> key = isolate->factory()->call_site_frame_array_symbol();
858   RETURN_ON_EXCEPTION(isolate, JSObject::SetOwnPropertyIgnoreAttributes(
859                                    obj, key, frame_array, DONT_ENUM),
860                       Object);
861 
862   key = isolate->factory()->call_site_frame_index_symbol();
863   Handle<Object> value(Smi::FromInt(frame_index), isolate);
864   RETURN_ON_EXCEPTION(isolate, JSObject::SetOwnPropertyIgnoreAttributes(
865                                    obj, key, value, DONT_ENUM),
866                       Object);
867 
868   return obj;
869 }
870 
871 // Convert the raw frames as written by Isolate::CaptureSimpleStackTrace into
872 // a JSArray of JSCallSite objects.
GetStackFrames(Isolate * isolate,Handle<FrameArray> elems)873 MaybeHandle<JSArray> GetStackFrames(Isolate* isolate,
874                                     Handle<FrameArray> elems) {
875   const int frame_count = elems->FrameCount();
876 
877   Handle<FixedArray> frames = isolate->factory()->NewFixedArray(frame_count);
878   for (int i = 0; i < frame_count; i++) {
879     Handle<Object> site;
880     ASSIGN_RETURN_ON_EXCEPTION(isolate, site,
881                                ConstructCallSite(isolate, elems, i), JSArray);
882     frames->set(i, *site);
883   }
884 
885   return isolate->factory()->NewJSArrayWithElements(frames);
886 }
887 
AppendErrorString(Isolate * isolate,Handle<Object> error,IncrementalStringBuilder * builder)888 MaybeHandle<Object> AppendErrorString(Isolate* isolate, Handle<Object> error,
889                                       IncrementalStringBuilder* builder) {
890   MaybeHandle<String> err_str =
891       ErrorUtils::ToString(isolate, Handle<Object>::cast(error));
892   if (err_str.is_null()) {
893     // Error.toString threw. Try to return a string representation of the thrown
894     // exception instead.
895 
896     DCHECK(isolate->has_pending_exception());
897     Handle<Object> pending_exception =
898         handle(isolate->pending_exception(), isolate);
899     isolate->clear_pending_exception();
900     isolate->set_external_caught_exception(false);
901 
902     err_str = ErrorUtils::ToString(isolate, pending_exception);
903     if (err_str.is_null()) {
904       // Formatting the thrown exception threw again, give up.
905       DCHECK(isolate->has_pending_exception());
906       isolate->clear_pending_exception();
907       isolate->set_external_caught_exception(false);
908       builder->AppendCString("<error>");
909     } else {
910       // Formatted thrown exception successfully, append it.
911       builder->AppendCString("<error: ");
912       builder->AppendString(err_str.ToHandleChecked());
913       builder->AppendCharacter('>');
914     }
915   } else {
916     builder->AppendString(err_str.ToHandleChecked());
917   }
918 
919   return error;
920 }
921 
922 class PrepareStackTraceScope {
923  public:
PrepareStackTraceScope(Isolate * isolate)924   explicit PrepareStackTraceScope(Isolate* isolate) : isolate_(isolate) {
925     DCHECK(!isolate_->formatting_stack_trace());
926     isolate_->set_formatting_stack_trace(true);
927   }
928 
~PrepareStackTraceScope()929   ~PrepareStackTraceScope() { isolate_->set_formatting_stack_trace(false); }
930 
931  private:
932   Isolate* isolate_;
933 
934   DISALLOW_COPY_AND_ASSIGN(PrepareStackTraceScope);
935 };
936 
937 }  // namespace
938 
939 // static
FormatStackTrace(Isolate * isolate,Handle<JSObject> error,Handle<Object> raw_stack)940 MaybeHandle<Object> ErrorUtils::FormatStackTrace(Isolate* isolate,
941                                                  Handle<JSObject> error,
942                                                  Handle<Object> raw_stack) {
943   DCHECK(raw_stack->IsJSArray());
944   Handle<JSArray> raw_stack_array = Handle<JSArray>::cast(raw_stack);
945 
946   DCHECK(raw_stack_array->elements()->IsFixedArray());
947   Handle<FrameArray> elems(FrameArray::cast(raw_stack_array->elements()),
948                            isolate);
949 
950   // If there's a user-specified "prepareStackFrames" function, call it on the
951   // frames and use its result.
952 
953   Handle<JSFunction> global_error = isolate->error_function();
954   Handle<Object> prepare_stack_trace;
955   ASSIGN_RETURN_ON_EXCEPTION(
956       isolate, prepare_stack_trace,
957       JSFunction::GetProperty(isolate, global_error, "prepareStackTrace"),
958       Object);
959 
960   const bool in_recursion = isolate->formatting_stack_trace();
961   if (prepare_stack_trace->IsJSFunction() && !in_recursion) {
962     PrepareStackTraceScope scope(isolate);
963 
964     isolate->CountUsage(v8::Isolate::kErrorPrepareStackTrace);
965 
966     Handle<JSArray> sites;
967     ASSIGN_RETURN_ON_EXCEPTION(isolate, sites, GetStackFrames(isolate, elems),
968                                Object);
969 
970     const int argc = 2;
971     ScopedVector<Handle<Object>> argv(argc);
972 
973     argv[0] = error;
974     argv[1] = sites;
975 
976     Handle<Object> result;
977     ASSIGN_RETURN_ON_EXCEPTION(
978         isolate, result, Execution::Call(isolate, prepare_stack_trace,
979                                          global_error, argc, argv.start()),
980         Object);
981 
982     return result;
983   }
984 
985   // Otherwise, run our internal formatting logic.
986 
987   IncrementalStringBuilder builder(isolate);
988 
989   RETURN_ON_EXCEPTION(isolate, AppendErrorString(isolate, error, &builder),
990                       Object);
991 
992   for (FrameArrayIterator it(isolate, elems); it.HasNext(); it.Next()) {
993     builder.AppendCString("\n    at ");
994 
995     StackFrameBase* frame = it.Frame();
996     MaybeHandle<String> maybe_frame_string = frame->ToString();
997     if (maybe_frame_string.is_null()) {
998       // CallSite.toString threw. Try to return a string representation of the
999       // thrown exception instead.
1000 
1001       DCHECK(isolate->has_pending_exception());
1002       Handle<Object> pending_exception =
1003           handle(isolate->pending_exception(), isolate);
1004       isolate->clear_pending_exception();
1005       isolate->set_external_caught_exception(false);
1006 
1007       maybe_frame_string = ErrorUtils::ToString(isolate, pending_exception);
1008       if (maybe_frame_string.is_null()) {
1009         // Formatting the thrown exception threw again, give up.
1010 
1011         builder.AppendCString("<error>");
1012       } else {
1013         // Formatted thrown exception successfully, append it.
1014         builder.AppendCString("<error: ");
1015         builder.AppendString(maybe_frame_string.ToHandleChecked());
1016         builder.AppendCString("<error>");
1017       }
1018     } else {
1019       // CallSite.toString completed without throwing.
1020       builder.AppendString(maybe_frame_string.ToHandleChecked());
1021     }
1022   }
1023 
1024   return builder.Finish();
1025 }
1026 
FormatMessage(Isolate * isolate,int template_index,Handle<Object> arg)1027 Handle<String> MessageTemplate::FormatMessage(Isolate* isolate,
1028                                               int template_index,
1029                                               Handle<Object> arg) {
1030   Factory* factory = isolate->factory();
1031   Handle<String> result_string = Object::NoSideEffectsToString(isolate, arg);
1032   MaybeHandle<String> maybe_result_string = MessageTemplate::FormatMessage(
1033       isolate, template_index, result_string, factory->empty_string(),
1034       factory->empty_string());
1035   if (!maybe_result_string.ToHandle(&result_string)) {
1036     DCHECK(isolate->has_pending_exception());
1037     isolate->clear_pending_exception();
1038     return factory->InternalizeOneByteString(STATIC_CHAR_VECTOR("<error>"));
1039   }
1040   // A string that has been obtained from JS code in this way is
1041   // likely to be a complicated ConsString of some sort.  We flatten it
1042   // here to improve the efficiency of converting it to a C string and
1043   // other operations that are likely to take place (see GetLocalizedMessage
1044   // for example).
1045   return String::Flatten(isolate, result_string);
1046 }
1047 
1048 
TemplateString(int template_index)1049 const char* MessageTemplate::TemplateString(int template_index) {
1050   switch (template_index) {
1051 #define CASE(NAME, STRING) \
1052   case k##NAME:            \
1053     return STRING;
1054     MESSAGE_TEMPLATES(CASE)
1055 #undef CASE
1056     case kLastMessage:
1057     default:
1058       return nullptr;
1059   }
1060 }
1061 
FormatMessage(Isolate * isolate,int template_index,Handle<String> arg0,Handle<String> arg1,Handle<String> arg2)1062 MaybeHandle<String> MessageTemplate::FormatMessage(Isolate* isolate,
1063                                                    int template_index,
1064                                                    Handle<String> arg0,
1065                                                    Handle<String> arg1,
1066                                                    Handle<String> arg2) {
1067   const char* template_string = TemplateString(template_index);
1068   if (template_string == nullptr) {
1069     isolate->ThrowIllegalOperation();
1070     return MaybeHandle<String>();
1071   }
1072 
1073   IncrementalStringBuilder builder(isolate);
1074 
1075   unsigned int i = 0;
1076   Handle<String> args[] = {arg0, arg1, arg2};
1077   for (const char* c = template_string; *c != '\0'; c++) {
1078     if (*c == '%') {
1079       // %% results in verbatim %.
1080       if (*(c + 1) == '%') {
1081         c++;
1082         builder.AppendCharacter('%');
1083       } else {
1084         DCHECK(i < arraysize(args));
1085         Handle<String> arg = args[i++];
1086         builder.AppendString(arg);
1087       }
1088     } else {
1089       builder.AppendCharacter(*c);
1090     }
1091   }
1092 
1093   return builder.Finish();
1094 }
1095 
Construct(Isolate * isolate,Handle<JSFunction> target,Handle<Object> new_target,Handle<Object> message,FrameSkipMode mode,Handle<Object> caller,bool suppress_detailed_trace)1096 MaybeHandle<Object> ErrorUtils::Construct(
1097     Isolate* isolate, Handle<JSFunction> target, Handle<Object> new_target,
1098     Handle<Object> message, FrameSkipMode mode, Handle<Object> caller,
1099     bool suppress_detailed_trace) {
1100   // 1. If NewTarget is undefined, let newTarget be the active function object,
1101   // else let newTarget be NewTarget.
1102 
1103   Handle<JSReceiver> new_target_recv =
1104       new_target->IsJSReceiver() ? Handle<JSReceiver>::cast(new_target)
1105                                  : Handle<JSReceiver>::cast(target);
1106 
1107   // 2. Let O be ? OrdinaryCreateFromConstructor(newTarget, "%ErrorPrototype%",
1108   //    « [[ErrorData]] »).
1109   Handle<JSObject> err;
1110   ASSIGN_RETURN_ON_EXCEPTION(isolate, err,
1111                              JSObject::New(target, new_target_recv), Object);
1112 
1113   // 3. If message is not undefined, then
1114   //  a. Let msg be ? ToString(message).
1115   //  b. Let msgDesc be the PropertyDescriptor{[[Value]]: msg, [[Writable]]:
1116   //     true, [[Enumerable]]: false, [[Configurable]]: true}.
1117   //  c. Perform ! DefinePropertyOrThrow(O, "message", msgDesc).
1118   // 4. Return O.
1119 
1120   if (!message->IsUndefined(isolate)) {
1121     Handle<String> msg_string;
1122     ASSIGN_RETURN_ON_EXCEPTION(isolate, msg_string,
1123                                Object::ToString(isolate, message), Object);
1124     RETURN_ON_EXCEPTION(isolate, JSObject::SetOwnPropertyIgnoreAttributes(
1125                                      err, isolate->factory()->message_string(),
1126                                      msg_string, DONT_ENUM),
1127                         Object);
1128   }
1129 
1130   // Optionally capture a more detailed stack trace for the message.
1131   if (!suppress_detailed_trace) {
1132     RETURN_ON_EXCEPTION(isolate, isolate->CaptureAndSetDetailedStackTrace(err),
1133                         Object);
1134   }
1135 
1136   // Capture a simple stack trace for the stack property.
1137   RETURN_ON_EXCEPTION(isolate,
1138                       isolate->CaptureAndSetSimpleStackTrace(err, mode, caller),
1139                       Object);
1140 
1141   return err;
1142 }
1143 
1144 namespace {
1145 
GetStringPropertyOrDefault(Isolate * isolate,Handle<JSReceiver> recv,Handle<String> key,Handle<String> default_str)1146 MaybeHandle<String> GetStringPropertyOrDefault(Isolate* isolate,
1147                                                Handle<JSReceiver> recv,
1148                                                Handle<String> key,
1149                                                Handle<String> default_str) {
1150   Handle<Object> obj;
1151   ASSIGN_RETURN_ON_EXCEPTION(isolate, obj,
1152                              JSObject::GetProperty(isolate, recv, key), String);
1153 
1154   Handle<String> str;
1155   if (obj->IsUndefined(isolate)) {
1156     str = default_str;
1157   } else {
1158     ASSIGN_RETURN_ON_EXCEPTION(isolate, str, Object::ToString(isolate, obj),
1159                                String);
1160   }
1161 
1162   return str;
1163 }
1164 
1165 }  // namespace
1166 
1167 // ES6 section 19.5.3.4 Error.prototype.toString ( )
ToString(Isolate * isolate,Handle<Object> receiver)1168 MaybeHandle<String> ErrorUtils::ToString(Isolate* isolate,
1169                                          Handle<Object> receiver) {
1170   // 1. Let O be the this value.
1171   // 2. If Type(O) is not Object, throw a TypeError exception.
1172   if (!receiver->IsJSReceiver()) {
1173     return isolate->Throw<String>(isolate->factory()->NewTypeError(
1174         MessageTemplate::kIncompatibleMethodReceiver,
1175         isolate->factory()->NewStringFromAsciiChecked(
1176             "Error.prototype.toString"),
1177         receiver));
1178   }
1179   Handle<JSReceiver> recv = Handle<JSReceiver>::cast(receiver);
1180 
1181   // 3. Let name be ? Get(O, "name").
1182   // 4. If name is undefined, let name be "Error"; otherwise let name be
1183   // ? ToString(name).
1184   Handle<String> name_key = isolate->factory()->name_string();
1185   Handle<String> name_default = isolate->factory()->Error_string();
1186   Handle<String> name;
1187   ASSIGN_RETURN_ON_EXCEPTION(
1188       isolate, name,
1189       GetStringPropertyOrDefault(isolate, recv, name_key, name_default),
1190       String);
1191 
1192   // 5. Let msg be ? Get(O, "message").
1193   // 6. If msg is undefined, let msg be the empty String; otherwise let msg be
1194   // ? ToString(msg).
1195   Handle<String> msg_key = isolate->factory()->message_string();
1196   Handle<String> msg_default = isolate->factory()->empty_string();
1197   Handle<String> msg;
1198   ASSIGN_RETURN_ON_EXCEPTION(
1199       isolate, msg,
1200       GetStringPropertyOrDefault(isolate, recv, msg_key, msg_default), String);
1201 
1202   // 7. If name is the empty String, return msg.
1203   // 8. If msg is the empty String, return name.
1204   if (name->length() == 0) return msg;
1205   if (msg->length() == 0) return name;
1206 
1207   // 9. Return the result of concatenating name, the code unit 0x003A (COLON),
1208   // the code unit 0x0020 (SPACE), and msg.
1209   IncrementalStringBuilder builder(isolate);
1210   builder.AppendString(name);
1211   builder.AppendCString(": ");
1212   builder.AppendString(msg);
1213 
1214   Handle<String> result;
1215   ASSIGN_RETURN_ON_EXCEPTION(isolate, result, builder.Finish(), String);
1216   return result;
1217 }
1218 
1219 namespace {
1220 
FormatMessage(Isolate * isolate,int template_index,Handle<Object> arg0,Handle<Object> arg1,Handle<Object> arg2)1221 Handle<String> FormatMessage(Isolate* isolate, int template_index,
1222                              Handle<Object> arg0, Handle<Object> arg1,
1223                              Handle<Object> arg2) {
1224   Handle<String> arg0_str = Object::NoSideEffectsToString(isolate, arg0);
1225   Handle<String> arg1_str = Object::NoSideEffectsToString(isolate, arg1);
1226   Handle<String> arg2_str = Object::NoSideEffectsToString(isolate, arg2);
1227 
1228   isolate->native_context()->IncrementErrorsThrown();
1229 
1230   Handle<String> msg;
1231   if (!MessageTemplate::FormatMessage(isolate, template_index, arg0_str,
1232                                       arg1_str, arg2_str)
1233            .ToHandle(&msg)) {
1234     DCHECK(isolate->has_pending_exception());
1235     isolate->clear_pending_exception();
1236     isolate->set_external_caught_exception(false);
1237     return isolate->factory()->NewStringFromAsciiChecked("<error>");
1238   }
1239 
1240   return msg;
1241 }
1242 
1243 }  // namespace
1244 
1245 // static
MakeGenericError(Isolate * isolate,Handle<JSFunction> constructor,int template_index,Handle<Object> arg0,Handle<Object> arg1,Handle<Object> arg2,FrameSkipMode mode)1246 MaybeHandle<Object> ErrorUtils::MakeGenericError(
1247     Isolate* isolate, Handle<JSFunction> constructor, int template_index,
1248     Handle<Object> arg0, Handle<Object> arg1, Handle<Object> arg2,
1249     FrameSkipMode mode) {
1250   if (FLAG_clear_exceptions_on_js_entry) {
1251     // This function used to be implemented in JavaScript, and JSEntryStub
1252     // clears
1253     // any pending exceptions - so whenever we'd call this from C++, pending
1254     // exceptions would be cleared. Preserve this behavior.
1255     isolate->clear_pending_exception();
1256   }
1257 
1258   DCHECK(mode != SKIP_UNTIL_SEEN);
1259 
1260   Handle<Object> no_caller;
1261   Handle<String> msg = FormatMessage(isolate, template_index, arg0, arg1, arg2);
1262   return ErrorUtils::Construct(isolate, constructor, constructor, msg, mode,
1263                                no_caller, false);
1264 }
1265 
1266 }  // namespace internal
1267 }  // namespace v8
1268