Summary: Forewarning: This patch looks big in #LOC changed. I promise it's not that bad, it just moves a lot of content from one file to another. I've gone ahead and left inline comments on Phabricator for sections where this has happened. This patch: 1. Introduces the crash handler API (crash_handler_api.h). 2. Moves information required for out-of-process crash handling into an AllocatorState. This is a trivially-copied POD struct that designed to be recovered from a deceased process, and used by the crash handler to create a GWP-ASan report (along with the other trivially-copied Metadata struct). 3. Implements the crash handler API using the AllocatorState and Metadata. 4. Adds tests for the crash handler. 5. Reimplements the (now optionally linked by the supporting allocator) in-process crash handler (i.e. the segv handler) using the new crash handler API. 6. Minor updates Scudo & Scudo Standalone to fix compatibility. 7. Changed capitalisation of errors (e.g. /s/Use after free/Use After Free). Reviewers: cryptoad, eugenis, jfb Reviewed By: eugenis Subscribers: merge_guards_bot, pcc, jfb, dexonsmith, mgorny, cryptoad, #sanitizers, llvm-commits Tags: #sanitizers, #llvm Differential Revision: https://reviews.llvm.org/D73557
70 lines
2.1 KiB
C++
70 lines
2.1 KiB
C++
//===-- thread_contention.cpp -----------------------------------*- C++ -*-===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "gwp_asan/tests/harness.h"
|
|
|
|
// Note: Compilation of <atomic> and <thread> are extremely expensive for
|
|
// non-opt builds of clang.
|
|
#include <atomic>
|
|
#include <cstdlib>
|
|
#include <thread>
|
|
#include <vector>
|
|
|
|
void asyncTask(gwp_asan::GuardedPoolAllocator *GPA,
|
|
std::atomic<bool> *StartingGun, unsigned NumIterations) {
|
|
while (!*StartingGun) {
|
|
// Wait for starting gun.
|
|
}
|
|
|
|
// Get ourselves a new allocation.
|
|
for (unsigned i = 0; i < NumIterations; ++i) {
|
|
volatile char *Ptr = reinterpret_cast<volatile char *>(
|
|
GPA->allocate(GPA->getAllocatorState()->maximumAllocationSize()));
|
|
// Do any other threads have access to this page?
|
|
EXPECT_EQ(*Ptr, 0);
|
|
|
|
// Mark the page as from malloc. Wait to see if another thread also takes
|
|
// this page.
|
|
*Ptr = 'A';
|
|
std::this_thread::sleep_for(std::chrono::nanoseconds(10000));
|
|
|
|
// Check we still own the page.
|
|
EXPECT_EQ(*Ptr, 'A');
|
|
|
|
// And now release it.
|
|
*Ptr = 0;
|
|
GPA->deallocate(const_cast<char *>(Ptr));
|
|
}
|
|
}
|
|
|
|
void runThreadContentionTest(unsigned NumThreads, unsigned NumIterations,
|
|
gwp_asan::GuardedPoolAllocator *GPA) {
|
|
|
|
std::atomic<bool> StartingGun{false};
|
|
std::vector<std::thread> Threads;
|
|
if (std::thread::hardware_concurrency() < NumThreads) {
|
|
NumThreads = std::thread::hardware_concurrency();
|
|
}
|
|
|
|
for (unsigned i = 0; i < NumThreads; ++i) {
|
|
Threads.emplace_back(asyncTask, GPA, &StartingGun, NumIterations);
|
|
}
|
|
|
|
StartingGun = true;
|
|
|
|
for (auto &T : Threads)
|
|
T.join();
|
|
}
|
|
|
|
TEST_F(CustomGuardedPoolAllocator, ThreadContention) {
|
|
unsigned NumThreads = 4;
|
|
unsigned NumIterations = 10000;
|
|
InitNumSlots(NumThreads);
|
|
runThreadContentionTest(NumThreads, NumIterations, &GPA);
|
|
}
|