1
6
7
14
15
16
17
18
19
24
29
30
37
38
45
46
53
54
60
61
66
67
68
69
70
71
72
73
74
75
76
77
78
108
109
112
113
116
117
118
119
120
123
124
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
169
170
172
173
175
176
177
178
179
180
181
182
183
187
188
189
190
191
192
193
194
195
196
199
200
205
206
207
212
213
214
218
219
220
221
222
227
228
229
230
231
232
233
234
235
243
244
245
246
247
248
249
250
251
252
253
260
261
262
263
264
265
266
267
268
269
270
276
277
278
279
286
287
290
291
295
296
300
301
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
378
379
381
382
383
384
385
386
387
388
389
390
391
392
393
398
399
400
401
402
405
406
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
426
427
429
430
431
432
433
434
435
436
437
439
440
444
445
446
447
448
449
450
451
452
453
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
509
510
511
512
513
514
515
516
517
518
519
520
528
529
530
531
532
533
534
535
536
537
538
539
553
554
555
556
559
560
561
562
563
564
567
568
569
570
571
572
575
576
577
578
579
580
581
582
586
587
588
589
590
597
598
601
602
603
604
605
606
607
610
611
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
644
645
646
647
648
649
650
651
667
668
674
675
678
679
682
683
688
689
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
/* ... */
#include "hardware/irq.h"
#include "pico/runtime_init.h"
#include "hardware/claim.h"
#include "pico/mutex.h"
#include "pico/assert.h"
5 includes
#if defined(PICO_RUNTIME_INIT_PER_CORE_IRQ_PRIORITIES) && !PICO_RUNTIME_SKIP_INIT_PER_CORE_IRQ_PRIORITIES
PICO_RUNTIME_INIT_FUNC_PER_CORE(runtime_init_per_core_irq_priorities, PICO_RUNTIME_INIT_PER_CORE_IRQ_PRIORITIES);
#endif
#if PICO_VTABLE_PER_CORE
static uint8_t user_irq_claimed[NUM_CORES];
static inline uint8_t *user_irq_claimed_ptr(void) {
return &user_irq_claimed[get_core_num()];
}user_irq_claimed_ptr (void) { ... }
/* ... */#else
static uint8_t user_irq_claimed;
static inline uint8_t *user_irq_claimed_ptr(void) {
return &user_irq_claimed;
}{ ... }
#endif/* ... */
static inline irq_handler_t *get_vtable(void) {
#ifdef __riscv
return (irq_handler_t *) (riscv_read_csr(RVCSR_MTVEC_OFFSET) & ~0x3u);
#else
return (irq_handler_t *) scb_hw->vtor;
#endif
}{ ... }
static inline void *add_thumb_bit(void *addr) {
#ifdef __riscv
return addr;
#else
return (void *) (((uintptr_t) addr) | 0x1);
#endif
}{ ... }
static inline void *remove_thumb_bit(void *addr) {
#ifdef __riscv
return addr;
#else
return (void *) (((uintptr_t) addr) & (uint)~0x1);
#endif
}{ ... }
static void set_raw_irq_handler_and_unlock(uint num, irq_handler_t handler, uint32_t save) {
get_vtable()[VTABLE_FIRST_IRQ + num] = handler;
__dmb();
spin_unlock(spin_lock_instance(PICO_SPINLOCK_ID_IRQ), save);
}{ ... }
void irq_set_enabled(uint num, bool enabled) {
check_irq_param(num);
irq_set_mask_n_enabled(num / 32, 1u << (num % 32), enabled);
}{ ... }
bool irq_is_enabled(uint num) {
check_irq_param(num);
#if PICO_RP2040
return 0 != ((1u << num) & *((io_rw_32 *) (PPB_BASE + M0PLUS_NVIC_ISER_OFFSET)));
#elif defined(__riscv)
return 0 != (hazard3_irqarray_read(RVCSR_MEIEA_OFFSET, num / 16) & (1u << (num % 16)));
#else
return 0 != (nvic_hw->iser[num/32] & (1 << num % 32));
#endif
}{ ... }
static inline void irq_set_mask_n_enabled_internal(uint n, uint32_t mask, bool enabled) {
invalid_params_if(HARDWARE_IRQ, n * 32u >= ((NUM_IRQS + 31u) & ~31u));
#if defined(__riscv)
if (enabled) {
hazard3_irqarray_clear(RVCSR_MEIFA_OFFSET, 2 * n, mask & 0xffffu);
hazard3_irqarray_clear(RVCSR_MEIFA_OFFSET, 2 * n + 1, mask >> 16);
hazard3_irqarray_set(RVCSR_MEIEA_OFFSET, 2 * n, mask & 0xffffu);
hazard3_irqarray_set(RVCSR_MEIEA_OFFSET, 2 * n + 1, mask >> 16);
}if (enabled) { ... } else {
hazard3_irqarray_clear(RVCSR_MEIEA_OFFSET, 2 * n, mask & 0xffffu);
hazard3_irqarray_clear(RVCSR_MEIEA_OFFSET, 2 * n + 1, mask >> 16);
}else { ... }
/* ... */#elif PICO_RP2040
((void)n);
if (enabled) {
nvic_hw->icpr = mask;
nvic_hw->iser = mask;
}if (enabled) { ... } else {
nvic_hw->icer = mask;
}else { ... }
/* ... */#else
if (enabled) {
nvic_hw->icpr[n] = mask;
nvic_hw->iser[n] = mask;
}if (enabled) { ... } else {
nvic_hw->icer[n] = mask;
}else { ... }
/* ... */#endif
}{ ... }
void irq_set_mask_enabled(uint32_t mask, bool enabled) {
irq_set_mask_n_enabled_internal(0, mask, enabled);
}{ ... }
void irq_set_mask_n_enabled(uint n, uint32_t mask, bool enabled) {
irq_set_mask_n_enabled_internal(n, mask, enabled);
}{ ... }
void irq_set_pending(uint num) {
check_irq_param(num);
#ifdef __riscv
hazard3_irqarray_set(RVCSR_MEIFA_OFFSET, num / 16, 1u << (num % 16));/* ... */
#else
#if PICO_RP2040
*((io_rw_32 *) (PPB_BASE + M0PLUS_NVIC_ISPR_OFFSET)) = 1u << num;
#else
nvic_hw->ispr[num/32] = 1 << (num % 32);
#endif/* ... */
#endif
}{ ... }
#if !PICO_DISABLE_SHARED_IRQ_HANDLERS
static_assert(PICO_MAX_SHARED_IRQ_HANDLERS >= 1 && PICO_MAX_SHARED_IRQ_HANDLERS < 0x7f, "");
extern void irq_handler_chain_first_slot(void);
extern void irq_handler_chain_remove_tail(void);
extern struct irq_handler_chain_slot {
#ifndef __riscv
uint16_t inst1;
uint16_t inst2;/* ... */
#else
uint32_t inst1;
uint32_t inst2;/* ... */
#endif
uint16_t inst3;
union {
struct {
int8_t link;
uint8_t priority;
...};
uint16_t inst4;
...};
#ifndef __riscv
irq_handler_t handler;
#endif
...} irq_handler_chain_slots[PICO_MAX_SHARED_IRQ_HANDLERS];
static int8_t irq_handler_chain_free_slot_head;
static inline bool is_shared_irq_raw_handler(irq_handler_t raw_handler) {
return (uintptr_t)raw_handler - (uintptr_t)irq_handler_chain_slots < sizeof(irq_handler_chain_slots);
}{ ... }
bool irq_has_shared_handler(uint irq_num) {
check_irq_param(irq_num);
irq_handler_t handler = irq_get_vtable_handler(irq_num);
return handler && is_shared_irq_raw_handler(handler);
}{ ... }
/* ... */#else
#define is_shared_irq_raw_handler(h) false
bool irq_has_shared_handler(uint irq_num) {
return false;
}irq_has_shared_handler (uint irq_num) { ... }
/* ... */#endif
irq_handler_t irq_get_vtable_handler(uint num) {
check_irq_param(num);
return get_vtable()[VTABLE_FIRST_IRQ + num];
}{ ... }
void irq_set_exclusive_handler(uint num, irq_handler_t handler) {
check_irq_param(num);
#if !PICO_NO_RAM_VECTOR_TABLE
spin_lock_t *lock = spin_lock_instance(PICO_SPINLOCK_ID_IRQ);
uint32_t save = spin_lock_blocking(lock);
__unused irq_handler_t current = irq_get_vtable_handler(num);
hard_assert(current == __unhandled_user_irq || current == handler);
set_raw_irq_handler_and_unlock(num, handler, save);/* ... */
#else
panic_unsupported();
#endif
}{ ... }
irq_handler_t irq_get_exclusive_handler(uint num) {
check_irq_param(num);
#if !PICO_NO_RAM_VECTOR_TABLE
spin_lock_t *lock = spin_lock_instance(PICO_SPINLOCK_ID_IRQ);
uint32_t save = spin_lock_blocking(lock);
irq_handler_t current = irq_get_vtable_handler(num);
spin_unlock(lock, save);
if (current == __unhandled_user_irq || is_shared_irq_raw_handler(current)) {
return NULL;
}if (current == __unhandled_user_irq || is_shared_irq_raw_handler(current)) { ... }
return current;/* ... */
#else
panic_unsupported();
#endif
}{ ... }
#if !PICO_DISABLE_SHARED_IRQ_HANDLERS
#ifndef __riscv
static uint16_t make_j_16(uint16_t *from, void *to) {
uint32_t ui_from = (uint32_t)from;
uint32_t ui_to = (uint32_t)to;
int32_t delta = (int32_t)(ui_to - ui_from - 4);
assert(delta >= -2048 && delta <= 2046 && !(delta & 1));
return (uint16_t)(0xe000 | ((delta >> 1) & 0x7ff));
}{ ... }
static void insert_bl_32(uint16_t *from, void *to) {
uint32_t ui_from = (uint32_t)from;
uint32_t ui_to = (uint32_t)to;
uint32_t delta = (ui_to - ui_from - 4) / 2;
assert(!(delta >> 11u));
from[0] = (uint16_t)(0xf000 | ((delta >> 11u) & 0x7ffu));
from[1] = (uint16_t)(0xf800 | (delta & 0x7ffu));
}{ ... }
static inline void *resolve_j_16(uint16_t *inst) {
assert(0x1c == (*inst)>>11u);
int32_t i_addr = (*inst) << 21u;
i_addr /= (int32_t)(1u<<21u);
return inst + 2 + i_addr;
}{ ... }
/* ... */#else
static uint16_t make_jal_16(uint16_t *from, void *to) {
uint32_t ui_from = (uint32_t)from;
uint32_t ui_to = (uint32_t)to;
int32_t delta = (int32_t)(ui_to - ui_from);
assert(delta >= -2048 && delta <= 2046 && !(delta & 1));
return 0x2001u | riscv_encode_imm_cj((uint32_t)delta);
}make_jal_16 (uint16_t *from, void *to) { ... }
static uint16_t make_j_16(uint16_t *from, void *to) {
return 0x8000u | make_jal_16(from, to);
}make_j_16 (uint16_t *from, void *to) { ... }
static inline uint32_t make_call_inst1(void *to) {
return 0x000000b7u | riscv_encode_imm_u_hi((uintptr_t)to);
}make_call_inst1 (void *to) { ... }
static inline uint32_t make_call_inst2(void *to) {
return 0x000080e7u | riscv_encode_imm_i((uintptr_t)to);
}make_call_inst2 (void *to) { ... }
static inline uint32_t make_jal_t0_32(uint32_t *from, void *to) {
return 0x000002efu | riscv_encode_imm_j((uintptr_t)to - (uintptr_t)from);
}make_jal_t0_32 (uint32_t *from, void *to) { ... }
static void *resolve_j_16(uint16_t *inst) {
uint32_t inst32 = (uint32_t)*inst;
uint32_t udiff =
((inst32 & 0x0038) >> 2) +
((inst32 & 0x0800) >> 7) +
((inst32 & 0x0004) << 3) +
((inst32 & 0x0080) >> 1) +
((inst32 & 0x0040) << 1) +
((inst32 & 0x0600) >> 1) +
((inst32 & 0x0100) << 2) -
((inst32 & 0x2000) >> 2);
return (void *)((uint32_t)inst + udiff);
}resolve_j_16 (uint16_t *inst) { ... }
/* ... */
#endif
static inline int8_t slot_diff(struct irq_handler_chain_slot *to, struct irq_handler_chain_slot *from) {
static_assert(sizeof(struct irq_handler_chain_slot) == 12, "");
#ifdef __riscv
return (int8_t)(to - from);/* ... */
#else
int32_t result = 0xaaaa;
pico_default_asm (
"subs %1, %2\n"
"adcs %1, %1\n"
"muls %0, %1\n"
"lsrs %0, %0, #20\n"
: "+l" (result), "+l" (to)
: "l" (from)
: "cc"
);
return (int8_t)result;/* ... */
#endif
}{ ... }
#ifndef __riscv
static const uint16_t inst16_return_from_last_slot = 0xbd01;
#else
static const uint16_t inst16_return_from_last_slot = 0xbe42;
#endif
static inline int8_t get_slot_index(struct irq_handler_chain_slot *slot) {
return slot_diff(slot, irq_handler_chain_slots);
}{ ... }
#endif/* ... */
void irq_add_shared_handler(uint num, irq_handler_t handler, uint8_t order_priority) {
check_irq_param(num);
#if PICO_NO_RAM_VECTOR_TABLE
panic_unsupported();
#elif PICO_DISABLE_SHARED_IRQ_HANDLERS
irq_set_exclusive_handler(num, handler);
#else
spin_lock_t *lock = spin_lock_instance(PICO_SPINLOCK_ID_IRQ);
uint32_t save = spin_lock_blocking(lock);
hard_assert(irq_handler_chain_free_slot_head >= 0);
struct irq_handler_chain_slot *slot = &irq_handler_chain_slots[irq_handler_chain_free_slot_head];
int8_t slot_index = irq_handler_chain_free_slot_head;
irq_handler_chain_free_slot_head = slot->link;
irq_handler_t vtable_handler = get_vtable()[VTABLE_FIRST_IRQ + num];
if (!is_shared_irq_raw_handler(vtable_handler)) {
hard_assert(vtable_handler == __unhandled_user_irq);
struct irq_handler_chain_slot slot_data = {
#ifndef __riscv
.inst1 = 0xa100,
.inst2 = make_j_16(&slot->inst2, (void *) irq_handler_chain_first_slot),
.handler = handler,/* ... */
#else
.inst1 = make_jal_t0_32(&slot->inst1, irq_handler_chain_first_slot),
.inst2 = (uint32_t)handler, /* ... */
#endif
.inst3 = inst16_return_from_last_slot,
.link = -1,
.priority = order_priority
...};
*slot = slot_data;
vtable_handler = (irq_handler_t)add_thumb_bit(slot);
}if (!is_shared_irq_raw_handler(vtable_handler)) { ... } else {
assert(!((((uintptr_t)remove_thumb_bit(vtable_handler)) - ((uintptr_t)irq_handler_chain_slots)) % sizeof(struct irq_handler_chain_slot)));
struct irq_handler_chain_slot *prev_slot = NULL;
struct irq_handler_chain_slot *existing_vtable_slot = remove_thumb_bit((void *) vtable_handler);
struct irq_handler_chain_slot *cur_slot = existing_vtable_slot;
while (cur_slot->priority > order_priority) {
prev_slot = cur_slot;
if (cur_slot->link < 0) break;
cur_slot = &irq_handler_chain_slots[cur_slot->link];
}while (cur_slot->priority > order_priority) { ... }
if (prev_slot) {
struct irq_handler_chain_slot slot_data = {
#ifndef __riscv
.inst1 = 0x4801,
.inst2 = 0x4780,
.handler = handler,/* ... */
#else
.inst1 = make_call_inst1(handler),
.inst2 = make_call_inst2(handler), /* ... */
#endif
.inst3 = prev_slot->link >= 0 ?
make_j_16(&slot->inst3, resolve_j_16(&prev_slot->inst3)) :
inst16_return_from_last_slot,
.link = prev_slot->link,
.priority = order_priority
...};
prev_slot->inst3 = make_j_16(&prev_slot->inst3, slot),
prev_slot->link = slot_index;
*slot = slot_data;
}if (prev_slot) { ... } else {
struct irq_handler_chain_slot slot_data = {
#ifndef __riscv
.inst1 = 0xa100,
.inst2 = make_j_16(&slot->inst2, (void *) irq_handler_chain_first_slot),
.handler = handler,/* ... */
#else
.inst1 = make_jal_t0_32(&slot->inst1, irq_handler_chain_first_slot),
.inst2 = (uint32_t)handler, /* ... */
#endif
.inst3 = make_j_16(&slot->inst3, existing_vtable_slot),
.link = get_slot_index(existing_vtable_slot),
.priority = order_priority,
...};
*slot = slot_data;
#ifndef __riscv
existing_vtable_slot->inst1 = 0x4801;
existing_vtable_slot->inst2 = 0x4780; /* ... */
#else
void *handler_of_existing_head = (void*)existing_vtable_slot->inst2;
existing_vtable_slot->inst1 = make_call_inst1(handler_of_existing_head);
existing_vtable_slot->inst2 = make_call_inst2(handler_of_existing_head);/* ... */
#endif
vtable_handler = (irq_handler_t)add_thumb_bit(slot);
}else { ... }
}else { ... }
set_raw_irq_handler_and_unlock(num, vtable_handler, save);/* ... */
#endif
}{ ... }
#if !PICO_DISABLE_SHARED_IRQ_HANDLERS
static inline irq_handler_t handler_from_slot(struct irq_handler_chain_slot *slot) {
#ifndef __riscv
return slot->handler;
#else
if (slot->inst1 & 0x8u) {
return (irq_handler_t)slot->inst2;
}if (slot->inst1 & 0x8u) { ... } else {
return (irq_handler_t)(
((slot->inst1 >> 12) << 12) + (uint32_t)((int32_t)slot->inst2 >> 20)
);
}else { ... }
/* ... */#endif
}{ ... }
/* ... */#endif
void irq_remove_handler(uint num, irq_handler_t handler) {
#if !PICO_NO_RAM_VECTOR_TABLE
spin_lock_t *lock = spin_lock_instance(PICO_SPINLOCK_ID_IRQ);
uint32_t save = spin_lock_blocking(lock);
irq_handler_t vtable_handler = get_vtable()[VTABLE_FIRST_IRQ + num];
if (vtable_handler != __unhandled_user_irq && vtable_handler != handler) {
#if !PICO_DISABLE_SHARED_IRQ_HANDLERS
if (is_shared_irq_raw_handler(vtable_handler)) {
bool was_enabled = irq_is_enabled(num);
irq_set_enabled(num, false);
__dmb();
uint exception = __get_current_exception();
hard_assert(!exception || exception == num + VTABLE_FIRST_IRQ);
struct irq_handler_chain_slot *prev_slot = NULL;
struct irq_handler_chain_slot *existing_vtable_slot = remove_thumb_bit((void *) vtable_handler);
struct irq_handler_chain_slot *to_free_slot = existing_vtable_slot;
while (handler_from_slot(to_free_slot) != handler) {
prev_slot = to_free_slot;
if (to_free_slot->link < 0) break;
to_free_slot = &irq_handler_chain_slots[to_free_slot->link];
}while (handler_from_slot(to_free_slot) != handler) { ... }
if (handler_from_slot(to_free_slot) == handler) {
int8_t next_slot_index = to_free_slot->link;
if (next_slot_index >= 0) {
struct irq_handler_chain_slot *next_slot = &irq_handler_chain_slots[next_slot_index];
#ifndef __riscv
to_free_slot->handler = next_slot->handler;
#else
irq_handler_t handler_of_next_slot = handler_from_slot(next_slot);
if (to_free_slot == existing_vtable_slot) {
to_free_slot->inst2 = (uint32_t)handler_of_next_slot;
}if (to_free_slot == existing_vtable_slot) { ... } else {
to_free_slot->inst1 = make_call_inst1(handler_of_next_slot);
to_free_slot->inst2 = make_call_inst2(handler_of_next_slot);
}else { ... }
/* ... */#endif
to_free_slot->priority = next_slot->priority;
to_free_slot->link = next_slot->link;
to_free_slot->inst3 = next_slot->link >= 0 ?
make_j_16(&to_free_slot->inst3, resolve_j_16(&next_slot->inst3)) :
inst16_return_from_last_slot,
next_slot->link = irq_handler_chain_free_slot_head;
irq_handler_chain_free_slot_head = next_slot_index;
}if (next_slot_index >= 0) { ... } else {
if (!exception) {
if (prev_slot) {
prev_slot->link = -1;
prev_slot->inst3 = inst16_return_from_last_slot;
}if (prev_slot) { ... } else {
vtable_handler = __unhandled_user_irq;
}else { ... }
to_free_slot->link = irq_handler_chain_free_slot_head;
irq_handler_chain_free_slot_head = get_slot_index(to_free_slot);
}if (!exception) { ... } else {
#ifndef __riscv
insert_bl_32(&to_free_slot->inst3, (void *) irq_handler_chain_remove_tail);/* ... */
#else
to_free_slot->inst3 = make_jal_16(&to_free_slot->inst3, (void*) irq_handler_chain_remove_tail);
#endif
}else { ... }
}else { ... }
}if (handler_from_slot(to_free_slot) == handler) { ... } else {
assert(false);
}else { ... }
irq_set_enabled(num, was_enabled);
}if (is_shared_irq_raw_handler(vtable_handler)) { ... }
/* ... */#else
assert(false);
#endif
}if (vtable_handler != __unhandled_user_irq && vtable_handler != handler) { ... } else {
vtable_handler = __unhandled_user_irq;
}else { ... }
set_raw_irq_handler_and_unlock(num, vtable_handler, save);/* ... */
#else
panic_unsupported();
#endif
}{ ... }
#ifndef __riscv
static io_rw_32 *nvic_ipr0(void) {
return (io_rw_32 *)(PPB_BASE + ARM_CPU_PREFIXED(NVIC_IPR0_OFFSET));
}{ ... }
/* ... */#endif
void irq_set_priority(uint num, uint8_t hardware_priority) {
check_irq_param(num);
#ifdef __riscv
hardware_priority = (uint8_t)((hardware_priority >> 4) ^ 0xf);
hazard3_irqarray_clear(RVCSR_MEIPRA_OFFSET, num / 4, 0xfu << (4 * (num % 4)));
hazard3_irqarray_set(RVCSR_MEIPRA_OFFSET, num / 4, hardware_priority << (4 * (num % 4)));/* ... */
#else
io_rw_32 *p = nvic_ipr0() + (num >> 2);
*p = (*p & ~(0xffu << (8 * (num & 3u)))) | (((uint32_t) hardware_priority) << (8 * (num & 3u)));/* ... */
#endif
}{ ... }
uint irq_get_priority(uint num) {
check_irq_param(num);
#ifdef __riscv
uint16_t priority_row = (uint16_t) hazard3_irqarray_read(RVCSR_MEIPRA_OFFSET, num / 4u);
uint8_t priority_4bit = (priority_row >> (4 * (num % 4))) & 0xfu;
return ((priority_4bit ^ 0xfu) << 4u);/* ... */
#else
io_rw_32 *p = nvic_ipr0() + (num >> 2);
return (uint8_t)(*p >> (8 * (num & 3u)));/* ... */
#endif
}{ ... }
#if !PICO_DISABLE_SHARED_IRQ_HANDLERS
void irq_add_tail_to_free_list(struct irq_handler_chain_slot *slot) {
irq_handler_t slot_handler = (irq_handler_t) add_thumb_bit(slot);
assert(is_shared_irq_raw_handler(slot_handler));
uint exception = __get_current_exception();
assert(exception);
spin_lock_t *lock = spin_lock_instance(PICO_SPINLOCK_ID_IRQ);
uint32_t save = spin_lock_blocking(lock);
int8_t slot_index = get_slot_index(slot);
if (slot_handler == get_vtable()[exception]) {
get_vtable()[exception] = __unhandled_user_irq;
}if (slot_handler == get_vtable()[exception]) { ... } else {
bool __unused found = false;
for(uint i=0;i<count_of(irq_handler_chain_slots);i++) {
if (irq_handler_chain_slots[i].link == slot_index) {
irq_handler_chain_slots[i].link = -1;
irq_handler_chain_slots[i].inst3 = inst16_return_from_last_slot;
found = true;
break;
}if (irq_handler_chain_slots[i].link == slot_index) { ... }
}for (uint i=0;i
assert(found);
}else { ... }
slot->link = irq_handler_chain_free_slot_head;
irq_handler_chain_free_slot_head = slot_index;
spin_unlock(lock, save);
}{ ... }
#endif/* ... */
__weak void runtime_init_per_core_irq_priorities(void) {
#if PICO_DEFAULT_IRQ_PRIORITY != 0
#ifndef __riscv
uint32_t prio4 = (PICO_DEFAULT_IRQ_PRIORITY & 0xff) * 0x1010101u;
io_rw_32 *p = nvic_ipr0();
for (uint i = 0; i < (NUM_IRQS + 3) / 4; i++) {
*p++ = prio4;
}for (uint i = 0; i < (NUM_IRQS + 3) / 4; i++) { ... }
/* ... */#else
for (uint i = 0; i < NUM_IRQS; ++i) {
irq_set_priority(i, PICO_DEFAULT_IRQ_PRIORITY);
}for (uint i = 0; i < NUM_IRQS; ++i) { ... }
/* ... */#endif/* ... */
#endif
}{ ... }
static uint get_user_irq_claim_index(uint irq_num) {
invalid_params_if(HARDWARE_IRQ, irq_num < FIRST_USER_IRQ || irq_num >= NUM_IRQS);
static_assert(NUM_IRQS - FIRST_USER_IRQ <= 8, "");
return NUM_IRQS - irq_num - 1u;
}{ ... }
void user_irq_claim(uint irq_num) {
hw_claim_or_assert(user_irq_claimed_ptr(), get_user_irq_claim_index(irq_num), "User IRQ is already claimed");
}{ ... }
void user_irq_unclaim(uint irq_num) {
hw_claim_clear(user_irq_claimed_ptr(), get_user_irq_claim_index(irq_num));
}{ ... }
int user_irq_claim_unused(bool required) {
int bit = hw_claim_unused_from_range(user_irq_claimed_ptr(), required, 0, NUM_USER_IRQS - 1, "No user IRQs are available");
if (bit >= 0) bit = (int)NUM_IRQS - bit - 1;
return bit;
}{ ... }
bool user_irq_is_claimed(uint irq_num) {
return hw_is_claimed(user_irq_claimed_ptr(), get_user_irq_claim_index(irq_num));
}{ ... }
#ifdef __riscv
static uint32_t encode_j_instruction(uintptr_t from, uintptr_t to) {
intptr_t delta = (intptr_t) (to - from);
invalid_params_if(HARDWARE_IRQ, delta & 1);
valid_params_if(HARDWARE_IRQ, ((delta >> 21) == 0 || (delta >> 21) == -1));
return 0x6fu | riscv_encode_imm_j((uint32_t)delta);
}encode_j_instruction (uintptr_t from, uintptr_t to) { ... }
irq_handler_t irq_set_riscv_vector_handler(enum riscv_vector_num index, irq_handler_t handler) {
invalid_params_if(HARDWARE_IRQ, index > RISCV_VEC_MACHINE_EXTERNAL_IRQ);
irq_handler_t *vtable = get_vtable();
valid_params_if(HARDWARE_IRQ, ((uintptr_t)vtable & 0x3) == 0x1);
irq_handler_t old = vtable[index];
vtable[index] = (irq_handler_t)encode_j_instruction((uintptr_t)&vtable[index], (uintptr_t)handler);
return old;
}irq_set_riscv_vector_handler (enum riscv_vector_num index, irq_handler_t handler) { ... }
/* ... */#endif