diff --git a/deps/v8ppc/include/v8.h b/deps/v8ppc/include/v8.h index 71a0d01eba5..e229ed90d23 100644 --- a/deps/v8ppc/include/v8.h +++ b/deps/v8ppc/include/v8.h @@ -2845,6 +2845,17 @@ class V8EXPORT Isolate { */ static Isolate* GetCurrent(); + /** + * Custom callback used by embedders to help V8 determine if it should abort + * when it throws and no internal handler can catch the exception. + * If FLAG_abort_on_uncaught_exception is true, then V8 will abort if either: + * - no custom callback is set. + * - the custom callback set returns true. + * Otherwise it won't abort. + */ + typedef bool (*abort_on_uncaught_exception_t)(); + void SetAbortOnUncaughtException(abort_on_uncaught_exception_t callback); + /** * Methods below this point require holding a lock (using Locker) in * a multi-threaded environment. diff --git a/deps/v8ppc/src/api.cc b/deps/v8ppc/src/api.cc index 96d564ff855..4b1aa6737cf 100644 --- a/deps/v8ppc/src/api.cc +++ b/deps/v8ppc/src/api.cc @@ -5565,6 +5565,11 @@ void Isolate::Enter() { isolate->Enter(); } +void Isolate::SetAbortOnUncaughtException( + abort_on_uncaught_exception_t callback) { + i::Isolate* isolate = reinterpret_cast(this); + isolate->SetAbortOnUncaughtException(callback); +} void Isolate::Exit() { i::Isolate* isolate = reinterpret_cast(this); diff --git a/deps/v8ppc/src/isolate.cc b/deps/v8ppc/src/isolate.cc index 495d928bf0d..13964f7decc 100644 --- a/deps/v8ppc/src/isolate.cc +++ b/deps/v8ppc/src/isolate.cc @@ -1154,18 +1154,26 @@ void Isolate::DoThrow(Object* exception, MessageLocation* location) { thread_local_top()->pending_message_end_pos_ = location->end_pos(); } - // If the abort-on-uncaught-exception flag is specified, abort on any - // exception not caught by JavaScript, even when an external handler is - // present. This flag is intended for use by JavaScript developers, so - // print a user-friendly stack trace (not an internal one). + // If the abort-on-uncaught-exception flag is specified, and if the + // exception is not caught by JavaScript (even when an external handler is + // present). if (fatal_exception_depth == 0 && FLAG_abort_on_uncaught_exception && (report_exception || can_be_caught_externally)) { - fatal_exception_depth++; - fprintf(stderr, "%s\n\nFROM\n", - *MessageHandler::GetLocalizedMessage(message_obj)); - PrintCurrentStackTrace(stderr); - OS::Abort(); + // If the embedder didn't specify a custom uncaught exception callback, + // or if the custom callback determined that V8 should abort, then + // abort + bool should_abort = !abort_on_uncaught_exception_callback_ || + abort_on_uncaught_exception_callback_(); + if (should_abort) { + fatal_exception_depth++; + // This flag is intended for use by JavaScript developers, so + // print a user-friendly stack trace (not an internal one). + fprintf(stderr, "%s\n\nFROM\n", + *MessageHandler::GetLocalizedMessage(message_obj)); + PrintCurrentStackTrace(stderr); + OS::Abort(); + } } } else if (location != NULL && !location->script().is_null()) { // We are bootstrapping and caught an error where the location is set @@ -1341,6 +1349,10 @@ void Isolate::SetCaptureStackTraceForUncaughtExceptions( stack_trace_for_uncaught_exceptions_options_ = options; } +void Isolate::SetAbortOnUncaughtException( + v8::Isolate::abort_on_uncaught_exception_t callback) { + abort_on_uncaught_exception_callback_ = callback; +} bool Isolate::is_out_of_memory() { if (has_pending_exception()) { @@ -1538,7 +1550,8 @@ Isolate::Isolate() date_cache_(NULL), context_exit_happened_(false), deferred_handles_head_(NULL), - optimizing_compiler_thread_(this) { + optimizing_compiler_thread_(this), + abort_on_uncaught_exception_callback_(NULL) { TRACE_ISOLATE(constructor); memset(isolate_addresses_, 0, diff --git a/deps/v8ppc/src/isolate.h b/deps/v8ppc/src/isolate.h index 05b9b06f750..15793069bcf 100644 --- a/deps/v8ppc/src/isolate.h +++ b/deps/v8ppc/src/isolate.h @@ -701,6 +701,9 @@ class Isolate { int frame_limit, StackTrace::StackTraceOptions options); + typedef bool (*abort_on_uncaught_exception_t)(); + void SetAbortOnUncaughtException(abort_on_uncaught_exception_t callback); + // Tells whether the current context has experienced an out of memory // exception. bool is_out_of_memory(); @@ -1305,6 +1308,8 @@ class Isolate { DeferredHandles* deferred_handles_head_; OptimizingCompilerThread optimizing_compiler_thread_; + abort_on_uncaught_exception_t abort_on_uncaught_exception_callback_; + friend class ExecutionAccess; friend class HandleScopeImplementer; friend class IsolateInitializer; diff --git a/deps/v8z/include/v8.h b/deps/v8z/include/v8.h index 71a0d01eba5..e229ed90d23 100644 --- a/deps/v8z/include/v8.h +++ b/deps/v8z/include/v8.h @@ -2845,6 +2845,17 @@ class V8EXPORT Isolate { */ static Isolate* GetCurrent(); + /** + * Custom callback used by embedders to help V8 determine if it should abort + * when it throws and no internal handler can catch the exception. + * If FLAG_abort_on_uncaught_exception is true, then V8 will abort if either: + * - no custom callback is set. + * - the custom callback set returns true. + * Otherwise it won't abort. + */ + typedef bool (*abort_on_uncaught_exception_t)(); + void SetAbortOnUncaughtException(abort_on_uncaught_exception_t callback); + /** * Methods below this point require holding a lock (using Locker) in * a multi-threaded environment. diff --git a/deps/v8z/src/api.cc b/deps/v8z/src/api.cc index 96d564ff855..4b1aa6737cf 100644 --- a/deps/v8z/src/api.cc +++ b/deps/v8z/src/api.cc @@ -5565,6 +5565,11 @@ void Isolate::Enter() { isolate->Enter(); } +void Isolate::SetAbortOnUncaughtException( + abort_on_uncaught_exception_t callback) { + i::Isolate* isolate = reinterpret_cast(this); + isolate->SetAbortOnUncaughtException(callback); +} void Isolate::Exit() { i::Isolate* isolate = reinterpret_cast(this); diff --git a/deps/v8z/src/isolate.cc b/deps/v8z/src/isolate.cc index a55ed917f18..644d7de0dd0 100644 --- a/deps/v8z/src/isolate.cc +++ b/deps/v8z/src/isolate.cc @@ -1156,18 +1156,26 @@ void Isolate::DoThrow(Object* exception, MessageLocation* location) { thread_local_top()->pending_message_end_pos_ = location->end_pos(); } - // If the abort-on-uncaught-exception flag is specified, abort on any - // exception not caught by JavaScript, even when an external handler is - // present. This flag is intended for use by JavaScript developers, so - // print a user-friendly stack trace (not an internal one). + // If the abort-on-uncaught-exception flag is specified, and if the + // exception is not caught by JavaScript (even when an external handler is + // present). if (fatal_exception_depth == 0 && FLAG_abort_on_uncaught_exception && (report_exception || can_be_caught_externally)) { - fatal_exception_depth++; - fprintf(stderr, "%s\n\nFROM\n", - *MessageHandler::GetLocalizedMessage(message_obj)); - PrintCurrentStackTrace(stderr); - OS::Abort(); + // If the embedder didn't specify a custom uncaught exception callback, + // or if the custom callback determined that V8 should abort, then + // abort + bool should_abort = !abort_on_uncaught_exception_callback_ || + abort_on_uncaught_exception_callback_(); + if (should_abort) { + fatal_exception_depth++; + // This flag is intended for use by JavaScript developers, so + // print a user-friendly stack trace (not an internal one). + fprintf(stderr, "%s\n\nFROM\n", + *MessageHandler::GetLocalizedMessage(message_obj)); + PrintCurrentStackTrace(stderr); + OS::Abort(); + } } } else if (location != NULL && !location->script().is_null()) { // We are bootstrapping and caught an error where the location is set @@ -1343,6 +1351,10 @@ void Isolate::SetCaptureStackTraceForUncaughtExceptions( stack_trace_for_uncaught_exceptions_options_ = options; } +void Isolate::SetAbortOnUncaughtException( + v8::Isolate::abort_on_uncaught_exception_t callback) { + abort_on_uncaught_exception_callback_ = callback; +} bool Isolate::is_out_of_memory() { if (has_pending_exception()) { @@ -1542,7 +1554,8 @@ Isolate::Isolate() date_cache_(NULL), context_exit_happened_(false), deferred_handles_head_(NULL), - optimizing_compiler_thread_(this) { + optimizing_compiler_thread_(this), + abort_on_uncaught_exception_callback_(NULL) { TRACE_ISOLATE(constructor); memset(isolate_addresses_, 0, diff --git a/deps/v8z/src/isolate.h b/deps/v8z/src/isolate.h index ac22ecf6120..ba83ca48e14 100644 --- a/deps/v8z/src/isolate.h +++ b/deps/v8z/src/isolate.h @@ -705,6 +705,9 @@ class Isolate { int frame_limit, StackTrace::StackTraceOptions options); + typedef bool (*abort_on_uncaught_exception_t)(); + void SetAbortOnUncaughtException(abort_on_uncaught_exception_t callback); + // Tells whether the current context has experienced an out of memory // exception. bool is_out_of_memory(); @@ -1311,6 +1314,8 @@ class Isolate { DeferredHandles* deferred_handles_head_; OptimizingCompilerThread optimizing_compiler_thread_; + abort_on_uncaught_exception_t abort_on_uncaught_exception_callback_; + friend class ExecutionAccess; friend class HandleScopeImplementer; friend class IsolateInitializer;