// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk <jason@zemos.net> // // Permission to use, copy, modify, and/or distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice is present in all copies. The // full license for this software is available in the LICENSE file. #pragma once #define HAVE_IRCD_UTIL_UNWIND_H namespace ircd { inline namespace util { template<class F> struct unwind; template<class F> struct unwind_nominal; template<class F> struct unwind_exceptional; struct unwind_defer; struct unwind_nominal_assertion; struct unwind_exceptional_assertion; } } // // Fundamental scope-unwind utilities establishing actions during destruction // /// Unconditionally executes the provided code when the object goes out of scope. /// template<class F> struct ircd::util::unwind { const F func; unwind(F&& func) :func{std::forward<F>(func)} {} unwind(const unwind &) = delete; unwind &operator=(const unwind &) = delete; ~unwind() noexcept __attribute__((always_inline)) { func(); } }; /// Executes function only if the unwind takes place without active exception /// /// The function is expected to be executed and the likely() should pipeline /// that branch and make this device cheaper to use under normal circumstances. /// template<class F> struct ircd::util::unwind_nominal { const F func; unwind_nominal(F&& func) :func{std::forward<F>(func)} {} ~unwind_nominal() noexcept __attribute__((always_inline)) { if(likely(!std::uncaught_exceptions())) func(); } unwind_nominal(const unwind_nominal &) = delete; }; /// Executes function only if unwind is taking place because exception thrown /// /// The unlikely() intends for the cost of a branch misprediction to be paid /// for fetching and executing this function. This is because we strive to /// optimize the pipeline for the nominal path, making this device as cheap /// as possible to use. /// template<class F> struct ircd::util::unwind_exceptional { const F func; unwind_exceptional(F&& func) :func{std::forward<F>(func)} {} ~unwind_exceptional() noexcept __attribute__((always_inline)) { if(unlikely(std::uncaught_exceptions())) func(); } unwind_exceptional(const unwind_exceptional &) = delete; }; /// Posts function to the event loop at the unwind rather than executing it /// as with nominal/exceptional. /// struct ircd::util::unwind_defer { std::function<void ()> func; template<class F> unwind_defer(F&& func) :func{std::forward<F>(func)} {} unwind_defer(const unwind_defer &) = delete; ~unwind_defer() noexcept; }; /// Asserts that unwind is occurring without any exception. This allows some /// section of code, for example, the latter half of a function, to assert that /// it is not throwing and disrupting the other half. This is useful when no /// exception is expected thus placing other guarantees should not be required /// if this guarantee is met. /// struct ircd::util::unwind_nominal_assertion { ~unwind_nominal_assertion() noexcept { assert(likely(!std::uncaught_exceptions())); } }; /// Complements exceptional::assertion. This triggers an assertion when /// unwinding WITHOUT an exception. /// struct ircd::util::unwind_exceptional_assertion { ~unwind_exceptional_assertion() noexcept { assert(likely(std::uncaught_exceptions())); } };