OLD | NEW |
1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/runtime/runtime-utils.h" | 5 #include "src/runtime/runtime-utils.h" |
6 | 6 |
7 #include <memory> | 7 #include <memory> |
8 | 8 |
9 #include "src/arguments.h" | 9 #include "src/arguments.h" |
10 #include "src/ast/prettyprinter.h" | 10 #include "src/ast/prettyprinter.h" |
11 #include "src/bootstrapper.h" | 11 #include "src/bootstrapper.h" |
12 #include "src/conversions.h" | 12 #include "src/conversions.h" |
13 #include "src/debug/debug.h" | 13 #include "src/debug/debug.h" |
14 #include "src/elements.h" | |
15 #include "src/frames-inl.h" | 14 #include "src/frames-inl.h" |
16 #include "src/isolate-inl.h" | 15 #include "src/isolate-inl.h" |
17 #include "src/messages.h" | 16 #include "src/messages.h" |
18 #include "src/parsing/parse-info.h" | 17 #include "src/parsing/parse-info.h" |
19 #include "src/parsing/parser.h" | 18 #include "src/parsing/parser.h" |
20 #include "src/wasm/wasm-module.h" | 19 #include "src/wasm/wasm-module.h" |
21 | 20 |
22 namespace v8 { | 21 namespace v8 { |
23 namespace internal { | 22 namespace internal { |
24 | 23 |
(...skipping 256 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
281 | 280 |
282 RUNTIME_FUNCTION(Runtime_ThrowApplyNonFunction) { | 281 RUNTIME_FUNCTION(Runtime_ThrowApplyNonFunction) { |
283 HandleScope scope(isolate); | 282 HandleScope scope(isolate); |
284 DCHECK_EQ(1, args.length()); | 283 DCHECK_EQ(1, args.length()); |
285 CONVERT_ARG_HANDLE_CHECKED(Object, object, 0); | 284 CONVERT_ARG_HANDLE_CHECKED(Object, object, 0); |
286 Handle<String> type = Object::TypeOf(isolate, object); | 285 Handle<String> type = Object::TypeOf(isolate, object); |
287 THROW_NEW_ERROR_RETURN_FAILURE( | 286 THROW_NEW_ERROR_RETURN_FAILURE( |
288 isolate, NewTypeError(MessageTemplate::kApplyNonFunction, object, type)); | 287 isolate, NewTypeError(MessageTemplate::kApplyNonFunction, object, type)); |
289 } | 288 } |
290 | 289 |
291 namespace { | |
292 | |
293 void PromiseRejectEvent(Isolate* isolate, Handle<JSObject> promise, | |
294 Handle<Object> rejected_promise, Handle<Object> value, | |
295 bool debug_event) { | |
296 if (isolate->debug()->is_active() && debug_event) { | |
297 isolate->debug()->OnPromiseReject(rejected_promise, value); | |
298 } | |
299 Handle<Symbol> key = isolate->factory()->promise_has_handler_symbol(); | |
300 // Do not report if we actually have a handler. | |
301 if (JSReceiver::GetDataProperty(promise, key)->IsUndefined(isolate)) { | |
302 isolate->ReportPromiseReject(promise, value, | |
303 v8::kPromiseRejectWithNoHandler); | |
304 } | |
305 } | |
306 | |
307 } // namespace | |
308 | |
309 RUNTIME_FUNCTION(Runtime_PromiseRejectEvent) { | |
310 DCHECK(args.length() == 3); | |
311 HandleScope scope(isolate); | |
312 CONVERT_ARG_HANDLE_CHECKED(JSObject, promise, 0); | |
313 CONVERT_ARG_HANDLE_CHECKED(Object, value, 1); | |
314 CONVERT_BOOLEAN_ARG_CHECKED(debug_event, 2); | |
315 | |
316 PromiseRejectEvent(isolate, promise, promise, value, debug_event); | |
317 return isolate->heap()->undefined_value(); | |
318 } | |
319 | |
320 RUNTIME_FUNCTION(Runtime_PromiseRejectEventFromStack) { | |
321 DCHECK(args.length() == 2); | |
322 HandleScope scope(isolate); | |
323 CONVERT_ARG_HANDLE_CHECKED(JSObject, promise, 0); | |
324 CONVERT_ARG_HANDLE_CHECKED(Object, value, 1); | |
325 | |
326 Handle<Object> rejected_promise = promise; | |
327 if (isolate->debug()->is_active()) { | |
328 // If the Promise.reject call is caught, then this will return | |
329 // undefined, which will be interpreted by PromiseRejectEvent | |
330 // as being a caught exception event. | |
331 rejected_promise = isolate->GetPromiseOnStackOnThrow(); | |
332 } | |
333 PromiseRejectEvent(isolate, promise, rejected_promise, value, true); | |
334 return isolate->heap()->undefined_value(); | |
335 } | |
336 | |
337 RUNTIME_FUNCTION(Runtime_PromiseRevokeReject) { | |
338 DCHECK(args.length() == 1); | |
339 HandleScope scope(isolate); | |
340 CONVERT_ARG_HANDLE_CHECKED(JSObject, promise, 0); | |
341 Handle<Symbol> key = isolate->factory()->promise_has_handler_symbol(); | |
342 // At this point, no revocation has been issued before | |
343 CHECK(JSReceiver::GetDataProperty(promise, key)->IsUndefined(isolate)); | |
344 isolate->ReportPromiseReject(promise, Handle<Object>(), | |
345 v8::kPromiseHandlerAddedAfterReject); | |
346 return isolate->heap()->undefined_value(); | |
347 } | |
348 | |
349 | 290 |
350 RUNTIME_FUNCTION(Runtime_StackGuard) { | 291 RUNTIME_FUNCTION(Runtime_StackGuard) { |
351 SealHandleScope shs(isolate); | 292 SealHandleScope shs(isolate); |
352 DCHECK(args.length() == 0); | 293 DCHECK(args.length() == 0); |
353 | 294 |
354 // First check if this is a real stack overflow. | 295 // First check if this is a real stack overflow. |
355 StackLimitCheck check(isolate); | 296 StackLimitCheck check(isolate); |
356 if (check.JsHasOverflowed()) { | 297 if (check.JsHasOverflowed()) { |
357 return isolate->StackOverflow(); | 298 return isolate->StackOverflow(); |
358 } | 299 } |
(...skipping 206 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
565 isolate->counters()->runtime_call_stats()->Print(stats_stream); | 506 isolate->counters()->runtime_call_stats()->Print(stats_stream); |
566 isolate->counters()->runtime_call_stats()->Reset(); | 507 isolate->counters()->runtime_call_stats()->Reset(); |
567 if (args[0]->IsString()) | 508 if (args[0]->IsString()) |
568 std::fclose(f); | 509 std::fclose(f); |
569 else | 510 else |
570 std::fflush(f); | 511 std::fflush(f); |
571 return isolate->heap()->undefined_value(); | 512 return isolate->heap()->undefined_value(); |
572 } | 513 } |
573 } | 514 } |
574 | 515 |
575 namespace { | |
576 void EnqueuePromiseReactionJob(Isolate* isolate, Handle<Object> value, | |
577 Handle<Object> tasks, Handle<Object> deferred, | |
578 Handle<Object> status) { | |
579 Handle<Object> debug_id = isolate->factory()->undefined_value(); | |
580 Handle<Object> debug_name = isolate->factory()->undefined_value(); | |
581 if (isolate->debug()->is_active()) { | |
582 MaybeHandle<Object> maybe_result; | |
583 Handle<Object> argv[] = {deferred, status}; | |
584 maybe_result = Execution::TryCall( | |
585 isolate, isolate->promise_debug_get_info(), | |
586 isolate->factory()->undefined_value(), arraysize(argv), argv); | |
587 Handle<Object> result; | |
588 if ((maybe_result).ToHandle(&result)) { | |
589 CHECK(result->IsJSArray()); | |
590 Handle<JSArray> array = Handle<JSArray>::cast(result); | |
591 ElementsAccessor* accessor = array->GetElementsAccessor(); | |
592 DCHECK(accessor->HasElement(array, 0)); | |
593 DCHECK(accessor->HasElement(array, 1)); | |
594 debug_id = accessor->Get(array, 0); | |
595 debug_name = accessor->Get(array, 1); | |
596 } | |
597 } | |
598 Handle<PromiseReactionJobInfo> info = | |
599 isolate->factory()->NewPromiseReactionJobInfo(value, tasks, deferred, | |
600 debug_id, debug_name, | |
601 isolate->native_context()); | |
602 isolate->EnqueueMicrotask(info); | |
603 } | |
604 | |
605 void PromiseFulfill(Isolate* isolate, Handle<JSReceiver> promise, | |
606 Handle<Smi> status, Handle<Object> value, | |
607 Handle<Symbol> reaction) { | |
608 Handle<Object> tasks = JSReceiver::GetDataProperty(promise, reaction); | |
609 if (!tasks->IsUndefined(isolate)) { | |
610 Handle<Object> deferred = JSReceiver::GetDataProperty( | |
611 promise, isolate->factory()->promise_deferred_reaction_symbol()); | |
612 EnqueuePromiseReactionJob(isolate, value, tasks, deferred, status); | |
613 } | |
614 } | |
615 } // namespace | |
616 | |
617 RUNTIME_FUNCTION(Runtime_PromiseReject) { | |
618 DCHECK(args.length() == 3); | |
619 HandleScope scope(isolate); | |
620 CONVERT_ARG_HANDLE_CHECKED(JSObject, promise, 0); | |
621 CONVERT_ARG_HANDLE_CHECKED(Object, reason, 1); | |
622 CONVERT_BOOLEAN_ARG_CHECKED(debug_event, 2); | |
623 | |
624 PromiseRejectEvent(isolate, promise, promise, reason, debug_event); | |
625 | |
626 Handle<Smi> status = handle(Smi::FromInt(kPromiseRejected), isolate); | |
627 Handle<Symbol> reaction = | |
628 isolate->factory()->promise_reject_reactions_symbol(); | |
629 PromiseFulfill(isolate, promise, status, reason, reaction); | |
630 return isolate->heap()->undefined_value(); | |
631 } | |
632 | |
633 RUNTIME_FUNCTION(Runtime_PromiseFulfill) { | |
634 DCHECK(args.length() == 4); | |
635 HandleScope scope(isolate); | |
636 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, promise, 0); | |
637 CONVERT_ARG_HANDLE_CHECKED(Smi, status, 1); | |
638 CONVERT_ARG_HANDLE_CHECKED(Object, value, 2); | |
639 CONVERT_ARG_HANDLE_CHECKED(Symbol, reaction, 3); | |
640 PromiseFulfill(isolate, promise, status, value, reaction); | |
641 return isolate->heap()->undefined_value(); | |
642 } | |
643 | |
644 RUNTIME_FUNCTION(Runtime_EnqueuePromiseReactionJob) { | |
645 HandleScope scope(isolate); | |
646 DCHECK(args.length() == 4); | |
647 CONVERT_ARG_HANDLE_CHECKED(Object, value, 0); | |
648 CONVERT_ARG_HANDLE_CHECKED(Object, tasks, 1); | |
649 CONVERT_ARG_HANDLE_CHECKED(Object, deferred, 2); | |
650 CONVERT_ARG_HANDLE_CHECKED(Object, status, 3); | |
651 EnqueuePromiseReactionJob(isolate, value, tasks, deferred, status); | |
652 return isolate->heap()->undefined_value(); | |
653 } | |
654 | |
655 RUNTIME_FUNCTION(Runtime_EnqueuePromiseResolveThenableJob) { | |
656 HandleScope scope(isolate); | |
657 DCHECK(args.length() == 4); | |
658 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, resolution, 0); | |
659 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, then, 1); | |
660 CONVERT_ARG_HANDLE_CHECKED(JSFunction, resolve, 2); | |
661 CONVERT_ARG_HANDLE_CHECKED(JSFunction, reject, 3); | |
662 Handle<Object> debug_id; | |
663 Handle<Object> debug_name; | |
664 if (isolate->debug()->is_active()) { | |
665 debug_id = | |
666 handle(Smi::FromInt(isolate->GetNextDebugMicrotaskId()), isolate); | |
667 debug_name = isolate->factory()->PromiseResolveThenableJob_string(); | |
668 isolate->debug()->OnAsyncTaskEvent(isolate->factory()->enqueue_string(), | |
669 debug_id, | |
670 Handle<String>::cast(debug_name)); | |
671 } else { | |
672 debug_id = isolate->factory()->undefined_value(); | |
673 debug_name = isolate->factory()->undefined_value(); | |
674 } | |
675 Handle<PromiseResolveThenableJobInfo> info = | |
676 isolate->factory()->NewPromiseResolveThenableJobInfo( | |
677 resolution, then, resolve, reject, debug_id, debug_name); | |
678 isolate->EnqueueMicrotask(info); | |
679 return isolate->heap()->undefined_value(); | |
680 } | |
681 | |
682 RUNTIME_FUNCTION(Runtime_EnqueueMicrotask) { | |
683 HandleScope scope(isolate); | |
684 DCHECK(args.length() == 1); | |
685 CONVERT_ARG_HANDLE_CHECKED(JSFunction, microtask, 0); | |
686 isolate->EnqueueMicrotask(microtask); | |
687 return isolate->heap()->undefined_value(); | |
688 } | |
689 | |
690 RUNTIME_FUNCTION(Runtime_RunMicrotasks) { | |
691 HandleScope scope(isolate); | |
692 DCHECK(args.length() == 0); | |
693 isolate->RunMicrotasks(); | |
694 return isolate->heap()->undefined_value(); | |
695 } | |
696 | |
697 RUNTIME_FUNCTION(Runtime_OrdinaryHasInstance) { | 516 RUNTIME_FUNCTION(Runtime_OrdinaryHasInstance) { |
698 HandleScope scope(isolate); | 517 HandleScope scope(isolate); |
699 DCHECK_EQ(2, args.length()); | 518 DCHECK_EQ(2, args.length()); |
700 CONVERT_ARG_HANDLE_CHECKED(Object, callable, 0); | 519 CONVERT_ARG_HANDLE_CHECKED(Object, callable, 0); |
701 CONVERT_ARG_HANDLE_CHECKED(Object, object, 1); | 520 CONVERT_ARG_HANDLE_CHECKED(Object, object, 1); |
702 RETURN_RESULT_OR_FAILURE( | 521 RETURN_RESULT_OR_FAILURE( |
703 isolate, Object::OrdinaryHasInstance(isolate, callable, object)); | 522 isolate, Object::OrdinaryHasInstance(isolate, callable, object)); |
704 } | 523 } |
705 | 524 |
706 RUNTIME_FUNCTION(Runtime_IsWasmInstance) { | 525 RUNTIME_FUNCTION(Runtime_IsWasmInstance) { |
707 HandleScope scope(isolate); | 526 HandleScope scope(isolate); |
708 DCHECK_EQ(1, args.length()); | 527 DCHECK_EQ(1, args.length()); |
709 CONVERT_ARG_CHECKED(Object, object, 0); | 528 CONVERT_ARG_CHECKED(Object, object, 0); |
710 bool is_wasm_instance = | 529 bool is_wasm_instance = |
711 object->IsJSObject() && wasm::IsWasmInstance(JSObject::cast(object)); | 530 object->IsJSObject() && wasm::IsWasmInstance(JSObject::cast(object)); |
712 return *isolate->factory()->ToBoolean(is_wasm_instance); | 531 return *isolate->factory()->ToBoolean(is_wasm_instance); |
713 } | 532 } |
714 | 533 |
715 RUNTIME_FUNCTION(Runtime_Typeof) { | 534 RUNTIME_FUNCTION(Runtime_Typeof) { |
716 HandleScope scope(isolate); | 535 HandleScope scope(isolate); |
717 DCHECK_EQ(1, args.length()); | 536 DCHECK_EQ(1, args.length()); |
718 CONVERT_ARG_HANDLE_CHECKED(Object, object, 0); | 537 CONVERT_ARG_HANDLE_CHECKED(Object, object, 0); |
719 return *Object::TypeOf(isolate, object); | 538 return *Object::TypeOf(isolate, object); |
720 } | 539 } |
721 | 540 |
722 } // namespace internal | 541 } // namespace internal |
723 } // namespace v8 | 542 } // namespace v8 |
OLD | NEW |