1c29fa5a6Sopenharmony_ci/*
2c29fa5a6Sopenharmony_ci * Copyright (c) 2023 Huawei Device Co., Ltd.
3c29fa5a6Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4c29fa5a6Sopenharmony_ci * you may not use this file except in compliance with the License.
5c29fa5a6Sopenharmony_ci * You may obtain a copy of the License at
6c29fa5a6Sopenharmony_ci *
7c29fa5a6Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8c29fa5a6Sopenharmony_ci *
9c29fa5a6Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10c29fa5a6Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11c29fa5a6Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12c29fa5a6Sopenharmony_ci * See the License for the specific language governing permissions and
13c29fa5a6Sopenharmony_ci * limitations under the License.
14c29fa5a6Sopenharmony_ci */
15c29fa5a6Sopenharmony_ci
16c29fa5a6Sopenharmony_ci#ifndef TASK_SCHEDULER_H
17c29fa5a6Sopenharmony_ci#define TASK_SCHEDULER_H
18c29fa5a6Sopenharmony_ci
19c29fa5a6Sopenharmony_ci#include <cinttypes>
20c29fa5a6Sopenharmony_ci#include <functional>
21c29fa5a6Sopenharmony_ci#include <future>
22c29fa5a6Sopenharmony_ci#include <memory>
23c29fa5a6Sopenharmony_ci#include <mutex>
24c29fa5a6Sopenharmony_ci#include <queue>
25c29fa5a6Sopenharmony_ci
26c29fa5a6Sopenharmony_ci#include "i_task_scheduler.h"
27c29fa5a6Sopenharmony_ci#include "id_factory.h"
28c29fa5a6Sopenharmony_ci#include "include/util.h"
29c29fa5a6Sopenharmony_ci
30c29fa5a6Sopenharmony_cinamespace OHOS {
31c29fa5a6Sopenharmony_cinamespace Msdp {
32c29fa5a6Sopenharmony_cinamespace DeviceStatus {
33c29fa5a6Sopenharmony_ciclass TaskScheduler final : public ITaskScheduler,
34c29fa5a6Sopenharmony_ci                            public IdFactory<int32_t> {
35c29fa5a6Sopenharmony_cipublic:
36c29fa5a6Sopenharmony_ci    struct TaskData {
37c29fa5a6Sopenharmony_ci        uint64_t tid { 0 };
38c29fa5a6Sopenharmony_ci        int32_t taskId { 0 };
39c29fa5a6Sopenharmony_ci    };
40c29fa5a6Sopenharmony_ci    class Task : public std::enable_shared_from_this<Task> {
41c29fa5a6Sopenharmony_ci    public:
42c29fa5a6Sopenharmony_ci        using Promise = std::promise<int32_t>;
43c29fa5a6Sopenharmony_ci        using Future = std::future<int32_t>;
44c29fa5a6Sopenharmony_ci        using TaskPtr = std::shared_ptr<TaskScheduler::Task>;
45c29fa5a6Sopenharmony_ci        Task(int32_t id, DTaskCallback fun, Promise *promise = nullptr)
46c29fa5a6Sopenharmony_ci            : id_(id), fun_(fun), promise_(promise) {}
47c29fa5a6Sopenharmony_ci        ~Task() = default;
48c29fa5a6Sopenharmony_ci
49c29fa5a6Sopenharmony_ci        TaskPtr GetSharedPtr()
50c29fa5a6Sopenharmony_ci        {
51c29fa5a6Sopenharmony_ci            return shared_from_this();
52c29fa5a6Sopenharmony_ci        }
53c29fa5a6Sopenharmony_ci        int32_t GetId() const
54c29fa5a6Sopenharmony_ci        {
55c29fa5a6Sopenharmony_ci            return id_;
56c29fa5a6Sopenharmony_ci        }
57c29fa5a6Sopenharmony_ci        void SetWaited()
58c29fa5a6Sopenharmony_ci        {
59c29fa5a6Sopenharmony_ci            hasWaited_ = true;
60c29fa5a6Sopenharmony_ci        }
61c29fa5a6Sopenharmony_ci        void ProcessTask();
62c29fa5a6Sopenharmony_ci
63c29fa5a6Sopenharmony_ci    private:
64c29fa5a6Sopenharmony_ci        int32_t id_ { 0 };
65c29fa5a6Sopenharmony_ci        std::atomic_bool hasWaited_ { false };
66c29fa5a6Sopenharmony_ci        DTaskCallback fun_ { nullptr };
67c29fa5a6Sopenharmony_ci        Promise* promise_ { nullptr };
68c29fa5a6Sopenharmony_ci    };
69c29fa5a6Sopenharmony_ci    using TaskPtr = Task::TaskPtr;
70c29fa5a6Sopenharmony_ci    using Promise = Task::Promise;
71c29fa5a6Sopenharmony_ci    using Future = Task::Future;
72c29fa5a6Sopenharmony_ci
73c29fa5a6Sopenharmony_cipublic:
74c29fa5a6Sopenharmony_ci    TaskScheduler() = default;
75c29fa5a6Sopenharmony_ci    ~TaskScheduler();
76c29fa5a6Sopenharmony_ci
77c29fa5a6Sopenharmony_ci    bool Init();
78c29fa5a6Sopenharmony_ci    void ProcessTasks();
79c29fa5a6Sopenharmony_ci    int32_t PostSyncTask(DTaskCallback cb) override;
80c29fa5a6Sopenharmony_ci    int32_t PostAsyncTask(DTaskCallback callback) override;
81c29fa5a6Sopenharmony_ci
82c29fa5a6Sopenharmony_ci    int32_t GetReadFd() const
83c29fa5a6Sopenharmony_ci    {
84c29fa5a6Sopenharmony_ci        return fds_[0];
85c29fa5a6Sopenharmony_ci    }
86c29fa5a6Sopenharmony_ci    void SetWorkerThreadId(uint64_t tid)
87c29fa5a6Sopenharmony_ci    {
88c29fa5a6Sopenharmony_ci        workerThreadId_ = tid;
89c29fa5a6Sopenharmony_ci    }
90c29fa5a6Sopenharmony_ci    bool IsCallFromWorkerThread() const
91c29fa5a6Sopenharmony_ci    {
92c29fa5a6Sopenharmony_ci        return (GetThisThreadId() == workerThreadId_);
93c29fa5a6Sopenharmony_ci    }
94c29fa5a6Sopenharmony_ci
95c29fa5a6Sopenharmony_ciprivate:
96c29fa5a6Sopenharmony_ci    void PopPendingTaskList(std::vector<TaskPtr> &tasks);
97c29fa5a6Sopenharmony_ci    TaskPtr PostTask(DTaskCallback callback, Promise *promise = nullptr);
98c29fa5a6Sopenharmony_ci
99c29fa5a6Sopenharmony_ciprivate:
100c29fa5a6Sopenharmony_ci    std::atomic<uint64_t> workerThreadId_ { 0 };
101c29fa5a6Sopenharmony_ci    int32_t fds_[2] {};
102c29fa5a6Sopenharmony_ci    std::mutex mux_;
103c29fa5a6Sopenharmony_ci    std::queue<TaskPtr> tasks_;
104c29fa5a6Sopenharmony_ci};
105c29fa5a6Sopenharmony_ci} // namespace DeviceStatus
106c29fa5a6Sopenharmony_ci} // namespace Msdp
107c29fa5a6Sopenharmony_ci} // namespace OHOS
108c29fa5a6Sopenharmony_ci#endif // TASK_SCHEDULER_H
109