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