// ***************************************************************************** // * This file is part of the FreeFileSync project. It is distributed under * // * GNU General Public License: http://www.gnu.org/licenses/gpl-3.0 * // * Copyright (C) Zenju (zenju AT freefilesync DOT org) - All Rights Reserved * // ***************************************************************************** #ifndef SCOPE_GUARD_H_8971632487321434 #define SCOPE_GUARD_H_8971632487321434 #include #include #include "type_tools.h" #ifdef ZEN_WIN inline int getUncaughtExceptionCount() { return std::uncaught_exceptions(); } #elif defined ZEN_LINUX || defined ZEN_MAC //std::uncaught_exceptions() currently unsupported on GCC and Clang => clean up ASAP #ifdef ZEN_LINUX static_assert(__GNUC__ < 6 || (__GNUC__ == 6 && (__GNUC_MINOR__ < 2 || (__GNUC_MINOR__ == 2 && __GNUC_PATCHLEVEL__ <= 1))), "check std::uncaught_exceptions support"); #else //std::uncaught_exceptions() requires "mmacosx-version-min=10.12" static_assert(__clang_major__ < 8 || (__clang_major__ == 8 && __clang_minor__ <= 0), "check std::uncaught_exceptions support"); #endif namespace __cxxabiv1 { struct __cxa_eh_globals; extern "C" __cxa_eh_globals* __cxa_get_globals() noexcept; } inline int getUncaughtExceptionCount() { return *(reinterpret_cast(static_cast(static_cast(__cxxabiv1::__cxa_get_globals())) + sizeof(void*))); } #endif //best of Zen, Loki and C++17 namespace zen { //Scope Guard /* auto guardAio = zen::makeGuard([&] { ::CloseHandle(hDir); }); ... guardAio.dismiss(); Scope Exit: ZEN_ON_SCOPE_EXIT(::CloseHandle(hDir)); ZEN_ON_SCOPE_FAIL(UndoPreviousWork()); ZEN_ON_SCOPE_SUCCESS(NotifySuccess()); */ enum class ScopeGuardRunMode { ON_EXIT, ON_SUCCESS, ON_FAIL }; //partially specialize scope guard destructor code and get rid of those pesky MSVC "4127 conditional expression is constant" template inline void runScopeGuardDestructor(F& fun, int /*exeptionCountOld*/, StaticEnum) { try { fun(); } catch (...) { assert(false); } //consistency: don't expect exceptions for ON_EXIT even if "!failed"! } template inline void runScopeGuardDestructor(F& fun, int exeptionCountOld, StaticEnum) { const bool failed = getUncaughtExceptionCount() > exeptionCountOld; if (!failed) fun(); //throw X } template inline void runScopeGuardDestructor(F& fun, int exeptionCountOld, StaticEnum) { const bool failed = getUncaughtExceptionCount() > exeptionCountOld; if (failed) try { fun(); } catch (...) { assert(false); } } template class ScopeGuard { public: explicit ScopeGuard(const F& fun) : fun_(fun) {} explicit ScopeGuard( F&& fun) : fun_(std::move(fun)) {} ScopeGuard(ScopeGuard&& other) : fun_(std::move(other.fun_)), exeptionCount(other.exeptionCount), dismissed(other.dismissed) { other.dismissed = true; } ~ScopeGuard() noexcept(runMode != ScopeGuardRunMode::ON_SUCCESS) { if (!dismissed) runScopeGuardDestructor(fun_, exeptionCount, StaticEnum()); } void dismiss() { dismissed = true; } private: ScopeGuard (const ScopeGuard&) = delete; ScopeGuard& operator=(const ScopeGuard&) = delete; F fun_; const int exeptionCount = getUncaughtExceptionCount(); bool dismissed = false; }; template inline auto makeGuard(F&& fun) { return ScopeGuard>(std::forward(fun)); } } #define ZEN_CONCAT_SUB(X, Y) X ## Y #define ZEN_CONCAT(X, Y) ZEN_CONCAT_SUB(X, Y) #define ZEN_ON_SCOPE_EXIT(X) auto ZEN_CONCAT(dummy, __LINE__) = zen::makeGuard([&]{ X; }); (void)ZEN_CONCAT(dummy, __LINE__); #define ZEN_ON_SCOPE_FAIL(X) auto ZEN_CONCAT(dummy, __LINE__) = zen::makeGuard([&]{ X; }); (void)ZEN_CONCAT(dummy, __LINE__); #define ZEN_ON_SCOPE_SUCCESS(X) auto ZEN_CONCAT(dummy, __LINE__) = zen::makeGuard([&]{ X; }); (void)ZEN_CONCAT(dummy, __LINE__); #endif //SCOPE_GUARD_H_8971632487321434