// Copyright ⓒ 2018-2021 ThePhD. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // // See https://github.com/ThePhD/out_ptr/blob/master/docs/out_ptr.adoc for documentation. #include #include #include TEST_CASE("inout_ptr/exceptions/basic fail in scope", "inout_ptr does not double-delete or destroy a pointer if the C function fails") { SECTION("unique_ptr") { std::unique_ptr> p(nullptr); ficapi_re_create(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_int); int* rawp = static_cast(p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); try { if (ficapi_re_create_fail(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_int, 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp == p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); } } SECTION("unique_ptr") { std::unique_ptr p(nullptr); ficapi_int_re_create(ztd::out_ptr::inout_ptr(p)); int* rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); try { if (ficapi_int_re_create_fail(ztd::out_ptr::inout_ptr(p), 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp == p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); } } SECTION("unique_ptr") { std::unique_ptr p(nullptr); ficapi_handle_re_create(ztd::out_ptr::inout_ptr(p)); ficapi::opaque_handle rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); try { if (ficapi_handle_re_create_fail(ztd::out_ptr::inout_ptr(p), 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp == p.get()); REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); } } SECTION("unique_ptr, void inout_ptr") { std::unique_ptr p(nullptr); ficapi_re_create(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_opaque); ficapi::opaque_handle rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); try { if (ficapi_re_create_fail(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_opaque, 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp == p.get()); REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); } } } TEST_CASE("inout_ptr/exceptions/stateful fail in scope", "inout_ptr type works with stateful deleters in smart pointers") { SECTION("unique_ptr") { std::unique_ptr p(nullptr, ficapi::stateful_deleter{ 0x12345678, ficapi_type::ficapi_type_int }); ficapi_re_create(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_int); int* rawp = static_cast(p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); try { if (ficapi_re_create_fail(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_int, 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); } } SECTION("unique_ptr") { std::unique_ptr p(nullptr, ficapi::stateful_int_deleter{ 0x12345678 }); ficapi_int_re_create(ztd::out_ptr::inout_ptr(p)); int* rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); try { if (ficapi_int_re_create_fail(ztd::out_ptr::inout_ptr(p), 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); } } SECTION("unique_ptr") { std::unique_ptr p(nullptr, ficapi::stateful_handle_deleter{ 0x12345678 }); ficapi_handle_re_create(ztd::out_ptr::inout_ptr(p)); ficapi::opaque_handle rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); try { if (ficapi_handle_re_create_fail(ztd::out_ptr::inout_ptr(p), 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); } } SECTION("unique_ptr, void inout_ptr") { std::unique_ptr p(nullptr, ficapi::stateful_deleter{ 0x12345678, ficapi_type::ficapi_type_int }); ficapi_re_create(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_opaque); ficapi::opaque_handle rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); try { if (ficapi_re_create_fail(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_opaque, 1) != 0) { throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(rawp != nullptr); REQUIRE(ficapi_handle_get_data(rawp) == ficapi_get_dynamic_data()); REQUIRE(p.get_deleter().state() == 0x12345678); } } } TEST_CASE("inout_ptr/exceptions/exception scope escape", "inout_ptr type properly deletes non-nullptr types from earlier") { struct reused_deleter { int store = 0; void operator()(void* x) { ++store; ficapi_delete(x, ficapi_type::ficapi_type_int); } }; struct reused_int_deleter { int store = 0; void operator()(int* x) { ++store; ficapi_int_delete(x); } }; SECTION("unique_ptr") { reused_deleter deleter{}; try { std::unique_ptr> p(nullptr, std::ref(deleter)); ficapi_re_create(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_int); { int* rawp = static_cast(p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(std::addressof(p.get_deleter().get()) == std::addressof(deleter)); REQUIRE(p.get_deleter().get().store == 0); } ficapi_re_create(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_int); { int* rawp = static_cast(p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(std::addressof(p.get_deleter().get()) == std::addressof(deleter)); REQUIRE(p.get_deleter().get().store == 0); } if (ficapi_re_create_fail(ztd::out_ptr::inout_ptr(p), ficapi_type::ficapi_type_opaque, 1) != 0) { int* rawp = static_cast(p.get()); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(std::addressof(p.get_deleter().get()) == std::addressof(deleter)); REQUIRE(p.get_deleter().get().store == 0); throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(deleter.store == 1); } } SECTION("unique_ptr") { reused_int_deleter deleter{}; try { std::unique_ptr> p(nullptr, std::ref(deleter)); ficapi_int_re_create(ztd::out_ptr::inout_ptr(p)); { int* rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(std::addressof(p.get_deleter().get()) == std::addressof(deleter)); REQUIRE(p.get_deleter().get().store == 0); } ficapi_int_re_create(ztd::out_ptr::inout_ptr(p)); { int* rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(std::addressof(p.get_deleter().get()) == std::addressof(deleter)); REQUIRE(p.get_deleter().get().store == 0); } if (ficapi_int_re_create_fail(ztd::out_ptr::inout_ptr(p), 1) != 0) { int* rawp = p.get(); REQUIRE(rawp != nullptr); REQUIRE(*rawp == ficapi_get_dynamic_data()); REQUIRE(std::addressof(p.get_deleter().get()) == std::addressof(deleter)); REQUIRE(p.get_deleter().get().store == 0); throw std::runtime_error("ficapi: failure to create"); } } catch (const std::runtime_error&) { REQUIRE(deleter.store == 1); } } }