1e0e9324cSopenharmony_ci/*
2e0e9324cSopenharmony_ci * Copyright (c) 2023 Shenzhen Kaihong Digital Industry Development Co., Ltd.
3e0e9324cSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4e0e9324cSopenharmony_ci * you may not use this file except in compliance with the License.
5e0e9324cSopenharmony_ci * You may obtain a copy of the License at
6e0e9324cSopenharmony_ci *
7e0e9324cSopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8e0e9324cSopenharmony_ci *
9e0e9324cSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10e0e9324cSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11e0e9324cSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12e0e9324cSopenharmony_ci * See the License for the specific language governing permissions and
13e0e9324cSopenharmony_ci * limitations under the License.
14e0e9324cSopenharmony_ci */
15e0e9324cSopenharmony_ci
16e0e9324cSopenharmony_ci#include "taskpool.h"
17e0e9324cSopenharmony_cinamespace OHOS {
18e0e9324cSopenharmony_cinamespace Sharing {
19e0e9324cSopenharmony_ci
20e0e9324cSopenharmony_ciTaskPool::TaskPool()
21e0e9324cSopenharmony_ci{
22e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
23e0e9324cSopenharmony_ci}
24e0e9324cSopenharmony_ci
25e0e9324cSopenharmony_ciTaskPool::~TaskPool()
26e0e9324cSopenharmony_ci{
27e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
28e0e9324cSopenharmony_ci    if (isRunning_) {
29e0e9324cSopenharmony_ci        Stop();
30e0e9324cSopenharmony_ci    }
31e0e9324cSopenharmony_ci}
32e0e9324cSopenharmony_ci
33e0e9324cSopenharmony_ciint32_t TaskPool::Start(int32_t threadsNum)
34e0e9324cSopenharmony_ci{
35e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
36e0e9324cSopenharmony_ci    if (!threads_.empty()) {
37e0e9324cSopenharmony_ci        SHARING_LOGE("Before start, theads is not empty.");
38e0e9324cSopenharmony_ci        return -1;
39e0e9324cSopenharmony_ci    }
40e0e9324cSopenharmony_ci
41e0e9324cSopenharmony_ci    if (threadsNum <= 0) {
42e0e9324cSopenharmony_ci        SHARING_LOGE("threadNum is illegal, %{public}d.", threadsNum);
43e0e9324cSopenharmony_ci        return -1;
44e0e9324cSopenharmony_ci    }
45e0e9324cSopenharmony_ci
46e0e9324cSopenharmony_ci    isRunning_ = true;
47e0e9324cSopenharmony_ci    threads_.reserve(threadsNum);
48e0e9324cSopenharmony_ci    for (int32_t i = 0; i < threadsNum; ++i) {
49e0e9324cSopenharmony_ci        threads_.push_back(std::thread(&TaskPool::TaskMainWorker, this));
50e0e9324cSopenharmony_ci        std::string name = "taskpool" + std::to_string(i);
51e0e9324cSopenharmony_ci        pthread_setname_np(threads_.back().native_handle(), name.c_str());
52e0e9324cSopenharmony_ci    }
53e0e9324cSopenharmony_ci
54e0e9324cSopenharmony_ci    return 0;
55e0e9324cSopenharmony_ci}
56e0e9324cSopenharmony_ci
57e0e9324cSopenharmony_civoid TaskPool::Stop()
58e0e9324cSopenharmony_ci{
59e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
60e0e9324cSopenharmony_ci    {
61e0e9324cSopenharmony_ci        std::unique_lock<std::mutex> lock(taskMutex_);
62e0e9324cSopenharmony_ci        isRunning_ = false;
63e0e9324cSopenharmony_ci        hasTask_.notify_all();
64e0e9324cSopenharmony_ci    }
65e0e9324cSopenharmony_ci
66e0e9324cSopenharmony_ci    for (auto &e : threads_) {
67e0e9324cSopenharmony_ci        e.join();
68e0e9324cSopenharmony_ci    }
69e0e9324cSopenharmony_ci}
70e0e9324cSopenharmony_ci
71e0e9324cSopenharmony_civoid TaskPool::PushTask(std::packaged_task<BindedTask> &task)
72e0e9324cSopenharmony_ci{
73e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
74e0e9324cSopenharmony_ci    if (threads_.empty()) {
75e0e9324cSopenharmony_ci    } else {
76e0e9324cSopenharmony_ci        std::unique_lock<std::mutex> lock(taskMutex_);
77e0e9324cSopenharmony_ci        while (IsOverload()) {
78e0e9324cSopenharmony_ci            SHARING_LOGE("task pool is over load.");
79e0e9324cSopenharmony_ci            acceptNewTask_.wait(lock);
80e0e9324cSopenharmony_ci        }
81e0e9324cSopenharmony_ci        tasks_.emplace_back(std::move(task));
82e0e9324cSopenharmony_ci        hasTask_.notify_one();
83e0e9324cSopenharmony_ci    }
84e0e9324cSopenharmony_ci}
85e0e9324cSopenharmony_ci
86e0e9324cSopenharmony_cibool TaskPool::IsOverload() const
87e0e9324cSopenharmony_ci{
88e0e9324cSopenharmony_ci    return (maxTaskNum_ > 0) && (tasks_.size() >= maxTaskNum_);
89e0e9324cSopenharmony_ci}
90e0e9324cSopenharmony_ci
91e0e9324cSopenharmony_civoid TaskPool::TaskMainWorker()
92e0e9324cSopenharmony_ci{
93e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
94e0e9324cSopenharmony_ci    std::unique_lock<std::mutex> lock(taskMutex_);
95e0e9324cSopenharmony_ci    while (isRunning_) {
96e0e9324cSopenharmony_ci        if (tasks_.empty() && isRunning_) {
97e0e9324cSopenharmony_ci            hasTask_.wait(lock);
98e0e9324cSopenharmony_ci        } else {
99e0e9324cSopenharmony_ci            std::packaged_task<BindedTask> task = std::move(tasks_.front());
100e0e9324cSopenharmony_ci            tasks_.pop_front();
101e0e9324cSopenharmony_ci            acceptNewTask_.notify_one();
102e0e9324cSopenharmony_ci            lock.unlock();
103e0e9324cSopenharmony_ci            SHARING_LOGD("task main worker unlocked.");
104e0e9324cSopenharmony_ci            task();
105e0e9324cSopenharmony_ci            lock.lock();
106e0e9324cSopenharmony_ci            SHARING_LOGD("task main worker locked.");
107e0e9324cSopenharmony_ci        }
108e0e9324cSopenharmony_ci    }
109e0e9324cSopenharmony_ci}
110e0e9324cSopenharmony_ci
111e0e9324cSopenharmony_civoid TaskPool::SetTimeoutInterval(uint32_t ms)
112e0e9324cSopenharmony_ci{
113e0e9324cSopenharmony_ci    SHARING_LOGD("trace.");
114e0e9324cSopenharmony_ci    timeoutInterval_ = std::chrono::milliseconds(ms);
115e0e9324cSopenharmony_ci}
116e0e9324cSopenharmony_ci
117e0e9324cSopenharmony_ci} // namespace Sharing
118e0e9324cSopenharmony_ci} // namespace OHOS
119