xref: /aosp_15_r20/external/pigweed/pw_containers/examples/multiple_containers.cc (revision 61c4878ac05f98d0ceed94b57d316916de578985)
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