2019-04-24 09:47:30 +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
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2020-06-01 22:38:23 +08:00
|
|
|
// UNSUPPORTED: c++03
|
2019-04-24 09:47:30 +08:00
|
|
|
|
|
|
|
#define TESTING_CXA_GUARD
|
|
|
|
#include "../src/cxa_guard_impl.h"
|
2019-09-26 19:32:02 +08:00
|
|
|
#include <cassert>
|
2019-04-24 09:47:30 +08:00
|
|
|
|
2021-08-17 02:07:55 +08:00
|
|
|
#if defined(__clang__)
|
|
|
|
# pragma clang diagnostic ignored "-Wtautological-pointer-compare"
|
|
|
|
#elif defined(__GNUC__)
|
|
|
|
# pragma GCC diagnostic ignored "-Waddress"
|
2020-10-31 05:33:02 +08:00
|
|
|
#endif
|
|
|
|
|
2019-04-24 09:47:30 +08:00
|
|
|
using namespace __cxxabiv1;
|
|
|
|
|
|
|
|
template <class GuardType, class Impl>
|
|
|
|
struct Tests {
|
|
|
|
private:
|
|
|
|
Tests() : g{}, impl(&g) {}
|
|
|
|
GuardType g;
|
|
|
|
Impl impl;
|
|
|
|
|
|
|
|
uint8_t first_byte() {
|
|
|
|
uint8_t first;
|
|
|
|
std::memcpy(&first, &g, 1);
|
|
|
|
return first;
|
|
|
|
}
|
|
|
|
|
|
|
|
void reset() { g = {}; }
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Test the post conditions on cxa_guard_acquire, cxa_guard_abort, and
|
|
|
|
// cxa_guard_release. Specifically, that they leave the first byte with
|
|
|
|
// the value 0 or 1 as specified by the ARM or Itanium specification.
|
|
|
|
static void test() {
|
|
|
|
Tests tests;
|
|
|
|
tests.test_acquire();
|
|
|
|
tests.test_abort();
|
|
|
|
tests.test_release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_acquire() {
|
|
|
|
{
|
|
|
|
reset();
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
assert(impl.cxa_guard_acquire() == INIT_IS_PENDING);
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
reset();
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
assert(impl.cxa_guard_acquire() == INIT_IS_PENDING);
|
|
|
|
impl.cxa_guard_release();
|
|
|
|
assert(first_byte() == 1);
|
|
|
|
assert(impl.cxa_guard_acquire() == INIT_IS_DONE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_release() {
|
|
|
|
{
|
|
|
|
reset();
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
assert(impl.cxa_guard_acquire() == INIT_IS_PENDING);
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
impl.cxa_guard_release();
|
|
|
|
assert(first_byte() == 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void test_abort() {
|
|
|
|
{
|
|
|
|
reset();
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
assert(impl.cxa_guard_acquire() == INIT_IS_PENDING);
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
impl.cxa_guard_abort();
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
assert(impl.cxa_guard_acquire() == INIT_IS_PENDING);
|
|
|
|
assert(first_byte() == 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct NopMutex {
|
|
|
|
bool lock() {
|
|
|
|
assert(!is_locked);
|
|
|
|
is_locked = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
bool unlock() {
|
|
|
|
assert(is_locked);
|
|
|
|
is_locked = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool is_locked = false;
|
|
|
|
};
|
2019-04-24 10:21:13 +08:00
|
|
|
NopMutex global_nop_mutex = {};
|
2019-04-24 09:47:30 +08:00
|
|
|
|
|
|
|
struct NopCondVar {
|
|
|
|
bool broadcast() { return false; }
|
|
|
|
bool wait(NopMutex&) { return false; }
|
|
|
|
};
|
2019-04-24 10:21:13 +08:00
|
|
|
NopCondVar global_nop_cond = {};
|
2019-04-24 09:47:30 +08:00
|
|
|
|
|
|
|
void NopFutexWait(int*, int) { assert(false); }
|
|
|
|
void NopFutexWake(int*) { assert(false); }
|
|
|
|
uint32_t MockGetThreadID() { return 0; }
|
|
|
|
|
2020-10-09 01:36:33 +08:00
|
|
|
int main(int, char**) {
|
2019-04-24 09:47:30 +08:00
|
|
|
{
|
|
|
|
#if defined(_LIBCXXABI_HAS_NO_THREADS)
|
|
|
|
static_assert(CurrentImplementation == Implementation::NoThreads, "");
|
2021-12-09 02:22:52 +08:00
|
|
|
static_assert(std::is_same<SelectedImplementation, NoThreadsGuard>::value, "");
|
2019-04-24 09:47:30 +08:00
|
|
|
#else
|
2021-08-11 22:47:48 +08:00
|
|
|
static_assert(CurrentImplementation == Implementation::GlobalMutex, "");
|
2021-12-09 02:22:52 +08:00
|
|
|
static_assert(std::is_same<SelectedImplementation,
|
|
|
|
GlobalMutexGuard<LibcppMutex, LibcppCondVar, GlobalStatic<LibcppMutex>::instance,
|
|
|
|
GlobalStatic<LibcppCondVar>::instance>>::value,
|
|
|
|
"");
|
2019-04-24 09:47:30 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
{
|
2020-01-20 10:49:14 +08:00
|
|
|
#if (defined(__APPLE__) || defined(__linux__)) && !defined(_LIBCXXABI_HAS_NO_THREADS)
|
2019-04-24 09:47:30 +08:00
|
|
|
assert(PlatformThreadID);
|
|
|
|
#endif
|
2021-08-17 02:07:55 +08:00
|
|
|
if (PlatformThreadID != nullptr) {
|
2019-04-24 09:47:30 +08:00
|
|
|
assert(PlatformThreadID() != 0);
|
|
|
|
assert(PlatformThreadID() == PlatformThreadID());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
{
|
2021-12-09 02:22:52 +08:00
|
|
|
Tests<uint32_t, NoThreadsGuard>::test();
|
|
|
|
Tests<uint64_t, NoThreadsGuard>::test();
|
2019-04-24 09:47:30 +08:00
|
|
|
}
|
|
|
|
{
|
2021-12-09 02:22:52 +08:00
|
|
|
using MutexImpl = GlobalMutexGuard<NopMutex, NopCondVar, global_nop_mutex, global_nop_cond, MockGetThreadID>;
|
2019-04-24 09:47:30 +08:00
|
|
|
Tests<uint32_t, MutexImpl>::test();
|
|
|
|
Tests<uint64_t, MutexImpl>::test();
|
|
|
|
}
|
|
|
|
{
|
2021-12-09 02:22:52 +08:00
|
|
|
using FutexImpl = FutexGuard<&NopFutexWait, &NopFutexWake, &MockGetThreadID>;
|
2019-04-24 09:47:30 +08:00
|
|
|
Tests<uint32_t, FutexImpl>::test();
|
|
|
|
Tests<uint64_t, FutexImpl>::test();
|
|
|
|
}
|
2020-10-09 01:36:33 +08:00
|
|
|
|
|
|
|
return 0;
|
2019-04-24 09:47:30 +08:00
|
|
|
}
|