// 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_QUEUE_H namespace ircd::ctx { template> struct queue; } template struct ircd::ctx::queue { using opts = dock::opts; private: dock d; std::deque q; size_t w {0}; public: size_t empty() const; size_t size() const; size_t waiting() const; // Consumer interface; waits for item and std::move() it off the queue template T pop_until(time_point&&, const opts & = (opts)0); template T pop_for(const duration &, const opts & = (opts)0); T pop(const opts & = (opts)0); // Producer interface; emplace item on the queue and notify consumer template void emplace(const opts &, args&&...); template void emplace(args&&...); void push(const opts &, const T &); void push(const T &); void push(const opts &, T &&) noexcept; void push(T &&) noexcept; queue(); queue(A&& alloc); ~queue() noexcept; }; template inline ircd::ctx::queue::queue() :q(std::allocator()) { } template inline ircd::ctx::queue::queue(A&& alloc) :q(std::forward(alloc)) { } template inline ircd::ctx::queue::~queue() noexcept { assert(q.empty()); } template inline void ircd::ctx::queue::push(T&& t) noexcept { static const opts opts {0}; push(opts, std::forward(t)); } template inline void ircd::ctx::queue::push(const opts &opts, T&& t) noexcept { if(opts & opts::LIFO) q.push_front(std::forward(t)); else q.push_back(std::forward(t)); d.notify(); } template inline void ircd::ctx::queue::push(const T &t) { static const opts opts {0}; push(opts, t); } template inline void ircd::ctx::queue::push(const opts &opts, const T &t) { if(opts & opts::LIFO) q.push_front(t); else q.push_back(t); d.notify(); } template template inline void ircd::ctx::queue::emplace(args&&... a) { static const opts opts {0}; emplace(opts, std::forward(a)...); } template template inline void ircd::ctx::queue::emplace(const opts &opts, args&&... a) { if(opts & opts::LIFO) q.emplace_front(std::forward(a)...); else q.emplace_back(std::forward(a)...); d.notify(); } template inline T ircd::ctx::queue::pop(const opts &opts) { const scope_count w { this->w }; const auto predicate{[this]() noexcept { return !q.empty(); }}; d.wait(predicate, opts); assert(!q.empty()); auto ret(std::move(q.front())); q.pop_front(); return ret; } template template inline T ircd::ctx::queue::pop_for(const duration &dur, const opts &opts) { const scope_count w { this->w }; const auto predicate{[this]() noexcept { return !q.empty(); }}; const bool ready { d.wait_for(dur, predicate, opts) }; if(unlikely(!ready)) throw timeout{}; assert(!q.empty()); auto ret(std::move(q.front())); q.pop(); return ret; } template template inline T ircd::ctx::queue::pop_until(time_point&& tp, const opts &opts) { const scope_count w { this->w }; const auto predicate{[this]() noexcept { return !q.empty(); }}; const bool ready { d.wait_until(tp, predicate, opts) }; if(unlikely(!ready)) throw timeout{}; assert(!q.empty()); auto ret(std::move(q.front())); q.pop(); return ret; } template inline size_t ircd::ctx::queue::waiting() const { return w; } template inline size_t ircd::ctx::queue::size() const { return q.size(); } template inline size_t ircd::ctx::queue::empty() const { return q.empty(); }