| Index: chrome_frame/vectored_handler-impl.h
|
| ===================================================================
|
| --- chrome_frame/vectored_handler-impl.h (revision 0)
|
| +++ chrome_frame/vectored_handler-impl.h (revision 0)
|
| @@ -0,0 +1,106 @@
|
| +// Copyright (c) 2009 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#ifndef CHROME_FRAME_VECTORED_HANDLER_IMPL_H_
|
| +#define CHROME_FRAME_VECTORED_HANDLER_IMPL_H_
|
| +#include "chrome_frame/vectored_handler.h"
|
| +
|
| +#if defined(_M_IX86)
|
| +typedef struct _EXCEPTION_REGISTRATION_RECORD {
|
| + struct _EXCEPTION_REGISTRATION_RECORD* Next;
|
| + PVOID Handler;
|
| +} EXCEPTION_REGISTRATION_RECORD;
|
| +#define EXCEPTION_CHAIN_END ((struct _EXCEPTION_REGISTRATION_RECORD*)-1)
|
| +#else
|
| +#error only x86 is supported for now.
|
| +#endif
|
| +
|
| +
|
| +// VEH handler flags settings.
|
| +// These are grabbed from winnt.h for PocketPC.
|
| +// Only EXCEPTION_NONCONTINUABLE in defined in "regular" winnt.h
|
| +// #define EXCEPTION_NONCONTINUABLE 0x1 // Noncontinuable exception
|
| +#define EXCEPTION_UNWINDING 0x2 // Unwind is in progress
|
| +#define EXCEPTION_EXIT_UNWIND 0x4 // Exit unwind is in progress
|
| +#define EXCEPTION_STACK_INVALID 0x8 // Stack out of limits or unaligned
|
| +#define EXCEPTION_NESTED_CALL 0x10 // Nested exception handler call
|
| +#define EXCEPTION_TARGET_UNWIND 0x20 // Target unwind in progress
|
| +#define EXCEPTION_COLLIDED_UNWIND 0x40 // Collided exception handler call
|
| +
|
| +#define EXCEPTION_UNWIND (EXCEPTION_UNWINDING | EXCEPTION_EXIT_UNWIND | \
|
| + EXCEPTION_TARGET_UNWIND | EXCEPTION_COLLIDED_UNWIND)
|
| +
|
| +#define IS_UNWINDING(Flag) (((Flag) & EXCEPTION_UNWIND) != 0)
|
| +#define IS_DISPATCHING(Flag) (((Flag) & EXCEPTION_UNWIND) == 0)
|
| +#define IS_TARGET_UNWIND(Flag) ((Flag) & EXCEPTION_TARGET_UNWIND)
|
| +// End of grabbed section
|
| +
|
| +template <class E>
|
| +LONG WINAPI VectoredHandlerT<E>::VectoredHandler(
|
| + EXCEPTION_POINTERS* exceptionInfo) {
|
| + // TODO(stoyan): Consider reentrancy
|
| + const DWORD exceptionCode = exceptionInfo->ExceptionRecord->ExceptionCode;
|
| +
|
| + // Not interested in non-error exceptions. In this category falls exceptions
|
| + // like:
|
| + // 0x40010006 - OutputDebugStringA. Seen when no debugger is attached
|
| + // (otherwise debugger swallows the exception and prints
|
| + // the string).
|
| + // 0x406D1388 - DebuggerProbe. Used by debug CRT - for example see source
|
| + // code of isatty(). Used to name a thread as well.
|
| + // RPC_E_DISCONNECTED and Co. - COM IPC non-fatal warnings
|
| + // STATUS_BREAKPOINT and Co. - Debugger related breakpoints
|
| +
|
| + if ((exceptionCode & ERROR_SEVERITY_ERROR) != ERROR_SEVERITY_ERROR) {
|
| + return ExceptionContinueSearch;
|
| + }
|
| +
|
| + ++VectoredHandlerT<E>::g_exceptions_seen;
|
| +
|
| + // TODO(stoyan): Check whether exception address is inbetween
|
| + // [IsBadReadPtr, IsBadReadPtr + 0xXX]
|
| +
|
| + const DWORD exceptionFlags = exceptionInfo->ExceptionRecord->ExceptionFlags;
|
| + // I don't think VEH is called on unwind. Just to be safe.
|
| + if (IS_DISPATCHING(exceptionFlags)) {
|
| + if (ModuleHasInstalledSEHFilter())
|
| + return ExceptionContinueSearch;
|
| +
|
| + if (E::IsOurModule(exceptionInfo->ExceptionRecord->ExceptionAddress)) {
|
| + E::WriteDump(exceptionInfo);
|
| + return ExceptionContinueSearch;
|
| + }
|
| +
|
| + // See whether our module is somewhere in the call stack.
|
| + void* back_trace[max_back_trace] = {0};
|
| + // Skip RtlCaptureStackBackTrace and VectoredHandler itself.
|
| + DWORD captured = E::RtlCaptureStackBackTrace(2, max_back_trace - 2,
|
| + &back_trace[0], NULL);
|
| + for (DWORD i = 0; i < captured; ++i) {
|
| + if (E::IsOurModule(back_trace[i])) {
|
| + E::WriteDump(exceptionInfo);
|
| + return ExceptionContinueSearch;
|
| + }
|
| + }
|
| + }
|
| +
|
| + return ExceptionContinueSearch;
|
| +}
|
| +
|
| +template <class E>
|
| +BOOL VectoredHandlerT<E>::ModuleHasInstalledSEHFilter() {
|
| + EXCEPTION_REGISTRATION_RECORD* RegistrationFrame = E::RtlpGetExceptionList();
|
| + // TODO(stoyan): Add the stack limits check and some sanity checks like
|
| + // decreasing addresses of registration records
|
| + while (RegistrationFrame != EXCEPTION_CHAIN_END) {
|
| + if (E::IsOurModule(RegistrationFrame->Handler)) {
|
| + return TRUE;
|
| + }
|
| +
|
| + RegistrationFrame = RegistrationFrame->Next;
|
| + }
|
| +
|
| + return FALSE;
|
| +}
|
| +#endif // CHROME_FRAME_VECTORED_HANDLER_IMPL_H_
|
|
|