|
|
@@ -19,7 +19,6 @@ |
|
|
|
*/ |
|
|
|
|
|
|
|
#include "unique_resource.h" |
|
|
|
#include <functional> |
|
|
|
#include <catch.hpp> |
|
|
|
#include <trompeloeil.hpp> |
|
|
|
|
|
|
@@ -36,211 +35,254 @@ namespace |
|
|
|
MAKE_MOCK1(deleter, void(Handle)); |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
struct ThrowOnCopyMock |
|
|
|
{ |
|
|
|
ThrowOnCopyMock() { } |
|
|
|
|
|
|
|
ThrowOnCopyMock(const ThrowOnCopyMock&) |
|
|
|
{ |
|
|
|
throw std::exception{}; |
|
|
|
} |
|
|
|
|
|
|
|
MAKE_CONST_MOCK1(deleter, void(ThrowOnCopyMock)); |
|
|
|
|
|
|
|
ThrowOnCopyMock& operator=(const ThrowOnCopyMock&) |
|
|
|
{ |
|
|
|
throw std::exception{}; |
|
|
|
} |
|
|
|
}; |
|
|
|
|
|
|
|
struct NotNothrowMoveMock |
|
|
|
{ |
|
|
|
NotNothrowMoveMock(CallMock* mo) : m_mock(mo) { } |
|
|
|
NotNothrowMoveMock(const NotNothrowMoveMock& other) : m_mock(other.m_mock) { } |
|
|
|
NotNothrowMoveMock(NotNothrowMoveMock&& other) noexcept(false) : m_mock(other.m_mock) { } |
|
|
|
|
|
|
|
void operator()(Handle h) const |
|
|
|
{ |
|
|
|
m_mock->deleter(h); |
|
|
|
} |
|
|
|
|
|
|
|
NotNothrowMoveMock& operator=(const NotNothrowMoveMock&) |
|
|
|
{ |
|
|
|
throw "Not implemented"; |
|
|
|
} |
|
|
|
|
|
|
|
NotNothrowMoveMock& operator=(NotNothrowMoveMock&&) |
|
|
|
{ |
|
|
|
throw "Not implemented"; |
|
|
|
} |
|
|
|
|
|
|
|
CallMock* m_mock; |
|
|
|
|
|
|
|
}; |
|
|
|
|
|
|
|
struct ConditialThrowOnCopyMock |
|
|
|
{ |
|
|
|
explicit ConditialThrowOnCopyMock(Handle h, bool shouldThrow) : m_handle(h), m_shouldThrow(shouldThrow) |
|
|
|
{ |
|
|
|
} |
|
|
|
|
|
|
|
ConditialThrowOnCopyMock(const ConditialThrowOnCopyMock& other) : m_handle(other.m_handle), m_shouldThrow(other.m_shouldThrow) |
|
|
|
{ |
|
|
|
if( m_shouldThrow == true ) |
|
|
|
{ |
|
|
|
throw std::exception{}; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
ConditialThrowOnCopyMock(ConditialThrowOnCopyMock&&) = default; |
|
|
|
|
|
|
|
ConditialThrowOnCopyMock& operator=(const ConditialThrowOnCopyMock& other) |
|
|
|
{ |
|
|
|
if( &other != this ) |
|
|
|
{ |
|
|
|
m_handle = other.m_handle; |
|
|
|
m_shouldThrow = other.m_shouldThrow; |
|
|
|
|
|
|
|
if( m_shouldThrow == true ) |
|
|
|
{ |
|
|
|
throw std::exception{}; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
return *this; |
|
|
|
} |
|
|
|
|
|
|
|
ConditialThrowOnCopyMock& operator=(ConditialThrowOnCopyMock&&) = default; |
|
|
|
|
|
|
|
Handle m_handle; |
|
|
|
bool m_shouldThrow; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
CallMock m; |
|
|
|
|
|
|
|
void deleter(Handle h) |
|
|
|
{ |
|
|
|
m.deleter(h); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
TEST_CASE("deleter called on destruction", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
|
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("deleter is not called if released", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)).TIMES(0); |
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
guard.release(); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("deleter called if checked valid", "[UniqueResource]") |
|
|
|
TEST_CASE("construction with move", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
auto guard = sr::unique_resource_checked(Handle{3}, Handle{6}, deleter); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("deleter not called if checked invalid", "[UniqueResource]") |
|
|
|
TEST_CASE("construction with copy", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)).TIMES(0); |
|
|
|
auto guard = sr::unique_resource_checked(Handle{3}, Handle{3}, deleter); |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
const Handle h{3}; |
|
|
|
const auto d = [](auto v) { m.deleter(v); }; |
|
|
|
auto guard = sr::make_unique_resource(h, d); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("release returns reference to resource", "[UniqueResource]") |
|
|
|
TEST_CASE("construction with copy calls deleter and rethrows on failed copy", "[UniqueResource]") |
|
|
|
{ |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [](auto) { }); |
|
|
|
const auto result = guard.release(); |
|
|
|
REQUIRE_THROWS([] { |
|
|
|
const ThrowOnCopyMock noMove; |
|
|
|
const auto d = [](const auto&) { m.deleter(3); }; |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
|
|
|
|
REQUIRE(3 == result); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("move releases moved-from object", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
auto movedFrom = sr::unique_resource(Handle{3}, deleter); |
|
|
|
auto guard = std::move(movedFrom); |
|
|
|
static_cast<void>(guard); |
|
|
|
sr::unique_resource<decltype(noMove), decltype(d)> guard{noMove, d}; |
|
|
|
static_cast<void>(guard); |
|
|
|
}()); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("move transfers state", "[UniqueResource]") |
|
|
|
TEST_CASE("move-construction with move", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
auto movedFrom = sr::unique_resource(Handle{3}, deleter); |
|
|
|
auto movedFrom = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
auto guard = std::move(movedFrom); |
|
|
|
static_cast<void>(guard); |
|
|
|
CHECK(guard.get() == 3); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("move transfers state if released", "[UniqueResource]") |
|
|
|
TEST_CASE("move-construction with copy", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(ANY(Handle))).TIMES(0); // TODO: Use ANY(T) |
|
|
|
|
|
|
|
auto movedFrom = sr::unique_resource(Handle{3}, deleter); |
|
|
|
movedFrom.release(); |
|
|
|
CallMock mock; |
|
|
|
REQUIRE_CALL(mock, deleter(3)); |
|
|
|
const NotNothrowMoveMock notNothrow{&mock}; |
|
|
|
Handle h{3}; |
|
|
|
sr::unique_resource<Handle, decltype(notNothrow)> movedFrom{h, notNothrow}; |
|
|
|
auto guard = std::move(movedFrom); |
|
|
|
static_cast<void>(guard); |
|
|
|
CHECK(guard.get() == 3); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("move assignment releases moved-from object", "[UniqueResource]") |
|
|
|
TEST_CASE("move assignment calls deleter", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(5)); |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
auto moveFrom = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
REQUIRE_CALL(m, deleter(4)); |
|
|
|
|
|
|
|
auto movedFrom = sr::unique_resource(Handle{3}, deleter); |
|
|
|
auto guard = sr::unique_resource(Handle{5}, deleter); |
|
|
|
guard = std::move(movedFrom); |
|
|
|
static_cast<void>(guard); |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
auto guard = sr::make_unique_resource(Handle{4}, deleter); |
|
|
|
guard = std::move(moveFrom); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("move assignment transfers state", "[UniqueResource]") |
|
|
|
TEST_CASE("deleter called on destruction", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(5)); |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
|
|
|
|
auto movedFrom = sr::unique_resource(Handle{3}, deleter); |
|
|
|
auto guard = sr::unique_resource(Handle{5}, deleter); |
|
|
|
guard = std::move(movedFrom); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("move assignment transfers state if released", "[UniqueResource]") |
|
|
|
TEST_CASE("reset calls deleter", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(5)); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
|
|
|
|
auto movedFrom = sr::unique_resource(Handle{3}, deleter); |
|
|
|
movedFrom.release(); |
|
|
|
auto guard = sr::unique_resource(Handle{5}, deleter); |
|
|
|
guard = std::move(movedFrom); |
|
|
|
static_cast<void>(guard); |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
guard.reset(); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("no exception propagation from deleter", "[UniqueResource]") |
|
|
|
TEST_CASE("reset does not call deleter if released", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_NOTHROW([] { |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [](auto) { throw "Don't propagate this!"; }); |
|
|
|
static_cast<void>(guard); |
|
|
|
}()); |
|
|
|
REQUIRE_CALL(m, deleter(3)).TIMES(0); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
guard.release(); |
|
|
|
guard.reset(); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("invoke executes deleter on resource", "[UniqueResource]") |
|
|
|
TEST_CASE("reset sets new value and calls deleter on previous", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
|
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
guard.invoke(); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("invoke executes only multiple times if again strategy", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)).TIMES(3); |
|
|
|
|
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
guard.invoke(sr::invoke_it::again); |
|
|
|
guard.invoke(sr::invoke_it::again); |
|
|
|
REQUIRE_CALL(m, deleter(7)); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
guard.reset(Handle{7}); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("invoke does nothing if released", "[UniqueResource]") |
|
|
|
TEST_CASE("release disables deleter", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(ANY(Handle))).TIMES(0); |
|
|
|
|
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
REQUIRE_CALL(m, deleter(3)).TIMES(0); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
guard.release(); |
|
|
|
guard.invoke(sr::invoke_it::once); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("invoke executes after release if again strategy", "[UniqueResource]") |
|
|
|
TEST_CASE("get returns resource", "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
|
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
guard.release(); |
|
|
|
guard.invoke(sr::invoke_it::again); |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
CHECK(guard.get() == 3); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("invoke does not propagate exception", "[UniqueResource]") |
|
|
|
TEST_CASE("pointer access returns resource" "[UniqueResource]") |
|
|
|
{ |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [](auto) { throw "Don't propagate this!"; }); |
|
|
|
REQUIRE_NOTHROW(guard.invoke()); |
|
|
|
const auto p = std::make_pair(3, 4); |
|
|
|
auto guard = sr::make_unique_resource(&p, [](auto*) { }); |
|
|
|
REQUIRE(guard->first == 3); |
|
|
|
REQUIRE(guard->second == 4); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("reset releases old ressource", "[UniqueResource]") |
|
|
|
TEST_CASE("pointer dereference returns resource" "[UniqueResource]") |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
REQUIRE_CALL(m, deleter(7)); |
|
|
|
|
|
|
|
auto guard = sr::unique_resource(Handle{3}, deleter); |
|
|
|
guard.reset(Handle{7}); |
|
|
|
Handle h{5}; |
|
|
|
auto guard = sr::make_unique_resource(PtrHandle{&h}, [](auto*) { }); |
|
|
|
REQUIRE(*guard == 5); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("reset sets ressource", "[UniqueResource]") |
|
|
|
TEST_CASE("deleter access", "[UniqueResource]") |
|
|
|
{ |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [](auto) { }); |
|
|
|
guard.reset(Handle{7}); |
|
|
|
REQUIRE(guard.get() == 7); |
|
|
|
} |
|
|
|
auto guard = sr::make_unique_resource(Handle{3}, deleter); |
|
|
|
guard.release(); |
|
|
|
|
|
|
|
TEST_CASE("get accesses ressource", "[UniqueResource]") |
|
|
|
{ |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [](auto) { }); |
|
|
|
REQUIRE(guard.get() == 3); |
|
|
|
{ |
|
|
|
REQUIRE_CALL(m, deleter(8)); |
|
|
|
guard.get_deleter()(8); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("conversion operator", "[UniqueResource]") |
|
|
|
TEST_CASE("make unique resource", "[UniqueResource]") |
|
|
|
{ |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [](auto) { }); |
|
|
|
const auto& ref = guard; |
|
|
|
REQUIRE(ref == 3); |
|
|
|
REQUIRE_CALL(m, deleter(7)); |
|
|
|
auto guard = sr::make_unique_resource(Handle{7}, deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("pointer access operator" "[UniqueResource]") |
|
|
|
TEST_CASE("make unique resource with reference wrapper", "[UniqueResource]") |
|
|
|
{ |
|
|
|
const auto p = std::make_pair(3, 4); |
|
|
|
auto guard = sr::unique_resource(&p, [](auto) { }); |
|
|
|
const auto x = guard.operator->(); |
|
|
|
REQUIRE(x->first == 3); |
|
|
|
REQUIRE_CALL(m, deleter(3)); |
|
|
|
Handle h{3}; |
|
|
|
auto guard = sr::make_unique_resource(std::ref(h), deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("dereference operator", "[UniqueResource]") |
|
|
|
TEST_CASE("make unique resource checked", "[UniqueResource]") |
|
|
|
{ |
|
|
|
Handle h{4}; |
|
|
|
auto guard = sr::unique_resource(PtrHandle{&h}, [](auto) { }); |
|
|
|
const auto x = guard.operator*(); |
|
|
|
REQUIRE(x == 4); |
|
|
|
REQUIRE_CALL(m, deleter(4)); |
|
|
|
auto guard = sr::make_unique_resource_checked(Handle{4}, Handle{-1}, deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|
|
|
|
TEST_CASE("deleter access", "[UniqueResource]") |
|
|
|
TEST_CASE("make unique resource checked releases if invalid", "[UniqueResource]") |
|
|
|
{ |
|
|
|
std::size_t value{0}; |
|
|
|
auto guard = sr::unique_resource(Handle{3}, [&value](auto v) { value = v; }); |
|
|
|
REQUIRE(value == 0); |
|
|
|
guard.get_deleter()(6); |
|
|
|
REQUIRE(value == 6); |
|
|
|
auto guard = sr::make_unique_resource_checked(Handle{-1}, Handle{-1}, deleter); |
|
|
|
static_cast<void>(guard); |
|
|
|
} |
|
|
|
|