//===----------------------------------------------------------------------===// // // 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 // //===----------------------------------------------------------------------===// #ifndef SUPPORT_EXCEPTION_SAFETY_HELPERS_H #define SUPPORT_EXCEPTION_SAFETY_HELPERS_H #include #include #include #include #include "test_macros.h" #if !defined(TEST_HAS_NO_EXCEPTIONS) template struct ThrowingCopy { static bool throwing_enabled; static int created_by_copying; static int destroyed; int x = 0; // Allows distinguishing between different instances. ThrowingCopy() = default; ThrowingCopy(int value) : x(value) {} ~ThrowingCopy() { ++destroyed; } ThrowingCopy(const ThrowingCopy& other) : x(other.x) { ++created_by_copying; if (throwing_enabled && created_by_copying == N) { throw -1; } } // Defined to silence GCC warnings. For test purposes, only copy construction is considered `created_by_copying`. ThrowingCopy& operator=(const ThrowingCopy& other) { x = other.x; return *this; } friend bool operator==(const ThrowingCopy& lhs, const ThrowingCopy& rhs) { return lhs.x == rhs.x; } friend bool operator<(const ThrowingCopy& lhs, const ThrowingCopy& rhs) { return lhs.x < rhs.x; } static void reset() { created_by_copying = destroyed = 0; } }; template bool ThrowingCopy::throwing_enabled = true; template int ThrowingCopy::created_by_copying = 0; template int ThrowingCopy::destroyed = 0; template struct std::hash> { std::size_t operator()(const ThrowingCopy& value) const { return value.x; } }; template void test_exception_safety_throwing_copy(Func&& func) { using T = ThrowingCopy; T::reset(); T in[Size]; try { func(in, in + Size); assert(false); // The function call above should throw. } catch (int) { assert(T::created_by_copying == ThrowOn); assert(T::destroyed == ThrowOn - 1); // No destructor call for the partially-constructed element. } } // Destroys the container outside the user callback to avoid destroying extra elements upon throwing (which would // complicate asserting that the expected number of elements was destroyed). template void test_exception_safety_throwing_copy_container(Func&& func) { using T = ThrowingCopy; T::throwing_enabled = false; T in[Size]; Container c(in, in + Size); T::throwing_enabled = true; T::reset(); try { func(std::move(c)); assert(false); // The function call above should throw. } catch (int) { assert(T::created_by_copying == ThrowOn); assert(T::destroyed == ThrowOn - 1); // No destructor call for the partially-constructed element. } } #endif // !defined(TEST_HAS_NO_EXCEPTIONS) #endif // SUPPORT_EXCEPTION_SAFETY_HELPERS_H