2018-12-31 22:04:05 +03:00
|
|
|
//
|
2021-01-01 15:57:46 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2021
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// 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)
|
|
|
|
//
|
|
|
|
#include "td/actor/actor.h"
|
2020-11-23 03:24:36 +03:00
|
|
|
#include "td/actor/ConcurrentScheduler.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
2021-05-17 15:21:11 +03:00
|
|
|
#include "td/utils/SliceBuilder.h"
|
2021-09-19 00:47:05 +03:00
|
|
|
#include "td/utils/tests.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
using namespace td;
|
|
|
|
|
|
|
|
class PowerWorker final : public Actor {
|
|
|
|
public:
|
|
|
|
class Callback {
|
|
|
|
public:
|
|
|
|
Callback() = default;
|
|
|
|
Callback(const Callback &) = delete;
|
|
|
|
Callback &operator=(const Callback &) = delete;
|
|
|
|
Callback(Callback &&) = delete;
|
|
|
|
Callback &operator=(Callback &&) = delete;
|
|
|
|
virtual ~Callback() = default;
|
|
|
|
virtual void on_ready(int query, int res) = 0;
|
|
|
|
virtual void on_closed() = 0;
|
|
|
|
};
|
|
|
|
void set_callback(unique_ptr<Callback> callback) {
|
|
|
|
callback_ = std::move(callback);
|
|
|
|
}
|
|
|
|
void task(uint32 x, uint32 p) {
|
|
|
|
uint32 res = 1;
|
|
|
|
for (uint32 i = 0; i < p; i++) {
|
|
|
|
res *= x;
|
|
|
|
}
|
|
|
|
callback_->on_ready(x, res);
|
|
|
|
}
|
|
|
|
void close() {
|
|
|
|
callback_->on_closed();
|
|
|
|
stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2018-09-27 04:19:03 +03:00
|
|
|
unique_ptr<Callback> callback_;
|
2018-12-31 22:04:05 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
class Manager final : public Actor {
|
|
|
|
public:
|
|
|
|
Manager(int queries_n, int query_size, std::vector<ActorId<PowerWorker>> workers)
|
2018-10-26 17:11:20 +03:00
|
|
|
: workers_(std::move(workers))
|
|
|
|
, ref_cnt_(static_cast<int>(workers_.size()))
|
|
|
|
, left_query_(queries_n)
|
|
|
|
, query_size_(query_size) {
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
2021-07-04 05:58:54 +03:00
|
|
|
class Callback final : public PowerWorker::Callback {
|
2018-12-31 22:04:05 +03:00
|
|
|
public:
|
|
|
|
Callback(ActorId<Manager> actor_id, int worker_id) : actor_id_(actor_id), worker_id_(worker_id) {
|
|
|
|
}
|
2021-07-03 23:51:36 +03:00
|
|
|
void on_ready(int query, int result) final {
|
2018-12-31 22:04:05 +03:00
|
|
|
send_closure(actor_id_, &Manager::on_ready, worker_id_, query, result);
|
|
|
|
}
|
2021-07-03 23:51:36 +03:00
|
|
|
void on_closed() final {
|
2018-12-31 22:04:05 +03:00
|
|
|
send_closure_later(actor_id_, &Manager::on_closed, worker_id_);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
ActorId<Manager> actor_id_;
|
|
|
|
int worker_id_;
|
|
|
|
};
|
|
|
|
|
2021-07-03 23:51:36 +03:00
|
|
|
void start_up() final {
|
2018-12-31 22:04:05 +03:00
|
|
|
int i = 0;
|
|
|
|
for (auto &worker : workers_) {
|
|
|
|
ref_cnt_++;
|
|
|
|
send_closure_later(worker, &PowerWorker::set_callback, make_unique<Callback>(actor_id(this), i));
|
|
|
|
i++;
|
|
|
|
send_closure_later(worker, &PowerWorker::task, 3, query_size_);
|
|
|
|
left_query_--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void on_ready(int worker_id, int query, int res) {
|
|
|
|
ref_cnt_--;
|
|
|
|
if (left_query_ == 0) {
|
|
|
|
send_closure(workers_[worker_id], &PowerWorker::close);
|
|
|
|
} else {
|
|
|
|
ref_cnt_++;
|
|
|
|
send_closure(workers_[worker_id], &PowerWorker::task, 3, query_size_);
|
|
|
|
left_query_--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void on_closed(int worker_id) {
|
|
|
|
ref_cnt_--;
|
|
|
|
if (ref_cnt_ == 0) {
|
|
|
|
Scheduler::instance()->finish();
|
|
|
|
stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<ActorId<PowerWorker>> workers_;
|
|
|
|
int ref_cnt_;
|
2018-10-26 17:11:20 +03:00
|
|
|
int left_query_;
|
2018-12-31 22:04:05 +03:00
|
|
|
int query_size_;
|
|
|
|
};
|
|
|
|
|
2020-11-21 20:45:09 +03:00
|
|
|
static void test_workers(int threads_n, int workers_n, int queries_n, int query_size) {
|
2018-12-31 22:04:05 +03:00
|
|
|
ConcurrentScheduler sched;
|
|
|
|
sched.init(threads_n);
|
|
|
|
|
|
|
|
std::vector<ActorId<PowerWorker>> workers;
|
|
|
|
for (int i = 0; i < workers_n; i++) {
|
|
|
|
int thread_id = threads_n ? i % (threads_n - 1) + 2 : 0;
|
2018-07-08 02:47:46 +03:00
|
|
|
workers.push_back(sched.create_actor_unsafe<PowerWorker>(thread_id, PSLICE() << "worker" << i).release());
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
2018-10-26 17:11:20 +03:00
|
|
|
sched.create_actor_unsafe<Manager>(threads_n ? 1 : 0, "Manager", queries_n, query_size, std::move(workers)).release();
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
sched.start();
|
|
|
|
while (sched.run_main(10)) {
|
|
|
|
// empty
|
|
|
|
}
|
|
|
|
sched.finish();
|
|
|
|
|
|
|
|
// sched.test_one_thread_run();
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST(Actors, workers_big_query_one_thread) {
|
|
|
|
test_workers(0, 10, 1000, 300000);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Actors, workers_big_query_two_threads) {
|
|
|
|
test_workers(2, 10, 1000, 300000);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Actors, workers_big_query_nine_threads) {
|
|
|
|
test_workers(9, 10, 1000, 300000);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Actors, workers_small_query_one_thread) {
|
|
|
|
test_workers(0, 10, 1000000, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Actors, workers_small_query_two_threads) {
|
|
|
|
test_workers(2, 10, 1000000, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Actors, workers_small_query_nine_threads) {
|
|
|
|
test_workers(9, 10, 1000000, 1);
|
|
|
|
}
|