OLD | NEW |
1 // Copyright 2006-2008 the V8 project authors. All rights reserved. | 1 // Copyright 2006-2008 the V8 project authors. All rights reserved. |
2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
4 // met: | 4 // met: |
5 // | 5 // |
6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
(...skipping 576 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
587 Handle<Object> exception = Copy(boilerplate); | 587 Handle<Object> exception = Copy(boilerplate); |
588 // TODO(1240995): To avoid having to call JavaScript code to compute | 588 // TODO(1240995): To avoid having to call JavaScript code to compute |
589 // the message for stack overflow exceptions which is very likely to | 589 // the message for stack overflow exceptions which is very likely to |
590 // double fault with another stack overflow exception, we use a | 590 // double fault with another stack overflow exception, we use a |
591 // precomputed message. This is somewhat problematic in that it | 591 // precomputed message. This is somewhat problematic in that it |
592 // doesn't use ReportUncaughtException to determine the location | 592 // doesn't use ReportUncaughtException to determine the location |
593 // from where the exception occurred. It should probably be | 593 // from where the exception occurred. It should probably be |
594 // reworked. | 594 // reworked. |
595 static const char* kMessage = | 595 static const char* kMessage = |
596 "Uncaught RangeError: Maximum call stack size exceeded"; | 596 "Uncaught RangeError: Maximum call stack size exceeded"; |
597 DoThrow(*exception, NULL, kMessage, false); | 597 DoThrow(*exception, NULL, kMessage); |
598 return Failure::Exception(); | 598 return Failure::Exception(); |
599 } | 599 } |
600 | 600 |
601 | 601 |
602 Failure* Top::Throw(Object* exception, MessageLocation* location) { | 602 Failure* Top::Throw(Object* exception, MessageLocation* location) { |
603 DoThrow(exception, location, NULL, false); | 603 DoThrow(exception, location, NULL); |
604 return Failure::Exception(); | 604 return Failure::Exception(); |
605 } | 605 } |
606 | 606 |
607 | 607 |
608 Failure* Top::ReThrow(Object* exception, MessageLocation* location) { | 608 Failure* Top::ReThrow(Object* exception, MessageLocation* location) { |
609 DoThrow(exception, location, NULL, true); | 609 // Check is this exception is externally caught. |
| 610 bool is_caught_externally = false; |
| 611 ShouldReportException(&is_caught_externally); |
| 612 thread_local_.external_caught_exception_ = is_caught_externally; |
| 613 |
| 614 // Set the exception beeing re-thrown. |
| 615 set_pending_exception(exception); |
610 return Failure::Exception(); | 616 return Failure::Exception(); |
611 } | 617 } |
612 | 618 |
613 | 619 |
614 void Top::ScheduleThrow(Object* exception) { | 620 void Top::ScheduleThrow(Object* exception) { |
615 // When scheduling a throw we first throw the exception to get the | 621 // When scheduling a throw we first throw the exception to get the |
616 // error reporting if it is uncaught before rescheduling it. | 622 // error reporting if it is uncaught before rescheduling it. |
617 Throw(exception); | 623 Throw(exception); |
618 thread_local_.scheduled_exception_ = pending_exception(); | 624 thread_local_.scheduled_exception_ = pending_exception(); |
619 thread_local_.external_caught_exception_ = false; | 625 thread_local_.external_caught_exception_ = false; |
(...skipping 97 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
717 // address of the external handler so we can compare the address to | 723 // address of the external handler so we can compare the address to |
718 // determine which one is closer to the top of the stack. | 724 // determine which one is closer to the top of the stack. |
719 bool has_external_handler = (thread_local_.try_catch_handler_ != NULL); | 725 bool has_external_handler = (thread_local_.try_catch_handler_ != NULL); |
720 Address external_handler_address = | 726 Address external_handler_address = |
721 reinterpret_cast<Address>(thread_local_.try_catch_handler_); | 727 reinterpret_cast<Address>(thread_local_.try_catch_handler_); |
722 | 728 |
723 // NOTE: The stack is assumed to grown towards lower addresses. If | 729 // NOTE: The stack is assumed to grown towards lower addresses. If |
724 // the handler is at a higher address than the external address it | 730 // the handler is at a higher address than the external address it |
725 // means that it is below it on the stack. | 731 // means that it is below it on the stack. |
726 | 732 |
727 // Find the top-most try-catch or try-finally handler. | |
728 while (handler != NULL && handler->is_entry()) { | |
729 handler = handler->next(); | |
730 } | |
731 | |
732 // The exception has been externally caught if and only if there is | |
733 // an external handler which is above any JavaScript try-catch or | |
734 // try-finally handlers. | |
735 *is_caught_externally = has_external_handler && | |
736 (handler == NULL || handler->address() > external_handler_address); | |
737 | |
738 // Find the top-most try-catch handler. | 733 // Find the top-most try-catch handler. |
739 while (handler != NULL && !handler->is_try_catch()) { | 734 while (handler != NULL && !handler->is_try_catch()) { |
740 handler = handler->next(); | 735 handler = handler->next(); |
741 } | 736 } |
742 | 737 |
| 738 // The exception has been externally caught if and only if there is |
| 739 // an external handler which is above any JavaScript try-catch but NOT |
| 740 // try-finally handlers. |
| 741 *is_caught_externally = has_external_handler && |
| 742 (handler == NULL || handler->address() > external_handler_address); |
| 743 |
743 // If we have a try-catch handler then the exception is caught in | 744 // If we have a try-catch handler then the exception is caught in |
744 // JavaScript code. | 745 // JavaScript code. |
745 bool is_uncaught_by_js = (handler == NULL); | 746 bool is_uncaught_by_js = (handler == NULL); |
746 | 747 |
747 // If there is no external try-catch handler, we report the | 748 // If there is no external try-catch handler, we report the |
748 // exception if it isn't caught by JavaScript code. | 749 // exception if it isn't caught by JavaScript code. |
749 if (!has_external_handler) return is_uncaught_by_js; | 750 if (!has_external_handler) return is_uncaught_by_js; |
750 | 751 |
751 if (is_uncaught_by_js || handler->address() > external_handler_address) { | 752 if (is_uncaught_by_js || handler->address() > external_handler_address) { |
752 // Only report the exception if the external handler is verbose. | 753 // Only report the exception if the external handler is verbose. |
753 return thread_local_.try_catch_handler_->is_verbose_; | 754 return thread_local_.try_catch_handler_->is_verbose_; |
754 } else { | 755 } else { |
755 // Report the exception if it isn't caught by JavaScript code. | 756 // Report the exception if it isn't caught by JavaScript code. |
756 return is_uncaught_by_js; | 757 return is_uncaught_by_js; |
757 } | 758 } |
758 } | 759 } |
759 | 760 |
760 | 761 |
761 void Top::DoThrow(Object* exception, | 762 void Top::DoThrow(Object* exception, |
762 MessageLocation* location, | 763 MessageLocation* location, |
763 const char* message, | 764 const char* message) { |
764 bool is_rethrow) { | |
765 ASSERT(!has_pending_exception()); | 765 ASSERT(!has_pending_exception()); |
766 ASSERT(!external_caught_exception()); | |
767 | 766 |
768 HandleScope scope; | 767 HandleScope scope; |
769 Handle<Object> exception_handle(exception); | 768 Handle<Object> exception_handle(exception); |
770 | 769 |
| 770 // Determine reporting and whether the exception is caught externally. |
771 bool is_caught_externally = false; | 771 bool is_caught_externally = false; |
772 bool report_exception = (exception != Failure::OutOfMemoryException()) && | 772 bool report_exception = (exception != Failure::OutOfMemoryException()) && |
773 ShouldReportException(&is_caught_externally); | 773 ShouldReportException(&is_caught_externally); |
774 if (is_rethrow) report_exception = false; | |
775 | 774 |
| 775 // Generate the message. |
776 Handle<Object> message_obj; | 776 Handle<Object> message_obj; |
777 MessageLocation potential_computed_location; | 777 MessageLocation potential_computed_location; |
778 bool try_catch_needs_message = | 778 bool try_catch_needs_message = |
779 is_caught_externally && thread_local_.try_catch_handler_->capture_message_; | 779 is_caught_externally && |
| 780 thread_local_.try_catch_handler_->capture_message_; |
780 if (report_exception || try_catch_needs_message) { | 781 if (report_exception || try_catch_needs_message) { |
781 if (location == NULL) { | 782 if (location == NULL) { |
782 // If no location was specified we use a computed one instead | 783 // If no location was specified we use a computed one instead |
783 ComputeLocation(&potential_computed_location); | 784 ComputeLocation(&potential_computed_location); |
784 location = &potential_computed_location; | 785 location = &potential_computed_location; |
785 } | 786 } |
786 Handle<String> stack_trace; | 787 Handle<String> stack_trace; |
787 if (FLAG_trace_exception) stack_trace = StackTrace(); | 788 if (FLAG_trace_exception) stack_trace = StackTrace(); |
788 message_obj = MessageHandler::MakeMessageObject("uncaught_exception", | 789 message_obj = MessageHandler::MakeMessageObject("uncaught_exception", |
789 location, HandleVector<Object>(&exception_handle, 1), stack_trace); | 790 location, HandleVector<Object>(&exception_handle, 1), stack_trace); |
(...skipping 15 matching lines...) Expand all Loading... |
805 Debugger::OnException(exception_handle, report_exception); | 806 Debugger::OnException(exception_handle, report_exception); |
806 | 807 |
807 if (report_exception) { | 808 if (report_exception) { |
808 if (message != NULL) { | 809 if (message != NULL) { |
809 MessageHandler::ReportMessage(message); | 810 MessageHandler::ReportMessage(message); |
810 } else { | 811 } else { |
811 MessageHandler::ReportMessage(location, message_obj); | 812 MessageHandler::ReportMessage(location, message_obj); |
812 } | 813 } |
813 } | 814 } |
814 thread_local_.external_caught_exception_ = is_caught_externally; | 815 thread_local_.external_caught_exception_ = is_caught_externally; |
| 816 |
815 // NOTE: Notifying the debugger or reporting the exception may have caused | 817 // NOTE: Notifying the debugger or reporting the exception may have caused |
816 // new exceptions. For now, we just ignore that and set the pending exception | 818 // new exceptions. For now, we just ignore that and set the pending exception |
817 // to the original one. | 819 // to the original one. |
818 set_pending_exception(*exception_handle); | 820 set_pending_exception(*exception_handle); |
819 } | 821 } |
820 | 822 |
821 | 823 |
822 void Top::TraceException(bool flag) { | 824 void Top::TraceException(bool flag) { |
823 FLAG_trace_exception = flag; | 825 FLAG_trace_exception = flag; |
824 } | 826 } |
(...skipping 69 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
894 Top::break_access_->Lock(); | 896 Top::break_access_->Lock(); |
895 } | 897 } |
896 | 898 |
897 | 899 |
898 ExecutionAccess::~ExecutionAccess() { | 900 ExecutionAccess::~ExecutionAccess() { |
899 Top::break_access_->Unlock(); | 901 Top::break_access_->Unlock(); |
900 } | 902 } |
901 | 903 |
902 | 904 |
903 } } // namespace v8::internal | 905 } } // namespace v8::internal |
OLD | NEW |