1
6
7
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
58
59
60
61
62
63
64
65
66
69
70
73
74
77
78
79
94
95
98
99
100
101
102
103
104
105
106
107
108
109
110
111
120
121
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
181
182
183
190
191
192
193
194
195
196
197
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
231
232
233
234
235
236
237
238
239
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
285
286
287
288
289
290
291
292
297
298
299
300
301
302
303
304
305
312
313
314
315
316
317
318
319
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
362
363
364
365
366
367
368
371
372
375
376
377
383
384
385
386
390
391
392
393
394
395
396
397
406
407
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
434
435
436
437
438
439
440
441
442
443
447
448
449
450
451
452
459
460
461
462
463
464
465
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
489
490
493
494
497
498
499
500
501
502
503
504
523
524
525
526
531
532
533
/* ... */
#include <stdlib.h>
#include "pico.h"
#include "pico/time.h"
#include "pico/sync.h"
#include "pico/runtime_init.h"
5 includes
const absolute_time_t ABSOLUTE_TIME_INITIALIZED_VAR(nil_time, 0);
const absolute_time_t ABSOLUTE_TIME_INITIALIZED_VAR(at_the_end_of_time, INT64_MAX);
typedef struct alarm_pool_entry {
int16_t next;
volatile uint16_t sequence;
int64_t target;
alarm_callback_t callback;
void *user_data;
...} alarm_pool_entry_t;
struct alarm_pool {
uint8_t timer_alarm_num;
uint8_t core_num;
int16_t free_head;
volatile int16_t new_head;
volatile bool has_pending_cancellations;
int16_t ordered_head;
uint16_t num_entries;
alarm_pool_timer_t *timer;
spin_lock_t *lock;
alarm_pool_entry_t *entries;
...};
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
static alarm_pool_entry_t default_alarm_pool_entries[PICO_TIME_DEFAULT_ALARM_POOL_MAX_TIMERS];
static alarm_pool_t default_alarm_pool = {
.entries = default_alarm_pool_entries,
...};
static inline bool default_alarm_pool_initialized(void) {
return default_alarm_pool.lock != NULL;
}{ ... }
static lock_core_t sleep_notifier;/* ... */
#endif
#include "pico/time_adapter.h"
static alarm_pool_t *pools[TA_NUM_TIMERS][TA_NUM_TIMER_ALARMS];
static void alarm_pool_post_alloc_init(alarm_pool_t *pool, alarm_pool_timer_t *timer, uint hardware_alarm_num, uint max_timers);
static inline int16_t alarm_index(alarm_id_t id) {
return (int16_t)(id >> 16);
}{ ... }
static inline uint16_t alarm_sequence(alarm_id_t id) {
return (uint16_t)id;
}{ ... }
static alarm_id_t make_alarm_id(int index, uint16_t counter) {
return index << 16 | counter;
}{ ... }
#if !PICO_RUNTIME_NO_INIT_DEFAULT_ALARM_POOL
void __weak runtime_init_default_alarm_pool(void) {
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
if (!default_alarm_pool_initialized()) {
alarm_pool_timer_t *timer = alarm_pool_get_default_timer();
ta_hardware_alarm_claim(timer, PICO_TIME_DEFAULT_ALARM_POOL_HARDWARE_ALARM_NUM);
alarm_pool_post_alloc_init(&default_alarm_pool,
timer,
PICO_TIME_DEFAULT_ALARM_POOL_HARDWARE_ALARM_NUM,
PICO_TIME_DEFAULT_ALARM_POOL_MAX_TIMERS);
}if (!default_alarm_pool_initialized()) { ... }
lock_init(&sleep_notifier, PICO_SPINLOCK_ID_TIMER);/* ... */
#endif
}{ ... }
/* ... */#endif
void alarm_pool_init_default(void) {
runtime_init_default_alarm_pool();
}{ ... }
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
alarm_pool_t *alarm_pool_get_default(void) {
assert(default_alarm_pool_initialized());
return &default_alarm_pool;
}{ ... }
#if defined(PICO_RUNTIME_INIT_DEFAULT_ALARM_POOL) && !PICO_RUNTIME_SKIP_INIT_DEFAULT_ALARM_POOL
PICO_RUNTIME_INIT_FUNC_RUNTIME(runtime_init_default_alarm_pool, PICO_RUNTIME_INIT_DEFAULT_ALARM_POOL);
#endif/* ... */
#endif
alarm_pool_t *alarm_pool_create_on_timer(alarm_pool_timer_t *timer, uint hardware_alarm_num, uint max_timers) {
alarm_pool_t *pool = (alarm_pool_t *) malloc(sizeof(alarm_pool_t));
if (pool) {
pool->entries = (alarm_pool_entry_t *) calloc(max_timers, sizeof(alarm_pool_entry_t));
ta_hardware_alarm_claim(timer, hardware_alarm_num);
alarm_pool_post_alloc_init(pool, timer, hardware_alarm_num, max_timers);
}if (pool) { ... }
return pool;
}{ ... }
alarm_pool_t *alarm_pool_create_on_timer_with_unused_hardware_alarm(alarm_pool_timer_t *timer, uint max_timers) {
alarm_pool_t *pool = (alarm_pool_t *) malloc(sizeof(alarm_pool_t));
if (pool) {
pool->entries = (alarm_pool_entry_t *) calloc(max_timers, sizeof(alarm_pool_entry_t));
alarm_pool_post_alloc_init(pool, timer, (uint) ta_hardware_alarm_claim_unused(timer, true), max_timers);
}if (pool) { ... }
return pool;
}{ ... }
static void alarm_pool_irq_handler(void);
#define repeating_timer_marker ((alarm_callback_t)alarm_pool_irq_handler)
#include "hardware/gpio.h"
static void alarm_pool_irq_handler(void) {
uint timer_alarm_num;
alarm_pool_timer_t *timer = ta_from_current_irq(&timer_alarm_num);
uint timer_num = ta_timer_num(timer);
alarm_pool_t *pool = pools[timer_num][timer_alarm_num];
assert(pool->timer_alarm_num == timer_alarm_num);
int64_t now = (int64_t) ta_time_us_64(timer);
int64_t earliest_target;
ta_clear_force_irq(timer, timer_alarm_num);
do {
ta_clear_irq(timer, timer_alarm_num);
int16_t earliest_index = pool->ordered_head;
if (earliest_index >= 0) {
alarm_pool_entry_t *earliest_entry = &pool->entries[earliest_index];
earliest_target = earliest_entry->target;
if ((now - earliest_target) >= 0) {
int64_t delta;
if (earliest_target >= 0) {
if (earliest_entry->callback == repeating_timer_marker) {
repeating_timer_t *rpt = (repeating_timer_t *)earliest_entry->user_data;
delta = rpt->callback(rpt) ? rpt->delay_us : 0;
}if (earliest_entry->callback == repeating_timer_marker) { ... } else {
alarm_id_t id = make_alarm_id(pool->ordered_head, earliest_entry->sequence);
delta = earliest_entry->callback(id, earliest_entry->user_data);
}else { ... }
}if (earliest_target >= 0) { ... } else {
delta = 0;
}else { ... }
if (delta) {
int64_t next_time;
if (delta < 0) {
next_time = earliest_target - delta;
}if (delta < 0) { ... } else {
next_time = (int64_t) ta_time_us_64(timer) + delta;
}else { ... }
earliest_entry->target = next_time;
if (earliest_entry->next >= 0 && next_time - pool->entries[earliest_entry->next].target >= 0) {
pool->ordered_head = earliest_entry->next;
int16_t *prev = &pool->ordered_head;
while (*prev >= 0 && (next_time - pool->entries[*prev].target) >= 0) {
prev = &pool->entries[*prev].next;
}while (*prev >= 0 && (next_time - pool->entries[*prev].target) >= 0) { ... }
earliest_entry->next = *prev;
*prev = earliest_index;
}if (earliest_entry->next >= 0 && next_time - pool->entries[earliest_entry->next].target >= 0) { ... }
}if (delta) { ... } else {
pool->ordered_head = earliest_entry->next;
uint32_t save = spin_lock_blocking(pool->lock);
earliest_entry->next = pool->free_head;
pool->free_head = earliest_index;
spin_unlock(pool->lock, save);
}else { ... }
}if ((now - earliest_target) >= 0) { ... }
}if (earliest_index >= 0) { ... }
if (pool->new_head >= 0) {
uint32_t save = spin_lock_blocking(pool->lock);
int16_t new_index = pool->new_head;
pool->new_head = -1;
spin_unlock(pool->lock, save);
while (new_index >= 0) {
alarm_pool_entry_t *new_entry = &pool->entries[new_index];
int64_t new_entry_time = new_entry->target;
int16_t *prev = &pool->ordered_head;
while (*prev >= 0 && (new_entry_time - pool->entries[*prev].target) >= 0) {
prev = &pool->entries[*prev].next;
}while (*prev >= 0 && (new_entry_time - pool->entries[*prev].target) >= 0) { ... }
int16_t next = *prev;
*prev = new_index;
new_index = new_entry->next;
new_entry->next = next;
}while (new_index >= 0) { ... }
}if (pool->new_head >= 0) { ... }
if (pool->has_pending_cancellations) {
pool->has_pending_cancellations = false;
__compiler_memory_barrier();
int16_t *prev = &pool->ordered_head;
for(int16_t index = pool->ordered_head; index != -1; ) {
alarm_pool_entry_t *entry = &pool->entries[index];
int16_t next = entry->next;
if ((int16_t)entry->sequence < 0) {
entry->target = -1;
if (index != pool->ordered_head) {
*prev = entry->next;
entry->next = pool->ordered_head;
pool->ordered_head = index;
}if (index != pool->ordered_head) { ... }
}if ((int16_t)entry->sequence < 0) { ... } else {
prev = &entry->next;
}else { ... }
index = next;
}for (int16_t index = pool->ordered_head; index != -1;) { ... }
}if (pool->has_pending_cancellations) { ... }
now = (int64_t) ta_time_us_64(timer);
earliest_index = pool->ordered_head;
if (earliest_index < 0) break;
alarm_pool_entry_t *earliest_entry = &pool->entries[earliest_index];
earliest_target = earliest_entry->target;
ta_set_timeout(timer, timer_alarm_num, earliest_target);
...} while ((earliest_target - now) <= 0);
}{ ... }
void alarm_pool_post_alloc_init(alarm_pool_t *pool, alarm_pool_timer_t *timer, uint hardware_alarm_num, uint max_timers) {
pool->timer = timer;
pool->lock = spin_lock_instance(next_striped_spin_lock_num());
pool->timer_alarm_num = (uint8_t) hardware_alarm_num;
invalid_params_if(PICO_TIME, max_timers > 65536);
pool->num_entries = (uint16_t)max_timers;
pool->core_num = (uint8_t) get_core_num();
pool->new_head = pool->ordered_head = -1;
pool->free_head = (int16_t)(max_timers - 1);
for(uint i=0;i<max_timers;i++) {
pool->entries[i].next = (int16_t)(i-1);
}for (uint i=0;i
pools[ta_timer_num(timer)][hardware_alarm_num] = pool;
ta_enable_irq_handler(timer, hardware_alarm_num, alarm_pool_irq_handler);
}{ ... }
void alarm_pool_destroy(alarm_pool_t *pool) {
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
if (pool == &default_alarm_pool) {
assert(false);
return;
}if (pool == &default_alarm_pool) { ... }
/* ... */#endif
ta_disable_irq_handler(pool->timer, pool->timer_alarm_num, alarm_pool_irq_handler);
assert(pools[ta_timer_num(pool->timer)][pool->timer_alarm_num] == pool);
pools[ta_timer_num(pool->timer)][pool->timer_alarm_num] = NULL;
free(pool->entries);
free(pool);
}{ ... }
alarm_id_t alarm_pool_add_alarm_at(alarm_pool_t *pool, absolute_time_t time, alarm_callback_t callback,
void *user_data, bool fire_if_past) {
if (!fire_if_past) {
absolute_time_t t = get_absolute_time();
if (absolute_time_diff_us(t, time) < 0) return 0;
}if (!fire_if_past) { ... }
return alarm_pool_add_alarm_at_force_in_context(pool, time, callback, user_data);
}{ ... }
alarm_id_t alarm_pool_add_alarm_at_force_in_context(alarm_pool_t *pool, absolute_time_t time, alarm_callback_t callback,
void *user_data) {
uint32_t save = spin_lock_blocking(pool->lock);
int16_t index = pool->free_head;
alarm_pool_entry_t *entry = &pool->entries[index];
if (index >= 0) {
pool->free_head = entry->next;
}if (index >= 0) { ... }
spin_unlock(pool->lock, save);
if (index < 0) return PICO_ERROR_GENERIC;
entry->callback = callback;
entry->user_data = user_data;
entry->target = (int64_t)to_us_since_boot(time);
uint16_t next_sequence = (entry->sequence + 1) & 0x7fff;
if (!next_sequence) next_sequence = 1;
entry->sequence = next_sequence;
alarm_id_t id = make_alarm_id(index, next_sequence);
save = spin_lock_blocking(pool->lock);
entry->next = pool->new_head;
pool->new_head = index;
spin_unlock(pool->lock, save);
ta_force_irq(pool->timer, pool->timer_alarm_num);
return id;
}{ ... }
bool alarm_pool_cancel_alarm(alarm_pool_t *pool, alarm_id_t alarm_id) {
int16_t index = alarm_index(alarm_id);
if (index >= pool->num_entries) return false;
uint16_t sequence = alarm_sequence(alarm_id);
bool canceled = false;
alarm_pool_entry_t *entry = &pool->entries[index];
uint32_t save = spin_lock_blocking(pool->lock);
uint current_sequence = entry->sequence;
if (sequence == current_sequence) {
entry->sequence = (uint16_t)(current_sequence | 0x8000);
__compiler_memory_barrier();
pool->has_pending_cancellations = true;
canceled = true;
}if (sequence == current_sequence) { ... }
spin_unlock(pool->lock, save);
if (canceled) ta_force_irq(pool->timer, pool->timer_alarm_num);
return canceled;
}{ ... }
uint alarm_pool_timer_alarm_num(alarm_pool_t *pool) {
return pool->timer_alarm_num;
}{ ... }
uint alarm_pool_core_num(alarm_pool_t *pool) {
return pool->core_num;
}{ ... }
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
static int64_t sleep_until_callback(__unused alarm_id_t id, __unused void *user_data) {
uint32_t save = spin_lock_blocking(sleep_notifier.spin_lock);
lock_internal_spin_unlock_with_notify(&sleep_notifier, save);
return 0;
}{ ... }
/* ... */#endif
void sleep_until(absolute_time_t t) {
#if PICO_ON_DEVICE && !defined(NDEBUG)
if (__get_current_exception()) {
panic("Attempted to sleep inside of an exception handler; use busy_wait if you must");
}if (__get_current_exception()) { ... }
/* ... */#endif
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
uint64_t t_us = to_us_since_boot(t);
uint64_t t_before_us = t_us - PICO_TIME_SLEEP_OVERHEAD_ADJUST_US;
if (t_before_us > t_us) t_before_us = 0;
absolute_time_t t_before;
update_us_since_boot(&t_before, t_before_us);
if (absolute_time_diff_us(get_absolute_time(), t_before) > 0) {
if (add_alarm_at(t_before, sleep_until_callback, NULL, false) >= 0) {
while (!time_reached(t_before)) {
uint32_t save = spin_lock_blocking(sleep_notifier.spin_lock);
lock_internal_spin_unlock_with_wait(&sleep_notifier, save);
}while (!time_reached(t_before)) { ... }
}if (add_alarm_at(t_before, sleep_until_callback, NULL, false) >= 0) { ... }
}if (absolute_time_diff_us(get_absolute_time(), t_before) > 0) { ... }
/* ... */#else
sync_internal_yield_until_before(t);/* ... */
#endif
busy_wait_until(t);
}{ ... }
void sleep_us(uint64_t us) {
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
sleep_until(make_timeout_time_us(us));
#else
if (us < PICO_TIME_SLEEP_OVERHEAD_ADJUST_US) {
busy_wait_us(us);
}if (us < PICO_TIME_SLEEP_OVERHEAD_ADJUST_US) { ... } else {
absolute_time_t t = make_timeout_time_us(us - PICO_TIME_SLEEP_OVERHEAD_ADJUST_US);
sync_internal_yield_until_before(t);
busy_wait_until(t);
}else { ... }
/* ... */#endif
}{ ... }
void sleep_ms(uint32_t ms) {
sleep_us(ms * 1000ull);
}{ ... }
bool best_effort_wfe_or_timeout(absolute_time_t timeout_timestamp) {
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
if (__get_current_exception()) {
tight_loop_contents();
return time_reached(timeout_timestamp);
}if (__get_current_exception()) { ... } else {
alarm_id_t id;
id = add_alarm_at(timeout_timestamp, sleep_until_callback, NULL, false);
if (id <= 0) {
tight_loop_contents();
return time_reached(timeout_timestamp);
}if (id <= 0) { ... } else {
__sev();
__wfe();
if (!time_reached(timeout_timestamp))
{
__wfe();
}if (!time_reached(timeout_timestamp)) { ... }
cancel_alarm(id);
return time_reached(timeout_timestamp);
}else { ... }
}else { ... }
/* ... */#else
tight_loop_contents();
return time_reached(timeout_timestamp);/* ... */
#endif
}{ ... }
bool alarm_pool_add_repeating_timer_us(alarm_pool_t *pool, int64_t delay_us, repeating_timer_callback_t callback, void *user_data, repeating_timer_t *out) {
if (!delay_us) delay_us = 1;
out->pool = pool;
out->callback = callback;
out->delay_us = delay_us;
out->user_data = user_data;
out->alarm_id = alarm_pool_add_alarm_at(pool, make_timeout_time_us((uint64_t)(delay_us >= 0 ? delay_us : -delay_us)),
repeating_timer_marker, out, true);
return out->alarm_id > 0;
}{ ... }
bool cancel_repeating_timer(repeating_timer_t *timer) {
bool rc = false;
if (timer->alarm_id) {
rc = alarm_pool_cancel_alarm(timer->pool, timer->alarm_id);
timer->alarm_id = 0;
}if (timer->alarm_id) { ... }
return rc;
}{ ... }
alarm_pool_timer_t *alarm_pool_timer_for_timer_num(uint timer_num) {
return ta_timer_instance(timer_num);
}{ ... }
alarm_pool_timer_t *alarm_pool_get_default_timer(void) {
return ta_default_timer_instance();
}{ ... }
int64_t alarm_pool_remaining_alarm_time_us(alarm_pool_t *pool, alarm_id_t alarm_id) {
int64_t rc = -1;
int16_t index = alarm_index(alarm_id);
if ((uint16_t)index < pool->num_entries) {
uint16_t sequence = alarm_sequence(alarm_id);
alarm_pool_entry_t *entry = &pool->entries[index];
if (entry->sequence == sequence) {
uint32_t save = spin_lock_blocking(pool->lock);
int16_t search_index = pool->ordered_head;
while (search_index >= 0) {
entry = &pool->entries[search_index];
if (index == search_index) {
if (entry->sequence == sequence) {
rc = entry->target - (int64_t) ta_time_us_64(pool->timer);
}if (entry->sequence == sequence) { ... }
break;
}if (index == search_index) { ... }
search_index = entry->next;
}while (search_index >= 0) { ... }
spin_unlock(pool->lock, save);
}if (entry->sequence == sequence) { ... }
}if ((uint16_t)index < pool->num_entries) { ... }
return rc;
}{ ... }
int32_t alarm_pool_remaining_alarm_time_ms(alarm_pool_t *pool, alarm_id_t alarm_id) {
int64_t rc = alarm_pool_remaining_alarm_time_us(pool, alarm_id);
if (rc >= 0) rc /= 1000;
return rc >= INT32_MAX ? INT32_MAX : (int32_t) rc;
}{ ... }
#if !PICO_TIME_DEFAULT_ALARM_POOL_DISABLED
int64_t remaining_alarm_time_us(alarm_id_t alarm_id) {
return alarm_pool_remaining_alarm_time_us(alarm_pool_get_default(), alarm_id);
}{ ... }
int32_t remaining_alarm_time_ms(alarm_id_t alarm_id) {
return alarm_pool_remaining_alarm_time_ms(alarm_pool_get_default(), alarm_id);
}{ ... }
#endif/* ... */