2019-08-02 01:53:25 +08:00
|
|
|
//===-- thread_contention.cpp -----------------------------------*- C++ -*-===//
|
[GWP-ASan] Core Guarded Pool Allocator [4].
Summary:
See D60593 for further information.
This patch introduces the core of GWP-ASan, being the guarded pool allocator. This class contains the logic for creating and maintaining allocations in the guarded pool. Its public interface is to be utilised by supporting allocators in order to provide sampled guarded allocation behaviour.
This patch also contains basic functionality tests of the allocator as unittests. The error-catching behaviour will be tested in upcoming patches that use Scudo as an implementing allocator.
Reviewers: vlad.tsyrklevich, eugenis, jfb
Reviewed By: vlad.tsyrklevich
Subscribers: dexonsmith, kubamracek, mgorny, cryptoad, jfb, #sanitizers, llvm-commits, morehouse
Tags: #sanitizers, #llvm
Differential Revision: https://reviews.llvm.org/D62872
llvm-svn: 362636
2019-06-06 03:42:48 +08:00
|
|
|
//
|
|
|
|
// 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->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);
|
|
|
|
}
|