/* * Based on Loki::ScopeGuard */ #pragma once #include <utility> namespace scope_guard_util { template <typename FunctionType> class ScopeGuard { public: void dismiss() noexcept { dismissed_ = true; } explicit ScopeGuard(const FunctionType &fn) : function_(fn) {} explicit ScopeGuard(FunctionType &&fn) : function_(std::move(fn)) {} ScopeGuard(ScopeGuard &&other) : dismissed_(other.dismissed_), function_(std::move(other.function_)) { other.dismissed_ = true; } ~ScopeGuard() noexcept { if (!dismissed_) execute(); } private: void* operator new(size_t) = delete; void execute() noexcept { function_(); } bool dismissed_ = false; FunctionType function_; }; template <typename FunctionType> ScopeGuard<typename std::decay<FunctionType>::type> make_guard(FunctionType &&fn) { return ScopeGuard<typename std::decay<FunctionType>::type>{ std::forward<FunctionType>(fn)}; } namespace detail { enum class ScopeGuardOnExit {}; template <typename FunctionType> ScopeGuard<typename std::decay<FunctionType>::type> operator+(detail::ScopeGuardOnExit, FunctionType &&fn) { return ScopeGuard<typename std::decay<FunctionType>::type>( std::forward<FunctionType>(fn)); } } } // namespace scope_guard_util #define SCOPE_EXIT_CONCAT2(x, y) x ## y #define SCOPE_EXIT_CONCAT(x, y) SCOPE_EXIT_CONCAT2(x, y) #define SCOPE_EXIT \ auto SCOPE_EXIT_CONCAT(SCOPE_EXIT_STATE, __LINE__) = \ ::scope_guard_util::detail::ScopeGuardOnExit() + [&]() noexcept