// Matrix Construct // // Copyright (C) Matrix Construct Developers, Authors & Contributors // Copyright (C) 2016-2018 Jason Volk // // 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_CTX_THIS_CTX_H namespace ircd::ctx { /// Interface to the currently running context inline namespace this_ctx { struct critical_assertion; // Assert no yielding for a section struct exception_handler; // Must be present to yield in a handler struct ctx &cur(); ///< Assumptional reference to *current const uint64_t &id(); // Unique ID for cur ctx string_view name(); // Optional label for cur ctx void wait(); // Returns when context is woken up. void yield(); // Allow other contexts to run before returning. void interruption_point(); // throws interrupted if interruption_requested() bool interruption_requested(); // interruption(cur()) struct stack_usage_assertion; // Assert safety factor (see ctx/prof.h) size_t stack_usage_here(); // calculates stack usage at call. // Return remaining time if notified; or <= 0 if not, and timeout thrown on throw overloads microseconds wait(const microseconds &, const std::nothrow_t &); template nothrow_overload wait(const duration &); template throw_overload wait(const duration &); // Returns false if notified; true if time point reached, timeout thrown on throw_overloads bool wait_until(const time_point &tp, const std::nothrow_t &); template nothrow_overload wait_until(const time_point &tp); template throw_overload wait_until(const time_point &tp); // Ignores notes. Throws if interrupted. void sleep_until(const time_point &tp); template void sleep(const duration &); void sleep(const int &secs); }} namespace ircd::ctx { /// Points to the currently running context or null for main stack (do not modify) extern __thread ctx *current; } /// An instance of stack_usage_assertion is placed on a ctx stack where one /// wants to test the stack usage at both construction and destruction points /// to ensure it is less than the value set in ctx::prof::settings which is /// generally some engineering safety factor of 2-3 etc. This should not be /// entirely relied upon except during debug builds, however we may try to /// provide an optimized build mode enabling these to account for any possible /// differences in the stack between the environments. /// struct ircd::ctx::this_ctx::stack_usage_assertion { stack_usage_assertion(); ~stack_usage_assertion() noexcept; }; /// An instance of critical_assertion detects an attempt to context switch. /// /// For when the developer specifically does not want any yielding in a /// section or anywhere up the stack from it. This device does not prevent /// a switch and may carry no meaning outside of debug-mode compilation. It is /// good practice to use this device even when it appears obvious the /// section's callgraph has no chance of yielding: code changes, and everything /// up the graph can change without taking notice of your section. /// class ircd::ctx::this_ctx::critical_assertion { bool theirs; public: critical_assertion(); ~critical_assertion() noexcept; }; /// An instance of exception_handler must be present to allow a context /// switch inside a catch block. This is due to ABI limitations that stack /// exceptions with thread-local assumptions and don't expect catch blocks /// on the same thread to interleave when we switch the stack. /// /// We first increment the refcount for the caught exception so it remains /// intuitively accessible for the rest of the catch block. Then the presence /// of this object makes the ABI believe the catch block has ended. /// /// The exception cannot then be rethrown. DO NOT RETHROW THE EXCEPTION. /// struct ircd::ctx::this_ctx::exception_handler :std::exception_ptr { exception_handler() noexcept; exception_handler(exception_handler &&) = delete; exception_handler(const exception_handler &) = delete; exception_handler &operator=(exception_handler &&) = delete; exception_handler &operator=(const exception_handler &) = delete; }; /// This overload matches ::sleep() and acts as a drop-in for ircd contexts. /// interruption point. inline void ircd::ctx::this_ctx::sleep(const int &secs) { sleep(seconds(secs)); } /// Yield the context for a period of time and ignore notifications. sleep() /// is like wait() but it only returns after the timeout and not because of a /// note. /// interruption point. template void ircd::ctx::this_ctx::sleep(const duration &d) { sleep_until(steady_clock::now() + d); } /// Wait for a notification until a point in time. If there is a notification /// then context continues normally. If there's never a notification then an /// exception (= timeout) is thrown. /// interruption point. template ircd::throw_overload ircd::ctx::this_ctx::wait_until(const time_point &tp) { if(wait_until(tp)) throw E(); } /// Wait for a notification until a point in time. If there is a notification /// then returns true. If there's never a notification then returns false. /// interruption point. this is not noexcept. template ircd::nothrow_overload ircd::ctx::this_ctx::wait_until(const time_point &tp) { return wait_until(tp, std::nothrow); } /// Wait for a notification for at most some amount of time. If the duration is /// reached without a notification then E (= timeout) is thrown. Otherwise, /// returns the time remaining on the duration. /// interruption point template ircd::throw_overload ircd::ctx::this_ctx::wait(const duration &d) { const auto ret(wait(d)); return ret <= duration(0)? throw E() : ret; } /// Wait for a notification for some amount of time. This function returns /// when a context is notified. It always returns the duration remaining which /// will be <= 0 to indicate a timeout without notification. /// interruption point. this is not noexcept. template ircd::nothrow_overload ircd::ctx::this_ctx::wait(const duration &d) { using std::chrono::duration_cast; const auto ret(wait(duration_cast(d), std::nothrow)); return duration_cast(ret); } /// Reference to the currently running context. Call if you expect to be in a /// context. Otherwise use the ctx::current pointer. inline ircd::ctx::ctx & ircd::ctx::this_ctx::cur() { assert(current); return *current; }