blob: e8c6cded989c403427ede03980927cba6c981640 [file] [log] [blame]
Angel Pons118a9c72020-04-02 23:48:34 +02001/* SPDX-License-Identifier: GPL-2.0-only */
Elyes HAOUASa3759e32020-07-10 10:58:57 +02002
Aaron Durbin340ca912013-04-30 09:58:12 -05003#include <timer.h>
4
5#define MAX_TIMER_QUEUE_ENTRIES 64
6
7/* The timer queue is implemented using a min heap. Therefore the first
8 * element is the one with smallest time to expiration. */
9struct timer_queue {
10 int num_entries;
11 int max_entries;
12 struct timeout_callback *queue[MAX_TIMER_QUEUE_ENTRIES];
13};
14
15static struct timer_queue global_timer_queue = {
16 .num_entries = 0,
17 .max_entries = MAX_TIMER_QUEUE_ENTRIES,
18 .queue = { 0 },
19};
20
21static inline int timer_queue_empty(struct timer_queue *tq)
22{
23 return tq->num_entries == 0;
24}
25
26static inline int timer_queue_full(struct timer_queue *tq)
27{
28 return tq->num_entries == tq->max_entries;
29}
30
31static inline struct timeout_callback *timer_queue_head(struct timer_queue *tq)
32{
33 if (timer_queue_empty(tq))
34 return NULL;
35 return tq->queue[0];
36}
37
38static int timer_queue_insert(struct timer_queue *tq,
Lee Leahye20a3192017-03-09 16:21:34 -080039 struct timeout_callback *tocb)
Aaron Durbin340ca912013-04-30 09:58:12 -050040{
41 int index;
42
43 /* No more slots. */
44 if (timer_queue_full(tq))
45 return -1;
46
47 index = tq->num_entries;
48 tq->num_entries++;
49 tq->queue[index] = tocb;
50
51 while (index != 0) {
52 struct timeout_callback *parent;
53 int parent_index;
54
55 parent_index = (index - 1) / 2;
56 parent = tq->queue[parent_index];
57
58 /* All other ancestors are less than or equal to the current. */
59 if (mono_time_cmp(&parent->expiration, &tocb->expiration) <= 0)
60 break;
61
62 /* The parent is greater than current. Swap them. */
63 tq->queue[parent_index] = tocb;
64 tq->queue[index] = parent;
65
66 index = parent_index;
67 }
68
69 return 0;
70}
71
72/* Get the index containing the entry with smallest value. */
73static int timer_queue_min_child_index(struct timer_queue *tq, int index)
74{
75 int left_child_index;
76 int right_child_index;
77
78 left_child_index = 2 * index + 1;
79
80 if (left_child_index >= tq->num_entries)
81 return -1;
82
83 right_child_index = left_child_index + 1;
84
85 if (right_child_index >= tq->num_entries)
86 return left_child_index;
87
88 if (mono_time_cmp(&tq->queue[left_child_index]->expiration,
Lee Leahye20a3192017-03-09 16:21:34 -080089 &tq->queue[right_child_index]->expiration) < 0) {
Aaron Durbin340ca912013-04-30 09:58:12 -050090 return left_child_index;
91 }
92 return right_child_index;
93}
94
95static void timer_queue_remove_head(struct timer_queue *tq)
96{
97 int index;
98 struct timeout_callback *tocb;
99
100 /* In order to remove the head the deepest child is replaced in the
101 * head slot and bubbled down the tree. */
102 tq->num_entries--;
103 tocb = tq->queue[tq->num_entries];
104 tq->queue[0] = tocb;
105
106 index = 0;
107 while (1) {
108 int min_child_index;
109 struct timeout_callback *child;
110
111 min_child_index = timer_queue_min_child_index(tq, index);
112
113 /* No more entries to compare against. */
114 if (min_child_index < 0)
115 break;
116
117 child = tq->queue[min_child_index];
118
119 /* Current index is the correct place since it is smaller or
120 * equal to the smallest child. */
121 if (mono_time_cmp(&tocb->expiration, &child->expiration) <= 0)
122 break;
123
124 /* Need to swap with smallest child. */
125 tq->queue[min_child_index] = tocb;
126 tq->queue[index] = child;
127
128 index = min_child_index;
129 }
130}
131
132static struct timeout_callback *
133timer_queue_expired(struct timer_queue *tq, struct mono_time *current_time)
134{
135 struct timeout_callback *tocb;
136
137 tocb = timer_queue_head(tq);
138
139 if (tocb == NULL)
140 return NULL;
141
142 /* The timeout callback hasn't expired yet. */
143 if (mono_time_before(current_time, &tocb->expiration))
144 return NULL;
145
146 timer_queue_remove_head(tq);
147
148 return tocb;
149}
150
Fred Reitberger3c705fa2022-09-21 15:48:18 -0400151int timer_sched_callback(struct timeout_callback *tocb, uint64_t us)
Aaron Durbin340ca912013-04-30 09:58:12 -0500152{
153 struct mono_time current_time;
154
Lee Leahy35af5c42017-03-09 17:35:28 -0800155 if ((long)us < 0)
Aaron Durbin340ca912013-04-30 09:58:12 -0500156 return -1;
157
158 timer_monotonic_get(&current_time);
159 tocb->expiration = current_time;
160 mono_time_add_usecs(&tocb->expiration, us);
161
162 /* The expiration overflowed. */
163 if (us != 0 && !mono_time_before(&current_time, &tocb->expiration))
164 return -1;
165
166 return timer_queue_insert(&global_timer_queue, tocb);
167}
168
169int timers_run(void)
170{
171 struct timeout_callback *tocb;
172 struct mono_time current_time;
173
174 timer_monotonic_get(&current_time);
175 tocb = timer_queue_expired(&global_timer_queue, &current_time);
176
177 if (tocb != NULL)
178 tocb->callback(tocb);
179
180 return !timer_queue_empty(&global_timer_queue);
181}