14514f5e3Sopenharmony_ci/*
24514f5e3Sopenharmony_ci * Copyright (c) 2021 Huawei Device Co., Ltd.
34514f5e3Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
44514f5e3Sopenharmony_ci * you may not use this file except in compliance with the License.
54514f5e3Sopenharmony_ci * You may obtain a copy of the License at
64514f5e3Sopenharmony_ci *
74514f5e3Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
84514f5e3Sopenharmony_ci *
94514f5e3Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
104514f5e3Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
114514f5e3Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
124514f5e3Sopenharmony_ci * See the License for the specific language governing permissions and
134514f5e3Sopenharmony_ci * limitations under the License.
144514f5e3Sopenharmony_ci */
154514f5e3Sopenharmony_ci
164514f5e3Sopenharmony_ci#include "ecmascript/taskpool/taskpool.h"
174514f5e3Sopenharmony_ci
184514f5e3Sopenharmony_ci#include "ecmascript/platform/os.h"
194514f5e3Sopenharmony_ci
204514f5e3Sopenharmony_cinamespace panda::ecmascript {
214514f5e3Sopenharmony_ciTaskpool *Taskpool::GetCurrentTaskpool()
224514f5e3Sopenharmony_ci{
234514f5e3Sopenharmony_ci    static Taskpool *taskpool = new Taskpool();
244514f5e3Sopenharmony_ci    return taskpool;
254514f5e3Sopenharmony_ci}
264514f5e3Sopenharmony_ci
274514f5e3Sopenharmony_civoid Taskpool::Initialize(int threadNum,
284514f5e3Sopenharmony_ci    std::function<void(os::thread::native_handle_type)> prologueHook,
294514f5e3Sopenharmony_ci    const std::function<void(os::thread::native_handle_type)> epilogueHook)
304514f5e3Sopenharmony_ci{
314514f5e3Sopenharmony_ci    LockHolder lock(mutex_);
324514f5e3Sopenharmony_ci    if (isInitialized_++ <= 0) {
334514f5e3Sopenharmony_ci        runner_ = std::make_unique<Runner>(TheMostSuitableThreadNum(threadNum), prologueHook, epilogueHook);
344514f5e3Sopenharmony_ci    }
354514f5e3Sopenharmony_ci}
364514f5e3Sopenharmony_ci
374514f5e3Sopenharmony_civoid Taskpool::Destroy(int32_t id)
384514f5e3Sopenharmony_ci{
394514f5e3Sopenharmony_ci    ASSERT(id != 0);
404514f5e3Sopenharmony_ci    LockHolder lock(mutex_);
414514f5e3Sopenharmony_ci    if (isInitialized_ <= 0) {
424514f5e3Sopenharmony_ci        return;
434514f5e3Sopenharmony_ci    }
444514f5e3Sopenharmony_ci    isInitialized_--;
454514f5e3Sopenharmony_ci    if (isInitialized_ == 0) {
464514f5e3Sopenharmony_ci        runner_->TerminateThread();
474514f5e3Sopenharmony_ci    } else {
484514f5e3Sopenharmony_ci        runner_->TerminateTask(id, TaskType::ALL);
494514f5e3Sopenharmony_ci    }
504514f5e3Sopenharmony_ci}
514514f5e3Sopenharmony_ci
524514f5e3Sopenharmony_civoid Taskpool::TerminateTask(int32_t id, TaskType type)
534514f5e3Sopenharmony_ci{
544514f5e3Sopenharmony_ci    if (isInitialized_ <= 0) {
554514f5e3Sopenharmony_ci        return;
564514f5e3Sopenharmony_ci    }
574514f5e3Sopenharmony_ci    runner_->TerminateTask(id, type);
584514f5e3Sopenharmony_ci}
594514f5e3Sopenharmony_ci
604514f5e3Sopenharmony_ciuint32_t Taskpool::TheMostSuitableThreadNum(uint32_t threadNum) const
614514f5e3Sopenharmony_ci{
624514f5e3Sopenharmony_ci    if (threadNum > 0) {
634514f5e3Sopenharmony_ci        return std::min<uint32_t>(threadNum, MAX_TASKPOOL_THREAD_NUM);
644514f5e3Sopenharmony_ci    }
654514f5e3Sopenharmony_ci    uint32_t numOfThreads = std::min<uint32_t>(NumberOfCpuCore() / 2, MAX_TASKPOOL_THREAD_NUM);
664514f5e3Sopenharmony_ci    if (numOfThreads > MIN_TASKPOOL_THREAD_NUM) {
674514f5e3Sopenharmony_ci        return numOfThreads - 1;        // 1 for daemon thread.
684514f5e3Sopenharmony_ci    }
694514f5e3Sopenharmony_ci    return MIN_TASKPOOL_THREAD_NUM;     // At least MIN_TASKPOOL_THREAD_NUM GC threads, and 1 extra daemon thread.
704514f5e3Sopenharmony_ci}
714514f5e3Sopenharmony_ci
724514f5e3Sopenharmony_civoid Taskpool::ForEachTask(const std::function<void(Task*)> &f)
734514f5e3Sopenharmony_ci{
744514f5e3Sopenharmony_ci    if (isInitialized_ <= 0) {
754514f5e3Sopenharmony_ci        return;
764514f5e3Sopenharmony_ci    }
774514f5e3Sopenharmony_ci    runner_->ForEachTask(f);
784514f5e3Sopenharmony_ci}
794514f5e3Sopenharmony_ci}  // namespace panda::ecmascript
80