1
6
7
8
9
10
11
12
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
47
48
49
50
51
52
53
54
55
61
62
71
72
73
74
79
80
89
90
91
92
97
98
99
100
101
102
103
104
105
106
107
116
117
118
119
120
121
122
123
124
125
126
136
137
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
169
179
180
187
191
192
198
201
202
208
211
212
213
214
215
216
217
218
219
220
223
224
225
226
227
228
229
230
231
232
235
236
237
244
252
253
260
268
269
276
286
287
294
304
305
306
307
308
309
310
311
312
313
314
321
322
323
324
325
326
327
328
329
330
331
338
339
340
341
342
343
344
345
346
347
348
355
356
357
358
359
360
361
362
363
364
365
372
373
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
409
410
411
412
413
414
415
416
417
418
419
430
431
432
433
434
435
436
437
438
439
440
449
450
451
452
453
454
455
456
457
458
459
470
471
472
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
495
496
503
504
505
506
510
511
512
513
514
515
/* ... */
#ifndef _HARDWARE_DIVIDER_H
#define _HARDWARE_DIVIDER_H
#include "pico.h"
/* ... */
#if HAS_SIO_DIVIDER
#include "hardware/structs/sio.h"
#else
#define PICO_EMULATE_DIVIDER 1
#endif
#ifdef __cplusplus
extern "C" {
#endif
typedef uint64_t divmod_result_t;
#if PICO_EMULATE_DIVIDER
extern divmod_result_t hw_divider_results[NUM_CORES];
static inline int __sign_of(int32_t v) {
return v > 0 ? 1 : (v < 0 ? -1 : 0);
}{ ... }
#endif/* ... */
/* ... */
#if !PICO_EMULATE_DIVIDER
divmod_result_t hw_divider_divmod_s32(int32_t a, int32_t b);
#else
static inline divmod_result_t hw_divider_divmod_s32(int32_t a, int32_t b) {
if (!b) return (((uint64_t)a)<<32u) | (uint32_t)(-__sign_of(a));
return (((uint64_t)(a%b))<<32u) | (uint32_t)(a/b);
}{ ... }
/* ... */#endif
/* ... */
#if !PICO_EMULATE_DIVIDER
divmod_result_t hw_divider_divmod_u32(uint32_t a, uint32_t b);
#else
static inline divmod_result_t hw_divider_divmod_u32(uint32_t a, uint32_t b) {
if (!b) return (((uint64_t)a)<<32u) | (uint32_t)(-1);
return (((uint64_t)(a%b))<<32u) | (a/b);
}{ ... }
/* ... */#endif
/* ... */
static inline void hw_divider_divmod_s32_start(int32_t a, int32_t b) {
#if !PICO_EMULATE_DIVIDER
check_hw_layout( sio_hw_t, div_sdividend, SIO_DIV_SDIVIDEND_OFFSET);
sio_hw->div_sdividend = (uint32_t)a;
sio_hw->div_sdivisor = (uint32_t)b;/* ... */
#else
hw_divider_divmod_s32(a, b);
#endif
}{ ... }
/* ... */
static inline void hw_divider_divmod_u32_start(uint32_t a, uint32_t b) {
#if !PICO_EMULATE_DIVIDER
check_hw_layout(
sio_hw_t, div_udividend, SIO_DIV_UDIVIDEND_OFFSET);
sio_hw->div_udividend = a;
sio_hw->div_udivisor = b;/* ... */
#else
hw_divider_divmod_u32(a, b);
#endif
}{ ... }
/* ... */
static inline void hw_divider_wait_ready(void) {
#if !PICO_EMULATE_DIVIDER
static_assert(SIO_DIV_CSR_READY_BITS == 1, "");
uint32_t tmp;
pico_default_asm_volatile (
"hw_divider_result_loop_%=:"
"ldr %0, [%1, %2]\n\t"
"lsrs %0, %0, #1\n\t"
"bcc hw_divider_result_loop_%=\n\t"
: "=&l" (tmp)
: "l" (sio_hw), "I" (SIO_DIV_CSR_OFFSET)
: "cc"
);/* ... */
#endif
}{ ... }
/* ... */
static inline divmod_result_t hw_divider_result_nowait(void) {
#if !PICO_EMULATE_DIVIDER
divmod_result_t rc = ((divmod_result_t) sio_hw->div_remainder) << 32u;
rc |= sio_hw->div_quotient;
return rc;/* ... */
#else
return hw_divider_results[get_core_num()];
#endif
}{ ... }
/* ... */
static inline divmod_result_t hw_divider_result_wait(void) {
hw_divider_wait_ready();
return hw_divider_result_nowait();
}{ ... }
/* ... */
inline static uint32_t to_quotient_u32(divmod_result_t r) {
return (uint32_t) r;
}{ ... }
/* ... */
inline static int32_t to_quotient_s32(divmod_result_t r) {
return (int32_t)(uint32_t)r;
}{ ... }
/* ... */
inline static uint32_t to_remainder_u32(divmod_result_t r) {
return (uint32_t)(r >> 32u);
}{ ... }
/* ... */
inline static int32_t to_remainder_s32(divmod_result_t r) {
return (int32_t)(r >> 32u);
}{ ... }
/* ... */
static inline uint32_t hw_divider_u32_quotient_wait(void) {
#if !PICO_EMULATE_DIVIDER
hw_divider_wait_ready();
return sio_hw->div_quotient;/* ... */
#else
return to_quotient_u32(hw_divider_result_wait());
#endif
}{ ... }
/* ... */
static inline int32_t hw_divider_s32_quotient_wait(void) {
#if !PICO_EMULATE_DIVIDER
hw_divider_wait_ready();
return (int32_t)sio_hw->div_quotient;/* ... */
#else
return to_quotient_s32(hw_divider_result_wait());
#endif
}{ ... }
/* ... */
static inline uint32_t hw_divider_u32_remainder_wait(void) {
#if !PICO_EMULATE_DIVIDER
hw_divider_wait_ready();
uint32_t rc = sio_hw->div_remainder;
sio_hw->div_quotient;
return rc;/* ... */
#else
return to_remainder_u32(hw_divider_result_wait());
#endif
}{ ... }
/* ... */
static inline int32_t hw_divider_s32_remainder_wait(void) {
#if !PICO_EMULATE_DIVIDER
hw_divider_wait_ready();
int32_t rc = (int32_t)sio_hw->div_remainder;
sio_hw->div_quotient;
return rc;/* ... */
#else
return to_remainder_s32(hw_divider_result_wait());
#endif
}{ ... }
/* ... */
static inline uint32_t hw_divider_u32_quotient(uint32_t a, uint32_t b) {
#if !PICO_EMULATE_DIVIDER
return to_quotient_u32(hw_divider_divmod_u32(a, b));
#else
return b ? (a / b) : (uint32_t)(-1);
#endif
}{ ... }
/* ... */
static inline uint32_t hw_divider_u32_remainder(uint32_t a, uint32_t b) {
#if !PICO_EMULATE_DIVIDER
return to_remainder_u32(hw_divider_divmod_u32(a, b));
#else
return b ? (a % b) : a;
#endif
}{ ... }
/* ... */
static inline int32_t hw_divider_quotient_s32(int32_t a, int32_t b) {
#if !PICO_EMULATE_DIVIDER
return to_quotient_s32(hw_divider_divmod_s32(a, b));
#else
return b ? (a / b) : -1;
#endif
}{ ... }
/* ... */
static inline int32_t hw_divider_remainder_s32(int32_t a, int32_t b) {
#if !PICO_EMULATE_DIVIDER
return to_remainder_s32(hw_divider_divmod_s32(a, b));
#else
return b ? (a % b) : a;
#endif
}{ ... }
/* ... */
static inline void hw_divider_pause(void) {
#if !PICO_EMULATE_DIVIDER
pico_default_asm_volatile(
"b _1_%=\n"
"_1_%=:\n"
"b _2_%=\n"
"_2_%=:\n"
"b _3_%=\n"
"_3_%=:\n"
"b _4_%=\n"
"_4_%=:\n"
:::);/* ... */
#endif
}{ ... }
/* ... */
static inline uint32_t hw_divider_u32_quotient_inlined(uint32_t a, uint32_t b) {
#if !PICO_EMULATE_DIVIDER
hw_divider_divmod_u32_start(a, b);
hw_divider_pause();
return sio_hw->div_quotient;/* ... */
#else
return hw_divider_u32_quotient(a,b);
#endif
}{ ... }
/* ... */
static inline uint32_t hw_divider_u32_remainder_inlined(uint32_t a, uint32_t b) {
#if !PICO_EMULATE_DIVIDER
hw_divider_divmod_u32_start(a, b);
hw_divider_pause();
uint32_t rc = sio_hw->div_remainder;
sio_hw->div_quotient;
return rc;/* ... */
#else
return hw_divider_u32_remainder(a,b);
#endif
}{ ... }
/* ... */
static inline int32_t hw_divider_s32_quotient_inlined(int32_t a, int32_t b) {
#if !PICO_EMULATE_DIVIDER
hw_divider_divmod_s32_start(a, b);
hw_divider_pause();
return (int32_t)sio_hw->div_quotient;/* ... */
#else
return hw_divider_quotient_s32(a,b);
#endif
}{ ... }
/* ... */
static inline int32_t hw_divider_s32_remainder_inlined(int32_t a, int32_t b) {
#if !PICO_EMULATE_DIVIDER
hw_divider_divmod_s32_start(a, b);
hw_divider_pause();
int32_t rc = (int32_t)sio_hw->div_remainder;
sio_hw->div_quotient;
return rc;/* ... */
#else
return hw_divider_remainder_s32(a,b);
#endif
}{ ... }
#if !PICO_EMULATE_DIVIDER
typedef struct {
uint32_t values[4];
...} hw_divider_state_t;/* ... */
#else
typedef uint64_t hw_divider_state_t;
#endif
/* ... */
#if !PICO_EMULATE_DIVIDER
void hw_divider_save_state(hw_divider_state_t *dest);
#else
static inline void hw_divider_save_state(hw_divider_state_t *dest) {
*dest = hw_divider_results[get_core_num()];
}{ ... }
/* ... */#endif
/* ... */
#if !PICO_EMULATE_DIVIDER
void hw_divider_restore_state(hw_divider_state_t *src);
#else
static inline void hw_divider_restore_state(hw_divider_state_t *src) {
hw_divider_results[get_core_num()] = *src;
}{ ... }
/* ... */#endif
#ifdef __cplusplus
}extern "C" { ... }
#endif
/* ... */
#endif