// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2018 // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #pragma once #include "td/actor/impl2/ActorLocker.h" #include "td/actor/impl2/SchedulerId.h" #include "td/utils/Closure.h" #include "td/utils/common.h" #include "td/utils/format.h" #include "td/utils/Heap.h" #include "td/utils/List.h" #include "td/utils/logging.h" #include "td/utils/MpmcQueue.h" #include "td/utils/MpmcWaiter.h" #include "td/utils/MpscLinkQueue.h" #include "td/utils/MpscPollableQueue.h" #include "td/utils/port/Fd.h" #include "td/utils/port/Poll.h" #include "td/utils/port/thread.h" #include "td/utils/port/thread_local.h" #include "td/utils/ScopeGuard.h" #include "td/utils/SharedObjectPool.h" #include "td/utils/Slice.h" #include "td/utils/Time.h" #include "td/utils/type_traits.h" #include #include #include #include #include #include #include namespace td { namespace actor2 { class Actor; template class Context { public: static Impl *get() { return context_; } class Guard { public: explicit Guard(Impl *new_context) { old_context_ = context_; context_ = new_context; } ~Guard() { context_ = old_context_; } Guard(const Guard &) = delete; Guard &operator=(const Guard &) = delete; Guard(Guard &&) = delete; Guard &operator=(Guard &&) = delete; private: Impl *old_context_; }; private: static TD_THREAD_LOCAL Impl *context_; }; template TD_THREAD_LOCAL Impl *Context::context_; enum : uint64 { EmptyLinkToken = std::numeric_limits::max() }; class ActorExecuteContext : public Context { public: explicit ActorExecuteContext(Actor *actor, Timestamp alarm_timestamp = Timestamp::never()) : actor_(actor), alarm_timestamp_(alarm_timestamp) { } Actor &actor() const { CHECK(actor_); return *actor_; } bool has_flags() const { return flags_ != 0; } void set_stop() { flags_ |= 1 << Stop; } bool get_stop() const { return (flags_ & (1 << Stop)) != 0; } void set_pause() { flags_ |= 1 << Pause; } bool get_pause() const { return (flags_ & (1 << Pause)) != 0; } void clear_actor() { actor_ = nullptr; } void set_link_token(uint64 link_token) { link_token_ = link_token; } uint64 get_link_token() const { return link_token_; } Timestamp &alarm_timestamp() { flags_ |= 1 << Alarm; return alarm_timestamp_; } bool get_alarm_flag() const { return (flags_ & (1 << Alarm)) != 0; } Timestamp get_alarm_timestamp() const { return alarm_timestamp_; } private: Actor *actor_; uint32 flags_{0}; uint64 link_token_{EmptyLinkToken}; Timestamp alarm_timestamp_; enum { Stop, Pause, Alarm }; }; class ActorMessageImpl : private MpscLinkQueueImpl::Node { public: ActorMessageImpl() = default; ActorMessageImpl(const ActorMessageImpl &) = delete; ActorMessageImpl &operator=(const ActorMessageImpl &) = delete; ActorMessageImpl(ActorMessageImpl &&other) = delete; ActorMessageImpl &operator=(ActorMessageImpl &&other) = delete; virtual ~ActorMessageImpl() = default; virtual void run() = 0; //virtual void run_anonymous() = 0; // ActorMessage <--> MpscLintQueue::Node // Each actor's mailbox will be a queue static ActorMessageImpl *from_mpsc_link_queue_node(MpscLinkQueueImpl::Node *node) { return static_cast(node); } MpscLinkQueueImpl::Node *to_mpsc_link_queue_node() { return static_cast(this); } uint64 link_token_{EmptyLinkToken}; bool is_big_{false}; }; class ActorMessage { public: ActorMessage() = default; explicit ActorMessage(std::unique_ptr impl) : impl_(std::move(impl)) { } void run() { CHECK(impl_); impl_->run(); } explicit operator bool() { return bool(impl_); } friend class ActorMailbox; void set_link_token(uint64 link_token) { impl_->link_token_ = link_token; } uint64 get_link_token() const { return impl_->link_token_; } bool is_big() const { return impl_->is_big_; } void set_big() { impl_->is_big_ = true; } private: std::unique_ptr impl_; template friend class td::MpscLinkQueue; static ActorMessage from_mpsc_link_queue_node(MpscLinkQueueImpl::Node *node) { return ActorMessage(std::unique_ptr(ActorMessageImpl::from_mpsc_link_queue_node(node))); } MpscLinkQueueImpl::Node *to_mpsc_link_queue_node() { return impl_.release()->to_mpsc_link_queue_node(); } }; class ActorMailbox { public: ActorMailbox() = default; ActorMailbox(const ActorMailbox &) = delete; ActorMailbox &operator=(const ActorMailbox &) = delete; ActorMailbox(ActorMailbox &&other) = delete; ActorMailbox &operator=(ActorMailbox &&other) = delete; ~ActorMailbox() { pop_all(); while (reader_.read()) { // skip } } class Reader; void push(ActorMessage message) { queue_.push(std::move(message)); } void push_unsafe(ActorMessage message) { queue_.push_unsafe(std::move(message)); } auto &reader() { return reader_; } void pop_all() { queue_.pop_all(reader_); } void pop_all_unsafe() { queue_.pop_all_unsafe(reader_); } private: td::MpscLinkQueue queue_; td::MpscLinkQueue::Reader reader_; }; class ActorInfo : private HeapNode , private ListNode { public: ActorInfo(std::unique_ptr actor, ActorState::Flags state_flags, Slice name) : actor_(std::move(actor)), name_(name.begin(), name.size()) { state_.set_flags_unsafe(state_flags); } bool has_actor() const { return bool(actor_); } Actor &actor() { CHECK(has_actor()); return *actor_; } Actor *actor_ptr() const { return actor_.get(); } void destroy_actor() { actor_.reset(); } ActorState &state() { return state_; } ActorMailbox &mailbox() { return mailbox_; } CSlice get_name() const { return name_; } HeapNode *as_heap_node() { return this; } static ActorInfo *from_heap_node(HeapNode *node) { return static_cast(node); } Timestamp &alarm_timestamp() { return alarm_timestamp_; } private: std::unique_ptr actor_; ActorState state_; ActorMailbox mailbox_; std::string name_; Timestamp alarm_timestamp_; }; using ActorInfoPtr = SharedObjectPool::Ptr; class Actor { public: Actor() = default; Actor(const Actor &) = delete; Actor &operator=(const Actor &) = delete; Actor(Actor &&other) = delete; Actor &operator=(Actor &&other) = delete; virtual ~Actor() = default; void set_actor_info_ptr(ActorInfoPtr actor_info_ptr) { actor_info_ptr_ = std::move(actor_info_ptr); } ActorInfoPtr get_actor_info_ptr() { return actor_info_ptr_; } protected: // Signal handlers virtual void start_up(); // StartUp signal handler virtual void tear_down(); // TearDown signal handler (or Kill) virtual void hang_up(); // HangUp signal handler virtual void wake_up(); // WakeUp signal handler virtual void alarm(); // Alarm signal handler friend class ActorMessageHangup; // Event handlers //virtual void hangup_shared(); // TODO: raw event? virtual void loop(); // default handler // Useful functions void yield(); // send wakeup signal to itself void stop(); // send Kill signal to itself Timestamp &alarm_timestamp() { return ActorExecuteContext::get()->alarm_timestamp(); } Timestamp get_alarm_timestamp() { return ActorExecuteContext::get()->get_alarm_timestamp(); } CSlice get_name() { return actor_info_ptr_->get_name(); } // Inteface to scheduler // Query will be just passed to current scheduler // Timeout functions //bool has_timeout() const; //void set_timeout_in(double timeout_in); //void set_timeout_at(double timeout_at); //void cancel_timeout(); //uint64 get_link_token(); // get current request's link_token //set context that will be inherited by all childrens //void set_context(std::shared_ptr context); //ActorShared<> actor_shared(); // ActorShared to itself //template //ActorShared actor_shared(SelfT *self, uint64 id = static_cast(-1)); // ActorShared with type // Create EventFull to itself //template //auto self_closure(FuncT &&func, ArgsT &&... args); //template //auto self_closure(SelfT *self, FuncT &&func, ArgsT &&... args); //template //auto self_lambda(LambdaT &&lambda); //void do_stop(); // process Kill signal immediately private: friend class ActorExecutor; ActorInfoPtr actor_info_ptr_; }; // Signal handlers inline void Actor::start_up() { yield(); } inline void Actor::tear_down() { // noop } inline void Actor::hang_up() { stop(); } inline void Actor::wake_up() { loop(); } inline void Actor::alarm() { loop(); } inline void Actor::loop() { // noop } // Useful functions inline void Actor::yield() { // TODO } inline void Actor::stop() { ActorExecuteContext::get()->set_stop(); } class ActorInfoCreator { public: class Options { public: Options() = default; Options &with_name(Slice new_name) { name = new_name; return *this; } Options &on_scheduler(SchedulerId new_scheduler_id) { scheduler_id = new_scheduler_id; return *this; } bool has_scheduler() const { return scheduler_id.is_valid(); } Options &with_poll() { is_shared = false; return *this; } private: friend class ActorInfoCreator; Slice name; SchedulerId scheduler_id; bool is_shared{true}; bool in_queue{true}; //TODO: rename }; //Create unlocked actor. One must send StartUp signal immediately. ActorInfoPtr create(std::unique_ptr actor, const Options &args) { ActorState::Flags flags; flags.set_scheduler_id(args.scheduler_id); flags.set_shared(args.is_shared); flags.set_in_queue(args.in_queue); flags.set_signals(ActorSignals::one(ActorSignals::StartUp)); auto actor_info_ptr = pool_.alloc(std::move(actor), flags, args.name); actor_info_ptr->actor().set_actor_info_ptr(actor_info_ptr); return actor_info_ptr; } ActorInfoCreator() = default; ActorInfoCreator(const ActorInfoCreator &) = delete; ActorInfoCreator &operator=(const ActorInfoCreator &) = delete; ActorInfoCreator(ActorInfoCreator &&other) = delete; ActorInfoCreator &operator=(ActorInfoCreator &&other) = delete; ~ActorInfoCreator() { pool_.for_each([](auto &actor_info) { actor_info.destroy_actor(); }); } private: SharedObjectPool pool_; }; using ActorOptions = ActorInfoCreator::Options; class SchedulerDispatcher { public: virtual SchedulerId get_scheduler_id() const = 0; virtual void add_to_queue(ActorInfoPtr actor_info_ptr, SchedulerId scheduler_id, bool need_poll) = 0; virtual void set_alarm_timestamp(const ActorInfoPtr &actor_info_ptr, Timestamp timestamp) = 0; SchedulerDispatcher() = default; SchedulerDispatcher(const SchedulerDispatcher &) = delete; SchedulerDispatcher &operator=(const SchedulerDispatcher &) = delete; SchedulerDispatcher(SchedulerDispatcher &&other) = delete; SchedulerDispatcher &operator=(SchedulerDispatcher &&other) = delete; virtual ~SchedulerDispatcher() = default; }; class ActorExecutor { public: struct Options { Options &with_from_queue() { from_queue = true; return *this; } Options &with_has_poll(bool new_has_poll) { this->has_poll = new_has_poll; return *this; } bool from_queue{false}; bool has_poll{false}; }; ActorExecutor(ActorInfo &actor_info, SchedulerDispatcher &dispatcher, Options options) : actor_info_(actor_info), dispatcher_(dispatcher), options_(options) { //LOG(ERROR) << "START " << actor_info_.get_name() << " " << tag("from_queue", from_queue); start(); } ActorExecutor(const ActorExecutor &) = delete; ActorExecutor &operator=(const ActorExecutor &) = delete; ActorExecutor(ActorExecutor &&other) = delete; ActorExecutor &operator=(ActorExecutor &&other) = delete; ~ActorExecutor() { //LOG(ERROR) << "FINISH " << actor_info_.get_name() << " " << tag("own_lock", actor_locker_.own_lock()); finish(); } // our best guess if actor is closed or not bool can_send() { return !flags().is_closed(); } bool can_send_immediate() { return actor_locker_.own_lock() && !actor_execute_context_.has_flags() && actor_locker_.can_execute(); } template void send_immediate(F &&f, uint64 link_token) { CHECK(can_send_immediate()); if (!can_send()) { return; } actor_execute_context_.set_link_token(link_token); f(); } void send_immediate(ActorMessage message) { CHECK(can_send_immediate()); if (message.is_big()) { actor_info_.mailbox().reader().delay(std::move(message)); pending_signals_.add_signal(ActorSignals::Message); actor_execute_context_.set_pause(); return; } actor_execute_context_.set_link_token(message.get_link_token()); message.run(); } void send_immediate(ActorSignals signals) { CHECK(can_send_immediate()); while (flush_one_signal(signals) && can_send_immediate()) { } pending_signals_.add_signals(signals); } void send(ActorMessage message) { if (!can_send()) { return; } if (can_send_immediate()) { return send_immediate(std::move(message)); } actor_info_.mailbox().push(std::move(message)); pending_signals_.add_signal(ActorSignals::Message); } void send(ActorSignals signals) { if (!can_send()) { return; } pending_signals_.add_signals(signals); } private: ActorInfo &actor_info_; SchedulerDispatcher &dispatcher_; Options options_; ActorLocker actor_locker_{ &actor_info_.state(), ActorLocker::Options().with_can_execute_paused(options_.from_queue).with_is_shared(!options_.has_poll)}; ActorExecuteContext actor_execute_context_{actor_info_.actor_ptr(), actor_info_.alarm_timestamp()}; ActorExecuteContext::Guard guard{&actor_execute_context_}; ActorState::Flags flags_; ActorSignals pending_signals_; ActorState::Flags &flags() { return flags_; } void start() { if (!can_send()) { return; } ActorSignals signals; SCOPE_EXIT { pending_signals_.add_signals(signals); }; if (options_.from_queue) { signals.add_signal(ActorSignals::Pop); } actor_locker_.try_lock(); flags_ = actor_locker_.flags(); if (!actor_locker_.own_lock()) { return; } if (options_.from_queue) { flags().set_pause(false); } if (!actor_locker_.can_execute()) { CHECK(!options_.from_queue); return; } signals.add_signals(flags().get_signals()); actor_info_.mailbox().pop_all(); while (!actor_execute_context_.has_flags() && flush_one(signals)) { } } void finish() { if (!actor_locker_.own_lock()) { if (!pending_signals_.empty() && actor_locker_.add_signals(pending_signals_)) { flags_ = actor_locker_.flags(); } else { return; } } CHECK(actor_locker_.own_lock()); if (actor_execute_context_.has_flags()) { if (actor_execute_context_.get_stop()) { if (actor_info_.alarm_timestamp()) { dispatcher_.set_alarm_timestamp(actor_info_.actor().get_actor_info_ptr(), Timestamp::never()); } flags_.set_closed(true); actor_info_.actor().tear_down(); actor_info_.destroy_actor(); return; } if (actor_execute_context_.get_pause()) { flags_.set_pause(true); } if (actor_execute_context_.get_alarm_flag()) { auto old_timestamp = actor_info_.alarm_timestamp(); auto new_timestamp = actor_execute_context_.get_alarm_timestamp(); if (!(old_timestamp == new_timestamp)) { actor_info_.alarm_timestamp() = new_timestamp; dispatcher_.set_alarm_timestamp(actor_info_.actor().get_actor_info_ptr(), new_timestamp); } } } flags_.set_signals(pending_signals_); bool add_to_queue = false; while (true) { // Drop InQueue flag if has pop signal // Can't delay this signal auto signals = flags().get_signals(); if (signals.has_signal(ActorSignals::Pop)) { signals.clear_signal(ActorSignals::Pop); flags().set_signals(signals); flags().set_in_queue(false); } if (flags().has_signals() && !flags().is_in_queue()) { add_to_queue = true; flags().set_in_queue(true); } if (actor_locker_.try_unlock(flags())) { if (add_to_queue) { dispatcher_.add_to_queue(actor_info_.actor().get_actor_info_ptr(), flags().get_scheduler_id(), !flags().is_shared()); } break; } flags_ = actor_locker_.flags(); } } bool flush_one(ActorSignals &signals) { return flush_one_signal(signals) || flush_one_message(); } bool flush_one_signal(ActorSignals &signals) { auto signal = signals.first_signal(); if (!signal) { return false; } switch (signal) { case ActorSignals::Wakeup: actor_info_.actor().wake_up(); break; case ActorSignals::Alarm: if (actor_execute_context_.get_alarm_timestamp().is_in_past()) { actor_execute_context_.alarm_timestamp() = Timestamp::never(); actor_info_.actor().alarm(); } break; case ActorSignals::Kill: actor_execute_context_.set_stop(); break; case ActorSignals::StartUp: actor_info_.actor().start_up(); break; case ActorSignals::TearDown: actor_info_.actor().tear_down(); break; case ActorSignals::Pop: flags().set_in_queue(false); break; case ActorSignals::Message: break; case ActorSignals::Io: case ActorSignals::Cpu: LOG(FATAL) << "TODO"; default: UNREACHABLE(); } signals.clear_signal(signal); return true; } bool flush_one_message() { auto message = actor_info_.mailbox().reader().read(); if (!message) { return false; } if (message.is_big() && !options_.from_queue) { actor_info_.mailbox().reader().delay(std::move(message)); pending_signals_.add_signal(ActorSignals::Message); actor_execute_context_.set_pause(); return false; } actor_execute_context_.set_link_token(message.get_link_token()); message.run(); return true; } }; using SchedulerMessage = ActorInfoPtr; struct WorkerInfo { enum class Type { Io, Cpu } type{Type::Io}; WorkerInfo() = default; explicit WorkerInfo(Type type) : type(type) { } ActorInfoCreator actor_info_creator; }; struct SchedulerInfo { SchedulerId id; // will be read by all workers is any thread std::unique_ptr> cpu_queue; std::unique_ptr cpu_queue_waiter; // only scheduler itself may read from io_queue_ std::unique_ptr> io_queue; size_t cpu_threads_count{0}; std::unique_ptr io_worker; std::vector> cpu_workers; }; struct SchedulerGroupInfo { explicit SchedulerGroupInfo(size_t n) : schedulers(n) { } std::atomic is_stop_requested{false}; int active_scheduler_count{0}; std::mutex active_scheduler_count_mutex; std::condition_variable active_scheduler_count_condition_variable; std::vector schedulers; }; class SchedulerContext : public Context , public SchedulerDispatcher { public: // DispatcherInterface SchedulerDispatcher &dispatcher() { return *this; } // ActorCreator Interface virtual ActorInfoCreator &get_actor_info_creator() = 0; // Poll interface virtual bool has_poll() = 0; virtual Poll &get_poll() = 0; // Timeout interface virtual bool has_heap() = 0; virtual KHeap &get_heap() = 0; // Stop all schedulers virtual bool is_stop_requested() = 0; virtual void stop() = 0; }; #if !TD_THREAD_UNSUPPORTED class Scheduler { public: Scheduler(std::shared_ptr scheduler_group_info, SchedulerId id, size_t cpu_threads_count) : scheduler_group_info_(std::move(scheduler_group_info)), cpu_threads_(cpu_threads_count) { scheduler_group_info_->active_scheduler_count++; info_ = &scheduler_group_info_->schedulers.at(id.value()); info_->id = id; if (cpu_threads_count != 0) { info_->cpu_threads_count = cpu_threads_count; info_->cpu_queue = std::make_unique>(1024, max_thread_count()); info_->cpu_queue_waiter = std::make_unique(); } info_->io_queue = std::make_unique>(); info_->io_queue->init(); info_->cpu_workers.resize(cpu_threads_count); for (auto &worker : info_->cpu_workers) { worker = std::make_unique(WorkerInfo::Type::Cpu); } info_->io_worker = std::make_unique(WorkerInfo::Type::Io); poll_.init(); io_worker_ = std::make_unique(*info_->io_queue); } Scheduler(const Scheduler &) = delete; Scheduler &operator=(const Scheduler &) = delete; Scheduler(Scheduler &&other) = delete; Scheduler &operator=(Scheduler &&other) = delete; ~Scheduler() { // should stop stop(); do_stop(); } void start() { for (size_t i = 0; i < cpu_threads_.size(); i++) { cpu_threads_[i] = td::thread([this, i] { this->run_in_context_impl(*this->info_->cpu_workers[i], [this] { CpuWorker(*info_->cpu_queue, *info_->cpu_queue_waiter).run(); }); }); } this->run_in_context([this] { this->io_worker_->start_up(); }); } template void run_in_context(F &&f) { run_in_context_impl(*info_->io_worker, std::forward(f)); } bool run(double timeout) { bool res; run_in_context_impl(*info_->io_worker, [this, timeout, &res] { if (SchedulerContext::get()->is_stop_requested()) { res = false; } else { res = io_worker_->run_once(timeout); } if (!res) { io_worker_->tear_down(); } }); if (!res) { do_stop(); } return res; } // Just syntactic sugar void stop() { run_in_context([] { SchedulerContext::get()->stop(); }); } SchedulerId get_scheduler_id() const { return info_->id; } private: std::shared_ptr scheduler_group_info_; SchedulerInfo *info_; std::vector cpu_threads_; bool is_stopped_{false}; Poll poll_; KHeap heap_; class IoWorker; std::unique_ptr io_worker_; class SchedulerContextImpl : public SchedulerContext { public: SchedulerContextImpl(WorkerInfo *worker, SchedulerInfo *scheduler, SchedulerGroupInfo *scheduler_group, Poll *poll, KHeap *heap) : worker_(worker), scheduler_(scheduler), scheduler_group_(scheduler_group), poll_(poll), heap_(heap) { } SchedulerId get_scheduler_id() const override { return scheduler()->id; } void add_to_queue(ActorInfoPtr actor_info_ptr, SchedulerId scheduler_id, bool need_poll) override { if (!scheduler_id.is_valid()) { scheduler_id = scheduler()->id; } auto &info = scheduler_group()->schedulers.at(scheduler_id.value()); if (need_poll) { info.io_queue->writer_put(std::move(actor_info_ptr)); } else { info.cpu_queue->push(std::move(actor_info_ptr), get_thread_id()); info.cpu_queue_waiter->notify(); } } ActorInfoCreator &get_actor_info_creator() override { return worker()->actor_info_creator; } bool has_poll() override { return poll_ != nullptr; } Poll &get_poll() override { CHECK(has_poll()); return *poll_; } bool has_heap() override { return heap_ != nullptr; } KHeap &get_heap() override { CHECK(has_heap()); return *heap_; } void set_alarm_timestamp(const ActorInfoPtr &actor_info_ptr, Timestamp timestamp) override { // we are in PollWorker CHECK(has_heap()); auto &heap = get_heap(); auto *heap_node = actor_info_ptr->as_heap_node(); if (timestamp) { if (heap_node->in_heap()) { heap.fix(timestamp.at(), heap_node); } else { heap.insert(timestamp.at(), heap_node); } } else { if (heap_node->in_heap()) { heap.erase(heap_node); } } // TODO: do something in plain worker } bool is_stop_requested() override { return scheduler_group()->is_stop_requested; } void stop() override { bool expect_false = false; // Trying to set close_flag_ to true with CAS auto &group = *scheduler_group(); if (!group.is_stop_requested.compare_exchange_strong(expect_false, true)) { return; } // Notify all workers of all schedulers for (auto &scheduler_info : group.schedulers) { scheduler_info.io_queue->writer_put({}); for (size_t i = 0; i < scheduler_info.cpu_threads_count; i++) { scheduler_info.cpu_queue->push({}, get_thread_id()); scheduler_info.cpu_queue_waiter->notify(); } } } private: WorkerInfo *worker() const { return worker_; } SchedulerInfo *scheduler() const { return scheduler_; } SchedulerGroupInfo *scheduler_group() const { return scheduler_group_; } WorkerInfo *worker_; SchedulerInfo *scheduler_; SchedulerGroupInfo *scheduler_group_; Poll *poll_; KHeap *heap_; }; template void run_in_context_impl(WorkerInfo &worker_info, F &&f) { bool is_io_worker = worker_info.type == WorkerInfo::Type::Io; SchedulerContextImpl context(&worker_info, info_, scheduler_group_info_.get(), is_io_worker ? &poll_ : nullptr, is_io_worker ? &heap_ : nullptr); SchedulerContext::Guard guard(&context); f(); } class CpuWorker { public: CpuWorker(MpmcQueue &queue, MpmcWaiter &waiter) : queue_(queue), waiter_(waiter) { } void run() { auto thread_id = get_thread_id(); auto &dispatcher = SchedulerContext::get()->dispatcher(); int yields = 0; while (true) { SchedulerMessage message; if (queue_.try_pop(message, thread_id)) { if (!message) { return; } ActorExecutor executor(*message, dispatcher, ActorExecutor::Options().with_from_queue()); yields = waiter_.stop_wait(yields, thread_id); } else { yields = waiter_.wait(yields, thread_id); } } } private: MpmcQueue &queue_; MpmcWaiter &waiter_; }; class IoWorker { public: explicit IoWorker(MpscPollableQueue &queue) : queue_(queue) { } void start_up() { auto &poll = SchedulerContext::get()->get_poll(); poll.subscribe(queue_.reader_get_event_fd().get_fd(), Fd::Flag::Read); } void tear_down() { auto &poll = SchedulerContext::get()->get_poll(); poll.unsubscribe(queue_.reader_get_event_fd().get_fd()); } bool run_once(double timeout) { auto &dispatcher = SchedulerContext::get()->dispatcher(); auto &poll = SchedulerContext::get()->get_poll(); auto &heap = SchedulerContext::get()->get_heap(); auto now = Time::now(); // update Time::now_cached() while (!heap.empty() && heap.top_key() <= now) { auto *heap_node = heap.pop(); auto *actor_info = ActorInfo::from_heap_node(heap_node); ActorExecutor executor(*actor_info, dispatcher, ActorExecutor::Options().with_has_poll(true)); if (executor.can_send_immediate()) { executor.send_immediate(ActorSignals::one(ActorSignals::Alarm)); } else { executor.send(ActorSignals::one(ActorSignals::Alarm)); } } const int size = queue_.reader_wait_nonblock(); for (int i = 0; i < size; i++) { auto message = queue_.reader_get_unsafe(); if (!message) { return false; } ActorExecutor executor(*message, dispatcher, ActorExecutor::Options().with_from_queue().with_has_poll(true)); } queue_.reader_flush(); bool can_sleep = size == 0 && timeout != 0; int32 timeout_ms = 0; if (can_sleep) { auto wakeup_timestamp = Timestamp::in(timeout); if (!heap.empty()) { wakeup_timestamp.relax(Timestamp::at(heap.top_key())); } timeout_ms = static_cast(wakeup_timestamp.in() * 1000) + 1; if (timeout_ms < 0) { timeout_ms = 0; } //const int thirty_seconds = 30 * 1000; //if (timeout_ms > thirty_seconds) { //timeout_ms = thirty_seconds; //} } poll.run(timeout_ms); return true; } private: MpscPollableQueue &queue_; }; void do_stop() { if (is_stopped_) { return; } // wait other threads to finish for (auto &thread : cpu_threads_) { thread.join(); } // Can't do anything else, other schedulers may send queries to this one. // Must wait till every scheduler is stopped first.. is_stopped_ = true; io_worker_.reset(); poll_.clear(); std::unique_lock lock(scheduler_group_info_->active_scheduler_count_mutex); scheduler_group_info_->active_scheduler_count--; scheduler_group_info_->active_scheduler_count_condition_variable.notify_all(); } public: static void close_scheduler_group(SchedulerGroupInfo &group_info) { LOG(ERROR) << "close scheduler group"; // Cannot close scheduler group before somebody asked to stop them CHECK(group_info.is_stop_requested); { std::unique_lock lock(group_info.active_scheduler_count_mutex); group_info.active_scheduler_count_condition_variable.wait(lock, [&] { return group_info.active_scheduler_count == 0; }); } // Drain all queues // Just to destroy all elements should be ok. for (auto &scheduler_info : group_info.schedulers) { // Drain io queue auto &io_queue = *scheduler_info.io_queue; while (true) { int n = io_queue.reader_wait_nonblock(); if (n == 0) { break; } while (n-- > 0) { auto message = io_queue.reader_get_unsafe(); // message's destructor is called } } scheduler_info.io_queue.reset(); // Drain cpu queue auto &cpu_queue = *scheduler_info.cpu_queue; while (true) { SchedulerMessage message; if (!cpu_queue.try_pop(message, get_thread_id())) { break; } // message's destructor is called } scheduler_info.cpu_queue.reset(); // Do not destroy worker infos. run_in_context will crash if they are empty } } }; // Actor messages template class ActorMessageLambda : public ActorMessageImpl { public: template explicit ActorMessageLambda(FromLambdaT &&lambda) : lambda_(std::forward(lambda)) { } void run() override { lambda_(); } private: LambdaT lambda_; }; class ActorMessageHangup : public ActorMessageImpl { public: void run() override { ActorExecuteContext::get()->actor().hang_up(); } }; class ActorMessageCreator { public: template static auto lambda(F &&f) { return ActorMessage(std::make_unique>(std::forward(f))); } static auto hangup() { return ActorMessage(std::make_unique()); } // Use faster allocation? }; // SYNTAX SHUGAR namespace detail { struct ActorRef { ActorRef(ActorInfo &actor_info, uint64 link_token = EmptyLinkToken) : actor_info(actor_info), link_token(link_token) { } ActorInfo &actor_info; uint64 link_token; }; template T ¤t_actor() { return static_cast(ActorExecuteContext::get()->actor()); } void send_message(ActorInfo &actor_info, ActorMessage message) { ActorExecutor executor(actor_info, SchedulerContext::get()->dispatcher(), ActorExecutor::Options()); executor.send(std::move(message)); } void send_message(ActorRef actor_ref, ActorMessage message) { message.set_link_token(actor_ref.link_token); send_message(actor_ref.actor_info, std::move(message)); } void send_message_later(ActorInfo &actor_info, ActorMessage message) { ActorExecutor executor(actor_info, SchedulerContext::get()->dispatcher(), ActorExecutor::Options()); executor.send(std::move(message)); } void send_message_later(ActorRef actor_ref, ActorMessage message) { message.set_link_token(actor_ref.link_token); send_message_later(actor_ref.actor_info, std::move(message)); } template void send_immediate(ActorRef actor_ref, ExecuteF &&execute, ToMessageF &&to_message) { auto &scheduler_context = *SchedulerContext::get(); ActorExecutor executor(actor_ref.actor_info, scheduler_context.dispatcher(), ActorExecutor::Options().with_has_poll(scheduler_context.has_poll())); if (executor.can_send_immediate()) { return executor.send_immediate(execute, actor_ref.link_token); } auto message = to_message(); message.set_link_token(actor_ref.link_token); executor.send(std::move(message)); } template void send_lambda(ActorRef actor_ref, F &&lambda) { send_immediate(actor_ref, lambda, [&lambda]() mutable { return ActorMessageCreator::lambda(std::move(lambda)); }); } template void send_lambda_later(ActorRef actor_ref, F &&lambda) { send_message_later(actor_ref, ActorMessageCreator::lambda(std::move(lambda))); } template void send_closure_impl(ActorRef actor_ref, ClosureT &&closure) { using ActorType = typename ClosureT::ActorType; send_immediate(actor_ref, [&closure]() mutable { closure.run(¤t_actor()); }, [&closure]() mutable { return ActorMessageCreator::lambda([closure = to_delayed_closure(std::move(closure))]() mutable { closure.run(¤t_actor()); }); }); } template void send_closure(ActorRef actor_ref, ArgsT &&... args) { send_closure_impl(actor_ref, create_immediate_closure(std::forward(args)...)); } template void send_closure_later_impl(ActorRef actor_ref, ClosureT &&closure) { using ActorType = typename ClosureT::ActorType; send_message_later(actor_ref, ActorMessageCreator::lambda([closure = std::move(closure)]() mutable { closure.run(¤t_actor()); })); } template void send_closure_later(ActorRef actor_ref, ArgsT &&... args) { send_closure_later_impl(actor_ref, create_delayed_closure(std::forward(args)...)); } void register_actor_info_ptr(ActorInfoPtr actor_info_ptr) { auto state = actor_info_ptr->state().get_flags_unsafe(); SchedulerContext::get()->add_to_queue(std::move(actor_info_ptr), state.get_scheduler_id(), !state.is_shared()); } template ActorInfoPtr create_actor(ActorOptions &options, ArgsT &&... args) { auto *scheduler_context = SchedulerContext::get(); if (!options.has_scheduler()) { options.on_scheduler(scheduler_context->get_scheduler_id()); } auto res = scheduler_context->get_actor_info_creator().create(std::make_unique(std::forward(args)...), options); register_actor_info_ptr(res); return res; } } // namespace detail // Essentially ActorInfoWeakPtr with Type template class ActorId { public: using ActorT = ActorType; ActorId() = default; ActorId(const ActorId &) = default; ActorId &operator=(const ActorId &) = default; ActorId(ActorId &&other) = default; ActorId &operator=(ActorId &&other) = default; // allow only conversion from child to parent template ::value>> explicit operator ActorId() const { return ActorId(ptr_); } const ActorInfoPtr &actor_info_ptr() const { return ptr_; } ActorInfo &actor_info() const { CHECK(ptr_); return *ptr_; } bool empty() const { return !ptr_; } template static ActorId create(ActorOptions &options, ArgsT &&... args) { return ActorId(detail::create_actor(options, std::forward(args)...)); } detail::ActorRef as_actor_ref() const { CHECK(!empty()); return detail::ActorRef(*actor_info_ptr()); } private: ActorInfoPtr ptr_; explicit ActorId(ActorInfoPtr ptr) : ptr_(std::move(ptr)) { } template friend ActorId actor_id(SelfT *self); }; template class ActorOwn { public: using ActorT = ActorType; ActorOwn() = default; explicit ActorOwn(ActorId id) : id_(std::move(id)) { } template explicit ActorOwn(ActorId id) : id_(std::move(id)) { } template explicit ActorOwn(ActorOwn &&other) : id_(other.release()) { } template ActorOwn &operator=(ActorOwn &&other) { reset(other.release()); } ActorOwn(ActorOwn &&other) : id_(other.release()) { } ActorOwn &operator=(ActorOwn &&other) { reset(other.release()); } ActorOwn(const ActorOwn &) = delete; ActorOwn &operator=(const ActorOwn &) = delete; ~ActorOwn() { reset(); } bool empty() const { return id_.empty(); } bool is_alive() const { return id_.is_alive(); } ActorId get() const { return id_; } ActorId release() { return std::move(id_); } void reset(ActorId other = ActorId()) { static_assert(sizeof(ActorType) > 0, "Can't use ActorOwn with incomplete type"); hangup(); id_ = std::move(other); } const ActorId *operator->() const { return &id_; } detail::ActorRef as_actor_ref() const { CHECK(!empty()); return detail::ActorRef(*id_.actor_info_ptr(), 0); } private: ActorId id_; void hangup() const { if (empty()) { return; } detail::send_message(as_actor_ref(), ActorMessageCreator::hangup()); } }; template class ActorShared { public: using ActorT = ActorType; ActorShared() = default; template ActorShared(ActorId id, uint64 token) : id_(std::move(id)), token_(token) { CHECK(token_ != 0); } template ActorShared(ActorShared &&other) : id_(other.release()), token_(other.token()) { } template ActorShared(ActorOwn &&other) : id_(other.release()), token_(other.token()) { } template ActorShared &operator=(ActorShared &&other) { reset(other.release(), other.token()); } ActorShared(ActorShared &&other) : id_(other.release()), token_(other.token()) { } ActorShared &operator=(ActorShared &&other) { reset(other.release(), other.token()); } ActorShared(const ActorShared &) = delete; ActorShared &operator=(const ActorShared &) = delete; ~ActorShared() { reset(); } uint64 token() const { return token_; } bool empty() const { return id_.empty(); } bool is_alive() const { return id_.is_alive(); } ActorId get() const { return id_; } ActorId release(); void reset(ActorId other = ActorId(), uint64 link_token = EmptyLinkToken) { static_assert(sizeof(ActorType) > 0, "Can't use ActorShared with incomplete type"); hangup(); id_ = other; token_ = link_token; } const ActorId *operator->() const { return &id_; } detail::ActorRef as_actor_ref() const { CHECK(!empty()); return detail::ActorRef(*id_.actor_info_ptr(), token_); } private: ActorId id_; uint64 token_; void hangup() const { } }; // common interface template ActorId actor_id(SelfT *self) { CHECK(self); CHECK(static_cast(self) == &ActorExecuteContext::get()->actor()); return ActorId(ActorExecuteContext::get()->actor().get_actor_info_ptr()); } inline ActorId<> actor_id() { return actor_id(&ActorExecuteContext::get()->actor()); } template ActorOwn create_actor(ActorOptions options, ArgsT &&... args) { return ActorOwn(ActorId::create(options, std::forward(args)...)); } template ActorOwn create_actor(Slice name, ArgsT &&... args) { return ActorOwn(ActorId::create(ActorOptions().with_name(name), std::forward(args)...)); } template void send_closure(ActorIdT &&actor_id, FunctionT function, ArgsT &&... args) { using ActorT = typename std::decay_t::ActorT; using FunctionClassT = member_function_class_t; static_assert(std::is_base_of::value, "unsafe send_closure"); ActorIdT id = std::forward(actor_id); detail::send_closure(id.as_actor_ref(), function, std::forward(args)...); } template void send_closure_later(ActorIdT &&actor_id, FunctionT function, ArgsT &&... args) { using ActorT = typename std::decay_t::ActorT; using FunctionClassT = member_function_class_t; static_assert(std::is_base_of::value, "unsafe send_closure"); ActorIdT id = std::forward(actor_id); detail::send_closure_later(id.as_actor_ref(), function, std::forward(args)...); } template void send_lambda(ActorIdT &&actor_id, ArgsT &&... args) { ActorIdT id = std::forward(actor_id); detail::send_lambda(id.as_actor_ref(), std::forward(args)...); } #endif //!TD_THREAD_UNSUPPORTED } // namespace actor2 } // namespace td