1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
base / win / wrapped_window_proc_unittest.cc [blame]
// Copyright 2011 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/win/wrapped_window_proc.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace {
DWORD kExceptionCode = 12345;
WPARAM kCrashMsg = 98765;
// A trivial WindowProc that generates an exception.
LRESULT CALLBACK TestWindowProc(HWND hwnd,
UINT message,
WPARAM wparam,
LPARAM lparam) {
if (message == kCrashMsg)
RaiseException(kExceptionCode, 0, 0, nullptr);
return DefWindowProc(hwnd, message, wparam, lparam);
}
// This class implements an exception filter that can be queried about a past
// exception.
class TestWrappedExceptionFiter {
public:
TestWrappedExceptionFiter() {
EXPECT_FALSE(s_filter_);
s_filter_ = this;
}
~TestWrappedExceptionFiter() {
EXPECT_EQ(s_filter_, this);
s_filter_ = nullptr;
}
bool called() { return called_; }
// The actual exception filter just records the exception.
static int Filter(EXCEPTION_POINTERS* info) {
EXPECT_FALSE(s_filter_->called_);
if (info->ExceptionRecord->ExceptionCode == kExceptionCode)
s_filter_->called_ = true;
return EXCEPTION_EXECUTE_HANDLER;
}
private:
bool called_ = false;
static TestWrappedExceptionFiter* s_filter_;
};
TestWrappedExceptionFiter* TestWrappedExceptionFiter::s_filter_ = nullptr;
} // namespace.
TEST(WrappedWindowProc, CatchesExceptions) {
HINSTANCE hinst = GetModuleHandle(nullptr);
std::wstring class_name(L"TestClass");
WNDCLASS wc = {0};
wc.lpfnWndProc = base::win::WrappedWindowProc<TestWindowProc>;
wc.hInstance = hinst;
wc.lpszClassName = class_name.c_str();
RegisterClass(&wc);
HWND window = CreateWindow(class_name.c_str(), nullptr, 0, 0, 0, 0, 0,
HWND_MESSAGE, nullptr, hinst, nullptr);
ASSERT_TRUE(window);
// Before generating the exception we make sure that the filter will see it.
TestWrappedExceptionFiter wrapper;
base::win::WinProcExceptionFilter old_filter =
base::win::SetWinProcExceptionFilter(TestWrappedExceptionFiter::Filter);
SendMessage(window, kCrashMsg, 0, 0);
EXPECT_TRUE(wrapper.called());
base::win::SetWinProcExceptionFilter(old_filter);
}