Roll src/third_party/skia 99c7c07:4af6580
[chromium-blink-merge.git] / components / crash / app / hard_error_handler_win.cc
blob71bd137b2b94bffe47400d8b546629ea0a9b38f3
1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "components/crash/app/hard_error_handler_win.h"
7 #include <DelayIMP.h>
8 #include <winternl.h>
10 #include "base/basictypes.h"
11 #include "base/strings/string_util.h"
12 #include "components/crash/app/crash_reporter_client.h"
14 namespace breakpad {
16 using crash_reporter::GetCrashReporterClient;
18 namespace {
19 const DWORD kExceptionModuleNotFound = VcppException(ERROR_SEVERITY_ERROR,
20 ERROR_MOD_NOT_FOUND);
21 const DWORD kExceptionEntryPtNotFound = VcppException(ERROR_SEVERITY_ERROR,
22 ERROR_PROC_NOT_FOUND);
23 // This is defined in <ntstatus.h> but we can't include this file here.
24 const DWORD FACILITY_GRAPHICS_KERNEL = 0x1E;
25 const DWORD NT_STATUS_ENTRYPOINT_NOT_FOUND = 0xC0000139;
26 const DWORD NT_STATUS_DLL_NOT_FOUND = 0xC0000135;
28 // We assume that exception codes are NT_STATUS codes.
29 DWORD FacilityFromException(DWORD exception_code) {
30 return (exception_code >> 16) & 0x0FFF;
33 // This is not a generic function. It only works with some |nt_status| values.
34 // Check the strings here http://msdn.microsoft.com/en-us/library/cc704588.aspx
35 // before attempting to use this function.
36 void RaiseHardErrorMsg(long nt_status, const std::string& p1,
37 const std::string& p2) {
38 // If headless just exit silently.
39 if (GetCrashReporterClient()->IsRunningUnattended())
40 return;
42 HMODULE ntdll = ::GetModuleHandleA("NTDLL.DLL");
43 wchar_t* msg_template = NULL;
44 size_t count = ::FormatMessage(
45 FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_IGNORE_INSERTS |
46 FORMAT_MESSAGE_FROM_HMODULE,
47 ntdll,
48 nt_status,
49 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
50 reinterpret_cast<wchar_t*>(&msg_template),
52 NULL);
54 if (!count)
55 return;
56 count += p1.size() + p2.size() + 1;
57 base::string16 message;
58 ::wsprintf(WriteInto(&message, count), msg_template, p1.c_str(), p2.c_str());
59 // The MB_SERVICE_NOTIFICATION causes this message to be displayed by
60 // csrss. This means that we are not creating windows or pumping WM messages
61 // in this process.
62 ::MessageBox(NULL, message.c_str(),
63 L"chrome.exe",
64 MB_OK | MB_SERVICE_NOTIFICATION);
65 ::LocalFree(msg_template);
68 void ModuleNotFoundHardError(const EXCEPTION_RECORD* ex_record) {
69 DelayLoadInfo* dli = reinterpret_cast<DelayLoadInfo*>(
70 ex_record->ExceptionInformation[0]);
71 if (!dli->szDll)
72 return;
73 RaiseHardErrorMsg(NT_STATUS_DLL_NOT_FOUND, dli->szDll, std::string());
76 void EntryPointNotFoundHardError(const EXCEPTION_RECORD* ex_record) {
77 DelayLoadInfo* dli = reinterpret_cast<DelayLoadInfo*>(
78 ex_record->ExceptionInformation[0]);
79 if (!dli->dlp.fImportByName)
80 return;
81 if (!dli->dlp.szProcName)
82 return;
83 if (!dli->szDll)
84 return;
85 RaiseHardErrorMsg(NT_STATUS_ENTRYPOINT_NOT_FOUND,
86 dli->dlp.szProcName, dli->szDll);
89 } // namespace
91 bool HardErrorHandler(EXCEPTION_POINTERS* ex_info) {
92 if (!ex_info)
93 return false;
94 if (!ex_info->ExceptionRecord)
95 return false;
97 long exception = ex_info->ExceptionRecord->ExceptionCode;
98 if (exception == kExceptionModuleNotFound) {
99 ModuleNotFoundHardError(ex_info->ExceptionRecord);
100 return true;
101 } else if (exception == kExceptionEntryPtNotFound) {
102 EntryPointNotFoundHardError(ex_info->ExceptionRecord);
103 return true;
104 } else if (FacilityFromException(exception) == FACILITY_GRAPHICS_KERNEL) {
105 RaiseHardErrorMsg(exception, std::string(), std::string());
106 return true;
108 return false;
111 } // namespace breakpad