Fix Wasm trap handler recursion on exceptions raised early

Check if storage for thread_local variables has been allocated before
attempting to access such variables, as exceptions may be raised in the
thread before this initializion is complete, causing an infinite loop.

Bug: v8:8966
Change-Id: Ifc6223b74999a55bfd0ed2d6ebf054bbffd7e809
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1507714
Commit-Queue: Ben Titzer <titzer@chromium.org>
Reviewed-by: Ben Titzer <titzer@chromium.org>
Cr-Commit-Position: refs/heads/master@{#60852}
This commit is contained in:
Bill Ticehurst 2019-03-22 12:23:09 -07:00 committed by Commit Bot
parent 4dd01774c6
commit 02703a099a
2 changed files with 48 additions and 7 deletions

View File

@ -34,7 +34,48 @@ namespace v8 {
namespace internal { namespace internal {
namespace trap_handler { namespace trap_handler {
// The below struct needed to access the offset in the Thread Environment Block
// to see if the thread local storage for the thread has been allocated yet.
//
// The ThreadLocalStorage pointer is located 12 pointers into the TEB (i.e. at
// offset 0x58 for 64-bit platforms, and 0x2c for 32-bit platforms). This is
// true for x64, x86, ARM, and ARM64 platforms (see the header files in the SDK
// named ksamd64.inc, ks386.inc, ksarm.h, and ksarm64.h respectively).
//
// These offsets are baked into compiled binaries, so can never be changed for
// backwards compatibility reasons.
struct TEB {
PVOID reserved[11];
PVOID thread_local_storage_pointer;
};
bool TryHandleWasmTrap(EXCEPTION_POINTERS* exception) { bool TryHandleWasmTrap(EXCEPTION_POINTERS* exception) {
// VectoredExceptionHandlers need extreme caution. Do as little as possible
// to determine if the exception should be handled or not. Exceptions can be
// thrown very early in a threads life, before the thread has even completed
// initializing. As a demonstrative example, there was a bug (#8966) where an
// exception would be raised before the thread local copy of the
// "__declspec(thread)" variables had been allocated, the handler tried to
// access the thread-local "g_thread_in_wasm_code", which would then raise
// another exception, and an infinite loop ensued.
// First ensure this is an exception type of interest
if (exception->ExceptionRecord->ExceptionCode != EXCEPTION_ACCESS_VIOLATION) {
return false;
}
// See if thread-local storage for __declspec(thread) variables has been
// allocated yet. This pointer is initially null in the TEB until the
// loader has completed allocating the memory for thread_local variables
// and copy constructed their initial values. (Note: Any functions that
// need to run to initialize values may not have run yet, but that is not
// the case for any thread_locals used here).
TEB* pteb = reinterpret_cast<TEB*>(NtCurrentTeb());
if (!pteb->thread_local_storage_pointer) {
return false;
}
// Now safe to run more advanced logic, which may access thread_locals
// Ensure the faulting thread was actually running Wasm code. // Ensure the faulting thread was actually running Wasm code.
if (!IsThreadInWasm()) { if (!IsThreadInWasm()) {
return false; return false;
@ -45,10 +86,6 @@ bool TryHandleWasmTrap(EXCEPTION_POINTERS* exception) {
const EXCEPTION_RECORD* record = exception->ExceptionRecord; const EXCEPTION_RECORD* record = exception->ExceptionRecord;
if (record->ExceptionCode != EXCEPTION_ACCESS_VIOLATION) {
return false;
}
uintptr_t fault_addr = reinterpret_cast<uintptr_t>(record->ExceptionAddress); uintptr_t fault_addr = reinterpret_cast<uintptr_t>(record->ExceptionAddress);
uintptr_t landing_pad = 0; uintptr_t landing_pad = 0;

View File

@ -406,10 +406,14 @@ TEST_P(TrapHandlerTest, TestCrashInWasmWrongCrashType) {
SetupTrapHandler(GetParam()); SetupTrapHandler(GetParam());
#if V8_OS_POSIX #if V8_OS_POSIX
// The V8 default trap handler does not register for SIGFPE, therefore the // On Posix, the V8 default trap handler does not register for SIGFPE,
// thread-in-wasm flag is never reset in this test. We therefore do not check // therefore the thread-in-wasm flag is never reset in this test. We
// the value of this flag. // therefore do not check the value of this flag.
bool check_wasm_flag = GetParam() != kDefault; bool check_wasm_flag = GetParam() != kDefault;
#elif V8_OS_WIN
// On Windows, the trap handler returns immediately if not an exception of
// interest.
bool check_wasm_flag = false;
#else #else
bool check_wasm_flag = true; bool check_wasm_flag = true;
#endif #endif