| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268 |
- #include "server-task.h"
- #include "server-queue.h"
- #include "log.h"
- #include <chrono>
- #define QUE_INF(fmt, ...) LOG_INF("que %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define QUE_WRN(fmt, ...) LOG_WRN("que %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define QUE_ERR(fmt, ...) LOG_ERR("que %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define QUE_DBG(fmt, ...) LOG_DBG("que %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define RES_INF(fmt, ...) LOG_INF("res %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define RES_WRN(fmt, ...) LOG_WRN("res %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define RES_ERR(fmt, ...) LOG_ERR("res %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- #define RES_DBG(fmt, ...) LOG_DBG("res %12.*s: " fmt, 12, __func__, __VA_ARGS__)
- //
- // server_queue
- //
- int server_queue::post(server_task && task, bool front) {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- GGML_ASSERT(task.id != -1);
- // if this is cancel task make sure to clean up pending tasks
- if (task.type == SERVER_TASK_TYPE_CANCEL) {
- cleanup_pending_task(task.id_target);
- }
- const int task_id = task.id;
- QUE_DBG("new task, id = %d, front = %d\n", task_id, front);
- if (front) {
- queue_tasks.push_front(std::move(task));
- } else {
- queue_tasks.push_back(std::move(task));
- }
- condition_tasks.notify_one();
- return task_id;
- }
- int server_queue::post(std::vector<server_task> && tasks, bool front) {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- for (auto & task : tasks) {
- if (task.id == -1) {
- task.id = id++;
- }
- // if this is cancel task make sure to clean up pending tasks
- if (task.type == SERVER_TASK_TYPE_CANCEL) {
- cleanup_pending_task(task.id_target);
- }
- QUE_DBG("new task, id = %d/%d, front = %d\n", task.id, (int) tasks.size(), front);
- if (front) {
- queue_tasks.push_front(std::move(task));
- } else {
- queue_tasks.push_back(std::move(task));
- }
- }
- condition_tasks.notify_one();
- return 0;
- }
- void server_queue::defer(server_task && task) {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- QUE_DBG("defer task, id = %d\n", task.id);
- queue_tasks_deferred.push_back(std::move(task));
- condition_tasks.notify_one();
- }
- int server_queue::get_new_id() {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- int new_id = id++;
- return new_id;
- }
- void server_queue::on_new_task(std::function<void(server_task &&)> callback) {
- callback_new_task = std::move(callback);
- }
- void server_queue::on_update_slots(std::function<void(void)> callback) {
- callback_update_slots = std::move(callback);
- }
- void server_queue::pop_deferred_task() {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- if (!queue_tasks_deferred.empty()) {
- queue_tasks.emplace_front(std::move(queue_tasks_deferred.front()));
- queue_tasks_deferred.pop_front();
- }
- condition_tasks.notify_one();
- }
- void server_queue::terminate() {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- running = false;
- condition_tasks.notify_all();
- }
- void server_queue::start_loop() {
- running = true;
- while (true) {
- QUE_DBG("%s", "processing new tasks\n");
- while (true) {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- if (!running) {
- QUE_DBG("%s", "terminate\n");
- return;
- }
- if (queue_tasks.empty()) {
- lock.unlock();
- break;
- }
- server_task task = std::move(queue_tasks.front());
- queue_tasks.pop_front();
- lock.unlock();
- QUE_DBG("processing task, id = %d\n", task.id);
- callback_new_task(std::move(task));
- }
- // all tasks in the current loop is processed, slots data is now ready
- QUE_DBG("%s", "update slots\n");
- callback_update_slots();
- QUE_DBG("%s", "waiting for new tasks\n");
- {
- std::unique_lock<std::mutex> lock(mutex_tasks);
- if (!running) {
- QUE_DBG("%s", "terminate\n");
- return;
- }
- if (queue_tasks.empty()) {
- condition_tasks.wait(lock, [&]{
- return (!queue_tasks.empty() || !running);
- });
- }
- }
- }
- }
- void server_queue::cleanup_pending_task(int id_target) {
- // no need lock because this is called exclusively by post()
- auto rm_func = [id_target](const server_task & task) {
- return task.id == id_target;
- };
- queue_tasks.erase(
- std::remove_if(queue_tasks.begin(), queue_tasks.end(), rm_func),
- queue_tasks.end());
- queue_tasks_deferred.erase(
- std::remove_if(queue_tasks_deferred.begin(), queue_tasks_deferred.end(), rm_func),
- queue_tasks_deferred.end());
- }
- //
- // server_response
- //
- void server_response::add_waiting_task_id(int id_task) {
- RES_DBG("add task %d to waiting list. current waiting = %d (before add)\n", id_task, (int) waiting_task_ids.size());
- std::unique_lock<std::mutex> lock(mutex_results);
- waiting_task_ids.insert(id_task);
- }
- void server_response::add_waiting_tasks(const std::vector<server_task> & tasks) {
- std::unique_lock<std::mutex> lock(mutex_results);
- for (const auto & task : tasks) {
- RES_DBG("add task %d to waiting list. current waiting = %d (before add)\n", task.id, (int) waiting_task_ids.size());
- waiting_task_ids.insert(task.id);
- }
- }
- void server_response::remove_waiting_task_id(int id_task) {
- RES_DBG("remove task %d from waiting list. current waiting = %d (before remove)\n", id_task, (int) waiting_task_ids.size());
- std::unique_lock<std::mutex> lock(mutex_results);
- waiting_task_ids.erase(id_task);
- // make sure to clean up all pending results
- queue_results.erase(
- std::remove_if(queue_results.begin(), queue_results.end(), [id_task](const server_task_result_ptr & res) {
- return res->id == id_task;
- }),
- queue_results.end());
- }
- void server_response::remove_waiting_task_ids(const std::unordered_set<int> & id_tasks) {
- std::unique_lock<std::mutex> lock(mutex_results);
- for (const auto & id_task : id_tasks) {
- RES_DBG("remove task %d from waiting list. current waiting = %d (before remove)\n", id_task, (int) waiting_task_ids.size());
- waiting_task_ids.erase(id_task);
- }
- }
- server_task_result_ptr server_response::recv(const std::unordered_set<int> & id_tasks) {
- while (true) {
- std::unique_lock<std::mutex> lock(mutex_results);
- condition_results.wait(lock, [&]{
- if (!running) {
- RES_DBG("%s : queue result stop\n", "recv");
- std::terminate(); // we cannot return here since the caller is HTTP code
- }
- return !queue_results.empty();
- });
- for (size_t i = 0; i < queue_results.size(); i++) {
- if (id_tasks.find(queue_results[i]->id) != id_tasks.end()) {
- server_task_result_ptr res = std::move(queue_results[i]);
- queue_results.erase(queue_results.begin() + i);
- return res;
- }
- }
- }
- // should never reach here
- }
- server_task_result_ptr server_response::recv_with_timeout(const std::unordered_set<int> & id_tasks, int timeout) {
- while (true) {
- std::unique_lock<std::mutex> lock(mutex_results);
- for (int i = 0; i < (int) queue_results.size(); i++) {
- if (id_tasks.find(queue_results[i]->id) != id_tasks.end()) {
- server_task_result_ptr res = std::move(queue_results[i]);
- queue_results.erase(queue_results.begin() + i);
- return res;
- }
- }
- std::cv_status cr_res = condition_results.wait_for(lock, std::chrono::seconds(timeout));
- if (!running) {
- RES_DBG("%s : queue result stop\n", __func__);
- std::terminate(); // we cannot return here since the caller is HTTP code
- }
- if (cr_res == std::cv_status::timeout) {
- return nullptr;
- }
- }
- // should never reach here
- }
- server_task_result_ptr server_response::recv(int id_task) {
- std::unordered_set<int> id_tasks = {id_task};
- return recv(id_tasks);
- }
- void server_response::send(server_task_result_ptr && result) {
- RES_DBG("sending result for task id = %d\n", result->id);
- std::unique_lock<std::mutex> lock(mutex_results);
- for (const auto & id_task : waiting_task_ids) {
- if (result->id == id_task) {
- RES_DBG("task id = %d pushed to result queue\n", result->id);
- queue_results.emplace_back(std::move(result));
- condition_results.notify_all();
- return;
- }
- }
- }
- void server_response::terminate() {
- running = false;
- condition_results.notify_all();
- }
|