|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- /*
- * Scope Guard
- * Copyright (C) 2017 offa
- *
- * This file is part of Scope Guard.
- *
- * Scope Guard is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * Scope Guard is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with Scope Guard. If not, see <http://www.gnu.org/licenses/>.
- */
-
- #pragma once
-
- #include <utility>
- #include <type_traits>
-
- namespace sr
- {
-
- enum class invoke_it
- {
- once,
- again
- };
-
-
- template<class Ressource, class Deleter>
- class unique_resource_t
- {
- public:
-
- explicit unique_resource_t(Ressource&& res, Deleter&& deleter, bool shouldrun = true) noexcept : m_resource(std::move(res)),
- m_deleter(std::move(deleter)),
- m_execute_on_destruction(shouldrun)
- {
- }
-
- unique_resource_t(const unique_resource_t&) = delete;
-
- unique_resource_t(unique_resource_t&& other) noexcept : m_resource(std::move(other.m_resource)),
- m_deleter(std::move(other.m_deleter)),
- m_execute_on_destruction(other.m_execute_on_destruction)
- {
- other.release();
- }
-
- ~unique_resource_t()
- {
- invoke(invoke_it::once);
- }
-
-
- void invoke(const invoke_it strategy = invoke_it::once) noexcept
- {
- if( m_execute_on_destruction == true )
- {
- call_deleter_safe();
- }
-
- m_execute_on_destruction = ( strategy == invoke_it::again );
- }
-
- Ressource release() noexcept
- {
- m_execute_on_destruction = false;
- return m_resource;
- }
-
- void reset(Ressource&& res) noexcept
- {
- invoke(invoke_it::again);
- m_resource = std::move(res);
- }
-
- const Ressource& get() const noexcept
- {
- return m_resource;
- }
-
- operator const Ressource&() const noexcept
- {
- return m_resource;
- }
-
- template<class R = Ressource,
- std::enable_if_t<std::is_pointer<R>::value &&
- ( std::is_class<std::remove_pointer_t<R>>::value
- || std::is_union<std::remove_pointer_t<R>>::value ), int> = 0>
- R operator->() const noexcept
- {
- return m_resource;
- }
-
- template<class R = Ressource,
- std::enable_if_t<std::is_pointer<R>::value, int> = 0>
- std::add_lvalue_reference_t<std::remove_pointer_t<Ressource>> operator*() const noexcept
- {
- return *get();
- }
-
- const Deleter& get_deleter() const noexcept
- {
- return m_deleter;
- }
-
- Deleter& get_deleter() noexcept
- {
- return m_deleter;
- }
-
- unique_resource_t& operator=(unique_resource_t&& other) noexcept
- {
- invoke(invoke_it::once);
- m_resource = std::move(other.m_resource);
- m_deleter = std::move(other.m_deleter);
- m_execute_on_destruction = other.m_execute_on_destruction;
- other.release();
-
- return *this;
- }
-
-
- unique_resource_t& operator=(const unique_resource_t&) = delete;
-
-
- private:
-
- void call_deleter_safe() noexcept
- {
- try
- {
- get_deleter()(m_resource);
- }
- catch( ... ) { /* Empty */ }
- }
-
- Ressource m_resource;
- Deleter m_deleter;
- bool m_execute_on_destruction;
- };
-
-
-
- template<class Ressource, class Deleter>
- unique_resource_t<Ressource, Deleter> unique_resource(Ressource&& res, Deleter d) noexcept
- {
- return unique_resource_t<Ressource, Deleter>{std::move(res), std::move(d), true};
- }
-
- template<class Ressource, class Deleter>
- unique_resource_t<Ressource, Deleter> unique_resource_checked(Ressource res, Ressource invalid, Deleter d) noexcept
- {
- return unique_resource_t<Ressource, Deleter>{std::move(res), std::move(d), (res != invalid)};
- }
-
- }
-
-
|