1*61c4878aSAndroid Build Coastguard Worker // Copyright 2024 The Pigweed Authors
2*61c4878aSAndroid Build Coastguard Worker //
3*61c4878aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License"); you may not
4*61c4878aSAndroid Build Coastguard Worker // use this file except in compliance with the License. You may obtain a copy of
5*61c4878aSAndroid Build Coastguard Worker // the License at
6*61c4878aSAndroid Build Coastguard Worker //
7*61c4878aSAndroid Build Coastguard Worker // https://www.apache.org/licenses/LICENSE-2.0
8*61c4878aSAndroid Build Coastguard Worker //
9*61c4878aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*61c4878aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11*61c4878aSAndroid Build Coastguard Worker // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12*61c4878aSAndroid Build Coastguard Worker // License for the specific language governing permissions and limitations under
13*61c4878aSAndroid Build Coastguard Worker // the License.
14*61c4878aSAndroid Build Coastguard Worker
15*61c4878aSAndroid Build Coastguard Worker #include <cstdint>
16*61c4878aSAndroid Build Coastguard Worker
17*61c4878aSAndroid Build Coastguard Worker #include "pw_containers/intrusive_list.h"
18*61c4878aSAndroid Build Coastguard Worker #include "pw_containers/intrusive_map.h"
19*61c4878aSAndroid Build Coastguard Worker #include "pw_result/result.h"
20*61c4878aSAndroid Build Coastguard Worker #include "pw_status/status.h"
21*61c4878aSAndroid Build Coastguard Worker #include "pw_unit_test/framework.h"
22*61c4878aSAndroid Build Coastguard Worker
23*61c4878aSAndroid Build Coastguard Worker // DOCSTAG: [pw_containers-multiple_containers]
24*61c4878aSAndroid Build Coastguard Worker
25*61c4878aSAndroid Build Coastguard Worker // The base type for lists can be trivially derived.
26*61c4878aSAndroid Build Coastguard Worker struct ListItem : public pw::containers::future::IntrusiveList<ListItem>::Item {
27*61c4878aSAndroid Build Coastguard Worker };
28*61c4878aSAndroid Build Coastguard Worker
29*61c4878aSAndroid Build Coastguard Worker // The base type for maps needs a constructor.
30*61c4878aSAndroid Build Coastguard Worker struct MapPair : public pw::IntrusiveMap<const uint32_t&, MapPair>::Pair {
MapPairMapPair31*61c4878aSAndroid Build Coastguard Worker constexpr MapPair(const uint32_t& id)
32*61c4878aSAndroid Build Coastguard Worker : pw::IntrusiveMap<const uint32_t&, MapPair>::Pair(id) {}
33*61c4878aSAndroid Build Coastguard Worker };
34*61c4878aSAndroid Build Coastguard Worker
35*61c4878aSAndroid Build Coastguard Worker struct Task : public ListItem, public MapPair {
36*61c4878aSAndroid Build Coastguard Worker uint32_t id = 0;
TaskTask37*61c4878aSAndroid Build Coastguard Worker constexpr explicit Task() : MapPair(id) {}
38*61c4878aSAndroid Build Coastguard Worker };
39*61c4878aSAndroid Build Coastguard Worker
40*61c4878aSAndroid Build Coastguard Worker namespace examples {
41*61c4878aSAndroid Build Coastguard Worker
42*61c4878aSAndroid Build Coastguard Worker class Scheduler {
43*61c4878aSAndroid Build Coastguard Worker public:
44*61c4878aSAndroid Build Coastguard Worker // Adds a task to the queue, and returns an opaque `id` that identifies it.
45*61c4878aSAndroid Build Coastguard Worker // Returns INVALID_ARGUMENT the task is already in the queue.
ScheduleTask(Task & task)46*61c4878aSAndroid Build Coastguard Worker pw::Result<uint32_t> ScheduleTask(Task& task) {
47*61c4878aSAndroid Build Coastguard Worker if (task.id != 0) {
48*61c4878aSAndroid Build Coastguard Worker return pw::Status::InvalidArgument();
49*61c4878aSAndroid Build Coastguard Worker }
50*61c4878aSAndroid Build Coastguard Worker task.id = ++num_ids_;
51*61c4878aSAndroid Build Coastguard Worker by_id_.insert(task);
52*61c4878aSAndroid Build Coastguard Worker queue_.push_back(task);
53*61c4878aSAndroid Build Coastguard Worker return task.id;
54*61c4878aSAndroid Build Coastguard Worker }
55*61c4878aSAndroid Build Coastguard Worker
56*61c4878aSAndroid Build Coastguard Worker // Removes a task associated with a given `id` from the queue.
57*61c4878aSAndroid Build Coastguard Worker // Returns NOT_FOUND if the task is not in the queue.
CancelTask(uint32_t id)58*61c4878aSAndroid Build Coastguard Worker pw::Status CancelTask(uint32_t id) {
59*61c4878aSAndroid Build Coastguard Worker auto iter = by_id_.find(id);
60*61c4878aSAndroid Build Coastguard Worker if (iter == by_id_.end()) {
61*61c4878aSAndroid Build Coastguard Worker return pw::Status::NotFound();
62*61c4878aSAndroid Build Coastguard Worker }
63*61c4878aSAndroid Build Coastguard Worker auto& task = static_cast<Task&>(*iter);
64*61c4878aSAndroid Build Coastguard Worker by_id_.erase(iter);
65*61c4878aSAndroid Build Coastguard Worker queue_.remove(task);
66*61c4878aSAndroid Build Coastguard Worker task.id = 0;
67*61c4878aSAndroid Build Coastguard Worker return pw::OkStatus();
68*61c4878aSAndroid Build Coastguard Worker }
69*61c4878aSAndroid Build Coastguard Worker
70*61c4878aSAndroid Build Coastguard Worker // Runs the next task, if any, and returns its `id`.
71*61c4878aSAndroid Build Coastguard Worker // Returns NOT_FOUND if the queue is empty.
RunTask()72*61c4878aSAndroid Build Coastguard Worker pw::Result<uint32_t> RunTask() {
73*61c4878aSAndroid Build Coastguard Worker if (queue_.empty()) {
74*61c4878aSAndroid Build Coastguard Worker return pw::Status::NotFound();
75*61c4878aSAndroid Build Coastguard Worker }
76*61c4878aSAndroid Build Coastguard Worker auto& task = static_cast<Task&>(queue_.front());
77*61c4878aSAndroid Build Coastguard Worker queue_.pop_front();
78*61c4878aSAndroid Build Coastguard Worker by_id_.erase(task.id);
79*61c4878aSAndroid Build Coastguard Worker return task.id;
80*61c4878aSAndroid Build Coastguard Worker }
81*61c4878aSAndroid Build Coastguard Worker
82*61c4878aSAndroid Build Coastguard Worker private:
83*61c4878aSAndroid Build Coastguard Worker // NOTE! The containers must be templated on their specific item types, not
84*61c4878aSAndroid Build Coastguard Worker // the composite `Task` type.
85*61c4878aSAndroid Build Coastguard Worker pw::containers::future::IntrusiveList<ListItem> queue_;
86*61c4878aSAndroid Build Coastguard Worker pw::IntrusiveMap<uint32_t, MapPair> by_id_;
87*61c4878aSAndroid Build Coastguard Worker uint32_t num_ids_ = 0;
88*61c4878aSAndroid Build Coastguard Worker };
89*61c4878aSAndroid Build Coastguard Worker
90*61c4878aSAndroid Build Coastguard Worker // DOCSTAG: [pw_containers-multiple_containers]
91*61c4878aSAndroid Build Coastguard Worker
92*61c4878aSAndroid Build Coastguard Worker } // namespace examples
93*61c4878aSAndroid Build Coastguard Worker
94*61c4878aSAndroid Build Coastguard Worker namespace {
95*61c4878aSAndroid Build Coastguard Worker
TEST(ListedAndMappedExampleTest,RunScheduler)96*61c4878aSAndroid Build Coastguard Worker TEST(ListedAndMappedExampleTest, RunScheduler) {
97*61c4878aSAndroid Build Coastguard Worker examples::Scheduler scheduler;
98*61c4878aSAndroid Build Coastguard Worker constexpr size_t kNumTasks = 10;
99*61c4878aSAndroid Build Coastguard Worker std::array<Task, kNumTasks> tasks;
100*61c4878aSAndroid Build Coastguard Worker std::array<uint32_t, kNumTasks> ids;
101*61c4878aSAndroid Build Coastguard Worker pw::Result<uint32_t> result;
102*61c4878aSAndroid Build Coastguard Worker
103*61c4878aSAndroid Build Coastguard Worker for (size_t i = 0; i < tasks.size(); ++i) {
104*61c4878aSAndroid Build Coastguard Worker result = scheduler.ScheduleTask(tasks[i]);
105*61c4878aSAndroid Build Coastguard Worker ASSERT_EQ(result.status(), pw::OkStatus());
106*61c4878aSAndroid Build Coastguard Worker ids[i] = *result;
107*61c4878aSAndroid Build Coastguard Worker }
108*61c4878aSAndroid Build Coastguard Worker result = scheduler.ScheduleTask(tasks[0]);
109*61c4878aSAndroid Build Coastguard Worker EXPECT_EQ(result.status(), pw::Status::InvalidArgument());
110*61c4878aSAndroid Build Coastguard Worker
111*61c4878aSAndroid Build Coastguard Worker EXPECT_EQ(scheduler.CancelTask(ids[3]), pw::OkStatus());
112*61c4878aSAndroid Build Coastguard Worker EXPECT_EQ(scheduler.CancelTask(ids[7]), pw::OkStatus());
113*61c4878aSAndroid Build Coastguard Worker EXPECT_EQ(scheduler.CancelTask(ids[7]), pw::Status::NotFound());
114*61c4878aSAndroid Build Coastguard Worker
115*61c4878aSAndroid Build Coastguard Worker for (size_t i = 0; i < tasks.size(); ++i) {
116*61c4878aSAndroid Build Coastguard Worker if (i % 4 == 3) {
117*61c4878aSAndroid Build Coastguard Worker continue;
118*61c4878aSAndroid Build Coastguard Worker }
119*61c4878aSAndroid Build Coastguard Worker result = scheduler.RunTask();
120*61c4878aSAndroid Build Coastguard Worker ASSERT_EQ(result.status(), pw::OkStatus());
121*61c4878aSAndroid Build Coastguard Worker EXPECT_EQ(*result, ids[i]);
122*61c4878aSAndroid Build Coastguard Worker }
123*61c4878aSAndroid Build Coastguard Worker result = scheduler.RunTask();
124*61c4878aSAndroid Build Coastguard Worker EXPECT_EQ(result.status(), pw::Status::NotFound());
125*61c4878aSAndroid Build Coastguard Worker }
126*61c4878aSAndroid Build Coastguard Worker
127*61c4878aSAndroid Build Coastguard Worker } // namespace
128