mirror of
https://github.com/chromium/crashpad.git
synced 2024-12-31 01:43:03 +08:00
cc166d71f4
This is a follow-up to c8a016b99d97, following the post-landing discussion at https://chromium-review.googlesource.com/c/crashpad/crashpad/+/1393921/5#message-2058541d8c4505d20a990ab7734cd758e437a5f7 base::size, and std::size that will eventually replace it when C++17 is assured, does not allow the size of non-static data members to be taken in constant expression context. The remaining uses of ArraySize are in: minidump/minidump_exception_writer.cc (×1) minidump/minidump_system_info_writer.cc (×2, also uses base::size) snapshot/cpu_context.cc (×4, also uses base::size) util/misc/arraysize_test.cc (×10, of course) The first of these occurs when initializing a constexpr variable. All others are in expressions used with static_assert. Includes: Update mini_chromium to 737433ebade4d446643c6c07daae02a67e8deccao f701716d9546 Add Windows ARM64 build target to mini_chromium 87a95a3d6ac2 Remove the arraysize macro 1f7255ead1f7 Placate MSVC in areas of base::size usage 737433ebade4 Add cast Bug: chromium:837308 Change-Id: I6a5162654461b1bdd9b7b6864d0d71a734bcde19 Reviewed-on: https://chromium-review.googlesource.com/c/1396108 Commit-Queue: Mark Mentovai <mark@chromium.org> Reviewed-by: Mark Mentovai <mark@chromium.org>
136 lines
3.8 KiB
C++
136 lines
3.8 KiB
C++
// Copyright 2016 The Crashpad Authors. All rights reserved.
|
||
//
|
||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
// you may not use this file except in compliance with the License.
|
||
// You may obtain a copy of the License at
|
||
//
|
||
// http://www.apache.org/licenses/LICENSE-2.0
|
||
//
|
||
// Unless required by applicable law or agreed to in writing, software
|
||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
// See the License for the specific language governing permissions and
|
||
// limitations under the License.
|
||
|
||
#include <stdio.h>
|
||
#include <windows.h>
|
||
|
||
#include "base/debug/alias.h"
|
||
#include "base/logging.h"
|
||
#include "base/stl_util.h"
|
||
#include "base/strings/stringprintf.h"
|
||
#include "base/strings/utf_string_conversions.h"
|
||
#include "client/crashpad_client.h"
|
||
#include "client/crashpad_info.h"
|
||
|
||
namespace {
|
||
|
||
DWORD WINAPI Thread1(LPVOID context) {
|
||
HANDLE event = context;
|
||
|
||
// Increase the thread priority as a hacky way to signal to
|
||
// crash_other_program.exe that this is the thread to dump.
|
||
PCHECK(SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_ABOVE_NORMAL));
|
||
|
||
// Let the main thread proceed.
|
||
PCHECK(SetEvent(event));
|
||
|
||
Sleep(INFINITE);
|
||
|
||
NOTREACHED();
|
||
return 0;
|
||
}
|
||
|
||
DWORD WINAPI Thread2(LPVOID dummy) {
|
||
Sleep(INFINITE);
|
||
NOTREACHED();
|
||
return 0;
|
||
}
|
||
|
||
DWORD WINAPI Thread3(LPVOID context) {
|
||
// This is a convenient way to pass the event handle to loader_lock_dll.dll.
|
||
HANDLE event = context;
|
||
PCHECK(SetEnvironmentVariable(
|
||
L"CRASHPAD_TEST_DLL_EVENT",
|
||
base::UTF8ToUTF16(base::StringPrintf("%p", event)).c_str()));
|
||
|
||
HMODULE dll = LoadLibrary(L"loader_lock_dll.dll");
|
||
if (!dll)
|
||
PLOG(FATAL) << "LoadLibrary";
|
||
|
||
// This call is not expected to return.
|
||
if (!FreeLibrary(dll))
|
||
PLOG(FATAL) << "FreeLibrary";
|
||
|
||
NOTREACHED();
|
||
return 0;
|
||
}
|
||
|
||
} // namespace
|
||
|
||
int wmain(int argc, wchar_t* argv[]) {
|
||
crashpad::CrashpadClient client;
|
||
|
||
if (argc == 2) {
|
||
if (!client.SetHandlerIPCPipe(argv[1])) {
|
||
LOG(ERROR) << "SetHandlerIPCPipe";
|
||
return EXIT_FAILURE;
|
||
}
|
||
} else {
|
||
fprintf(stderr, "Usage: %ls <server_pipe_name>\n", argv[0]);
|
||
return EXIT_FAILURE;
|
||
}
|
||
|
||
// Make sure this module has a CrashpadInfo structure.
|
||
crashpad::CrashpadInfo* crashpad_info =
|
||
crashpad::CrashpadInfo::GetCrashpadInfo();
|
||
base::debug::Alias(crashpad_info);
|
||
|
||
HANDLE event = CreateEvent(nullptr,
|
||
false, // bManualReset
|
||
false,
|
||
nullptr);
|
||
if (!event) {
|
||
PLOG(ERROR) << "CreateEvent";
|
||
return EXIT_FAILURE;
|
||
}
|
||
|
||
HANDLE threads[3];
|
||
threads[0] = CreateThread(nullptr, 0, Thread1, event, 0, nullptr);
|
||
|
||
threads[1] = CreateThread(nullptr, 0, Thread2, nullptr, 0, nullptr);
|
||
|
||
// Wait for Thread1() to complete its work and reach its Sleep() before
|
||
// starting the next thread, which will hold the loader lock and potentially
|
||
// block any further progress.
|
||
if (WaitForSingleObject(event, INFINITE) != WAIT_OBJECT_0) {
|
||
PLOG(ERROR) << "WaitForSingleObject";
|
||
return EXIT_FAILURE;
|
||
}
|
||
|
||
// Use the same event object, which was automatically reset.
|
||
threads[2] = CreateThread(nullptr, 0, Thread3, event, 0, nullptr);
|
||
|
||
// Wait for loader_lock_dll.dll to signal that the loader lock is held and
|
||
// won’t be released.
|
||
if (WaitForSingleObject(event, INFINITE) != WAIT_OBJECT_0) {
|
||
PLOG(ERROR) << "WaitForSingleObject";
|
||
return EXIT_FAILURE;
|
||
}
|
||
|
||
// Signal to the parent that everything is ready.
|
||
fprintf(stdout, " ");
|
||
fflush(stdout);
|
||
|
||
// This is not expected to return.
|
||
DWORD count = WaitForMultipleObjects(
|
||
static_cast<DWORD>(base::size(threads)), threads, true, INFINITE);
|
||
if (count == WAIT_FAILED) {
|
||
PLOG(ERROR) << "WaitForMultipleObjects";
|
||
} else {
|
||
LOG(ERROR) << "WaitForMultipleObjects: " << count;
|
||
}
|
||
|
||
return EXIT_FAILURE;
|
||
}
|