// 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_CTX_THIS_CTX_H /// Interface to the currently running context namespace ircd::ctx { inline namespace this_ctx { /// ios descriptor used by yield(). extern ios::descriptor courtesy_yield_desc; struct ctx &cur() noexcept; ///< Assumptional reference to *current const uint64_t &id() noexcept; // Unique ID for cur ctx string_view name() noexcept; // Optional label for cur ctx ulong cycles() noexcept; // misc profiling related bool interruption_requested() noexcept; // interruption(cur()) void interruption_point(); // throws if interruption_requested() bool interruptible() noexcept; void interruptible(const bool &, std::nothrow_t) noexcept; void interruptible(const bool &); void yield(); // Allow other contexts to run before returning. }} namespace ircd::ctx { /// Points to the currently running context or null for main stack (do not modify) extern thread_local ctx *current; } namespace ircd { namespace this_ctx = ctx::this_ctx; } /// Post the currently running context to the event queue and then suspend to /// allow other contexts in the queue to run. /// /// Until we have our own queue the ios queue makes no guarantees if the queue /// is FIFO or LIFO etc :-/ It is generally bad practice to use this function, /// as one should make the effort to devise a specific cooperative strategy for /// how context switching occurs rather than this coarse/brute technique. inline void ircd::ctx::this_ctx::yield() { ircd::post { courtesy_yield_desc, ios::synchronous }; } inline void ircd::ctx::this_ctx::interruptible(const bool &b, std::nothrow_t) noexcept { interruptible(cur(), b); } inline bool ircd::ctx::this_ctx::interruptible() noexcept { return interruptible(cur()); } /// Throws interrupted if the currently running context was interrupted /// and clears the interrupt flag. inline void ircd::ctx::this_ctx::interruptible(const bool &b) { const bool theirs { interruptible(cur()) }; if(unlikely(theirs && !b && interruption_requested())) interruption_point(); interruptible(cur(), b); if(unlikely(!theirs && b && interruption_requested())) interruption_point(); } /// Returns true if the currently running context was interrupted and clears /// the interrupt flag. inline bool ircd::ctx::this_ctx::interruption_requested() noexcept { return interruption(cur()) || termination(cur()); } /// Calculate the current TSC (reference cycle count) accumulated for this /// context only. This is done by first calculating a cycle count for the /// current slice/epoch (see: ctx/prof.h) which is where the RDTSC sample /// occurs. This count is added to an accumulator value saved in the ctx /// structure. The accumulator value is updated at the end of each execution /// slice, thus giving us the cycle count for this ctx only, up to this point. extern inline ulong __attribute__((flatten, always_inline, gnu_inline, artificial)) ircd::ctx::this_ctx::cycles() noexcept { const auto slice(prof::cur_slice_cycles()); const auto accumulated(cycles(cur())); return accumulated + slice; } /// View the name of the currently running context, or "*" if no context is /// currently running. inline ircd::string_view ircd::ctx::this_ctx::name() noexcept { return current? name(cur()) : "*"_sv; } /// 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 & __attribute__((always_inline)) ircd::ctx::this_ctx::cur() noexcept { assert(current); return *current; }