2020-12-26 05:56:11 +01:00
|
|
|
#pragma once
|
2015-04-09 18:32:04 +02:00
|
|
|
|
2017-07-02 00:06:39 +02:00
|
|
|
#include <inttypes.h>
|
|
|
|
|
2015-04-09 18:32:04 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2021-01-13 02:46:22 +01:00
|
|
|
#if defined(__ARMEL__) || defined(__ARMEB__)
|
|
|
|
# ifndef __OPTIMIZE__
|
|
|
|
# pragma message "Compiler optimizations disabled; wait_cpuclock() won't work as designed"
|
|
|
|
# endif
|
|
|
|
|
|
|
|
# define wait_cpuclock(x) wait_cpuclock_allnop(x)
|
|
|
|
|
|
|
|
# define CLOCK_DELAY_NOP8 "nop\n\t nop\n\t nop\n\t nop\n\t nop\n\t nop\n\t nop\n\t nop\n\t"
|
|
|
|
|
|
|
|
__attribute__((always_inline))
|
|
|
|
static inline void wait_cpuclock_allnop(unsigned int n) { /* n: 1..135 */
|
|
|
|
/* The argument n must be a constant expression.
|
|
|
|
* That way, compiler optimization will remove unnecessary code. */
|
|
|
|
if (n < 1) { return; }
|
|
|
|
if (n > 8) {
|
|
|
|
unsigned int n8 = n/8;
|
|
|
|
n = n - n8*8;
|
|
|
|
switch (n8) {
|
|
|
|
case 16: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 15: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 14: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 13: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 12: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 11: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 10: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 9: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 8: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 7: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 6: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 5: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 4: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 3: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 2: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 1: asm volatile (CLOCK_DELAY_NOP8::: "memory");
|
|
|
|
case 0: break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
switch (n) {
|
|
|
|
case 8: asm volatile ("nop"::: "memory");
|
|
|
|
case 7: asm volatile ("nop"::: "memory");
|
|
|
|
case 6: asm volatile ("nop"::: "memory");
|
|
|
|
case 5: asm volatile ("nop"::: "memory");
|
|
|
|
case 4: asm volatile ("nop"::: "memory");
|
|
|
|
case 3: asm volatile ("nop"::: "memory");
|
|
|
|
case 2: asm volatile ("nop"::: "memory");
|
|
|
|
case 1: asm volatile ("nop"::: "memory");
|
|
|
|
case 0: break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-04-09 18:32:04 +02:00
|
|
|
#if defined(__AVR__)
|
2019-08-30 20:19:03 +02:00
|
|
|
# include <util/delay.h>
|
|
|
|
# define wait_ms(ms) _delay_ms(ms)
|
|
|
|
# define wait_us(us) _delay_us(us)
|
2021-01-13 02:46:22 +01:00
|
|
|
# define wait_cpuclock(x) __builtin_avr_delay_cycles(x)
|
2017-09-30 01:17:30 +02:00
|
|
|
#elif defined PROTOCOL_CHIBIOS
|
2020-12-11 03:45:24 +01:00
|
|
|
# include <ch.h>
|
2019-08-30 20:19:03 +02:00
|
|
|
# define wait_ms(ms) \
|
|
|
|
do { \
|
|
|
|
if (ms != 0) { \
|
|
|
|
chThdSleepMilliseconds(ms); \
|
|
|
|
} else { \
|
|
|
|
chThdSleepMicroseconds(1); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
# define wait_us(us) \
|
|
|
|
do { \
|
|
|
|
if (us != 0) { \
|
|
|
|
chThdSleepMicroseconds(us); \
|
|
|
|
} else { \
|
|
|
|
chThdSleepMicroseconds(1); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
2018-12-18 21:21:25 +01:00
|
|
|
#elif defined PROTOCOL_ARM_ATSAM
|
2019-08-30 20:19:03 +02:00
|
|
|
# include "clks.h"
|
|
|
|
# define wait_ms(ms) CLK_delay_ms(ms)
|
|
|
|
# define wait_us(us) CLK_delay_us(us)
|
2017-06-16 20:41:34 +02:00
|
|
|
#else // Unit tests
|
2017-07-02 00:06:39 +02:00
|
|
|
void wait_ms(uint32_t ms);
|
2019-08-30 20:19:03 +02:00
|
|
|
# define wait_us(us) wait_ms(us / 1000)
|
2017-06-16 20:41:34 +02:00
|
|
|
#endif
|
2015-04-09 18:32:04 +02:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|