From ec41c437515981b5b00d8453965d23dcf01cd000 Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:53:44 +0100 Subject: [PATCH 01/10] sys: introduce coreclk utility function --- sys/include/clk.h | 49 +++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 49 insertions(+) create mode 100644 sys/include/clk.h diff --git a/sys/include/clk.h b/sys/include/clk.h new file mode 100644 index 0000000000..577642bba9 --- /dev/null +++ b/sys/include/clk.h @@ -0,0 +1,49 @@ +/* + * Copyright (C) 2021 Inria + * + * This file is subject to the terms and conditions of the GNU Lesser + * General Public License v2.1. See the file LICENSE in the top level + * directory for more details. + */ + +/** + * @defgroup sys_clk System core clock + * @ingroup sys + * @{ + * + * @file + * @brief System core clock utility functions + */ + +#ifndef CLK_H +#define CLK_H + +#include +#include +#include "periph_conf.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * Get the current system core clock frequency in Hz. + * + * @returns current system core clock frequency in Hz + */ +static inline uint32_t coreclk(void) { +#if defined(CLOCK_CORECLOCK) + return CLOCK_CORECLOCK; +#else + extern uint32_t cpu_coreclk; + assert(cpu_coreclk != 0); + return cpu_coreclk; +#endif +} + +#ifdef __cplusplus +} +#endif + +#endif /* CLK_H */ +/** @} */ From b206658b51a4e5e18307f28bbecdbac9e928ac16 Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:54:31 +0100 Subject: [PATCH 02/10] cpu/native: define default CLOCK_CORECLOCK constant --- cpu/native/include/periph_conf.h | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/cpu/native/include/periph_conf.h b/cpu/native/include/periph_conf.h index be853f52d9..eaf611029d 100644 --- a/cpu/native/include/periph_conf.h +++ b/cpu/native/include/periph_conf.h @@ -16,10 +16,21 @@ #ifndef PERIPH_CONF_H #define PERIPH_CONF_H +#include "macros/units.h" + #ifdef __cplusplus extern "C" { #endif +/** + * @brief System core clock in Hz + * + * 1GHz is an arbitrary value used for compatibility with other platforms. + */ +#ifndef CLOCK_CORECLOCK +#define CLOCK_CORECLOCK GHZ(1) +#endif + /** * @name hardware timer clock skew avoidance * @{ From 2eb800cb8bc0f4f8c4dc0042a12a65a6cbdb480c Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:55:07 +0100 Subject: [PATCH 03/10] cpu/fe310: use coreclk instead of cpu_freq --- cpu/fe310/clock.c | 20 +++++--------------- cpu/fe310/cpu.c | 3 ++- cpu/fe310/include/clk_conf.h | 2 +- cpu/fe310/include/periph_cpu.h | 7 ------- cpu/fe310/periph/i2c.c | 3 ++- cpu/fe310/periph/spi.c | 3 ++- cpu/fe310/periph/uart.c | 3 ++- 7 files changed, 14 insertions(+), 27 deletions(-) diff --git a/cpu/fe310/clock.c b/cpu/fe310/clock.c index 4e3e586561..cfe26a2b8a 100644 --- a/cpu/fe310/clock.c +++ b/cpu/fe310/clock.c @@ -17,13 +17,14 @@ * @} */ +#include "clk.h" #include "cpu.h" #include "periph_conf.h" #include "vendor/prci_driver.h" #if IS_ACTIVE(CONFIG_USE_CLOCK_HFROSC) || IS_ACTIVE(CONFIG_USE_CLOCK_HFROSC_PLL) -static uint32_t _cpu_frequency = 0; +uint32_t cpu_coreclk = 0; #endif void fe310_clock_init(void) @@ -92,21 +93,10 @@ void fe310_clock_init(void) /* Don't use PLL clock source */ PRCI_REG(PRCI_PLLCFG) &= ~PLL_SEL(PLL_SEL_PLL); } -} -uint32_t cpu_freq(void) -{ #if IS_ACTIVE(CONFIG_USE_CLOCK_HFROSC) || IS_ACTIVE(CONFIG_USE_CLOCK_HFROSC_PLL) - /* Clock frequency with HFROSC cannot be determined precisely from - settings */ - /* If not done already, estimate the CPU frequency */ - if (_cpu_frequency == 0) { - /* Ignore the first run (for icache reasons) */ - _cpu_frequency = PRCI_measure_mcycle_freq(3000, RTC_FREQ); - _cpu_frequency = PRCI_measure_mcycle_freq(3000, RTC_FREQ); - } - return _cpu_frequency; -#else - return CLOCK_CORECLOCK; + /* Ignore the first run (for icache reasons) */ + cpu_coreclk = PRCI_measure_mcycle_freq(3000, RTC_FREQ); + cpu_coreclk = PRCI_measure_mcycle_freq(3000, RTC_FREQ); #endif } diff --git a/cpu/fe310/cpu.c b/cpu/fe310/cpu.c index a10ea94dd6..ffa271705d 100644 --- a/cpu/fe310/cpu.c +++ b/cpu/fe310/cpu.c @@ -17,6 +17,7 @@ * @} */ +#include "clk.h" #include "cpu.h" #include "periph/init.h" #include "periph_conf.h" @@ -80,7 +81,7 @@ void flash_init(void) * by the following formula (Fin is processor/tile-link clock): * Fsck = Fin/(2(div + 1)) */ - uint32_t freq = cpu_freq(); + uint32_t freq = coreclk(); uint32_t sckdiv = (freq - 1) / (MAX_FLASH_FREQ * 2); if (sckdiv > SCKDIV_SAFE) { diff --git a/cpu/fe310/include/clk_conf.h b/cpu/fe310/include/clk_conf.h index 4a45657116..61f54bf409 100644 --- a/cpu/fe310/include/clk_conf.h +++ b/cpu/fe310/include/clk_conf.h @@ -113,7 +113,7 @@ extern "C" { /* When using HFROSC input clock, the core clock cannot be computed from settings, - call cpu_freq() to get the configured CPU frequency. + in this case, coreclk() returns the configured CPU frequency. */ #ifndef CONFIG_CLOCK_DESIRED_FREQUENCY #define CONFIG_CLOCK_DESIRED_FREQUENCY MHZ(320) diff --git a/cpu/fe310/include/periph_cpu.h b/cpu/fe310/include/periph_cpu.h index e96d4f7cdb..ee9c02d892 100644 --- a/cpu/fe310/include/periph_cpu.h +++ b/cpu/fe310/include/periph_cpu.h @@ -175,13 +175,6 @@ typedef struct { */ void fe310_clock_init(void); -/** - * @brief Get and eventually compute the current CPU core clock frequency - * - * @return the cpu core clock frequency in Hz - */ -uint32_t cpu_freq(void); - /** @} */ #ifdef __cplusplus diff --git a/cpu/fe310/periph/i2c.c b/cpu/fe310/periph/i2c.c index f3b2f4b9a2..bc8a288230 100644 --- a/cpu/fe310/periph/i2c.c +++ b/cpu/fe310/periph/i2c.c @@ -23,6 +23,7 @@ #include #include +#include "clk.h" #include "cpu.h" #include "mutex.h" @@ -69,7 +70,7 @@ void i2c_init(i2c_t dev) /* Compute prescale: presc = (CORE_CLOCK / (5 * I2C_SPEED)) - 1 */ uint16_t presc = - ((uint16_t)(cpu_freq() / 1000) / + ((uint16_t)(coreclk() / 1000) / (5 * _fe310_i2c_speed[i2c_config[dev].speed])) - 1; DEBUG("[i2c] init: computed prescale: %i (0x%02X|0x%02X)\n", presc, diff --git a/cpu/fe310/periph/spi.c b/cpu/fe310/periph/spi.c index e880415b9f..f79c6b3346 100644 --- a/cpu/fe310/periph/spi.c +++ b/cpu/fe310/periph/spi.c @@ -24,6 +24,7 @@ #include +#include "clk.h" #include "cpu.h" #include "mutex.h" #include "periph/spi.h" @@ -62,7 +63,7 @@ void spi_init(spi_t dev) mutex_init(&lock); for (uint8_t i = 0; i < SPI_CLK_NUMOF; ++i) { - _spi_clks_config[i] = SPI_DIV_UP(cpu_freq(), 2 * _spi_clks[i]) - 1; + _spi_clks_config[i] = SPI_DIV_UP(coreclk(), 2 * _spi_clks[i]) - 1; } /* trigger pin initialization */ diff --git a/cpu/fe310/periph/uart.c b/cpu/fe310/periph/uart.c index 259f3e43fe..b9f45875e6 100644 --- a/cpu/fe310/periph/uart.c +++ b/cpu/fe310/periph/uart.c @@ -22,6 +22,7 @@ #include #include +#include "clk.h" #include "irq.h" #include "cpu.h" #include "periph/uart.h" @@ -87,7 +88,7 @@ int uart_init(uart_t dev, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg) uart_poweron(dev); /* Calculate baudrate divisor given current CPU clk rate */ - uartDiv = cpu_freq() / baudrate; + uartDiv = coreclk() / baudrate; /* Enable UART 8-N-1 at given baudrate */ _REG32(uart_config[dev].addr, UART_REG_DIV) = uartDiv; From ea3c59f41a03e09573952f9a24e5f30a8649d072 Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:55:35 +0100 Subject: [PATCH 04/10] tests: use coreclk() instead of CLOCK_CORECLOCK --- tests/bench_msg_pingpong/main.c | 5 ++--- tests/bench_mutex_pingpong/main.c | 5 ++--- tests/bench_sched_nop/main.c | 5 ++--- tests/bench_thread_flags_pingpong/main.c | 5 ++--- tests/bench_thread_yield_pingpong/main.c | 5 ++--- tests/leds/main.c | 7 ++----- tests/periph_timer/Makefile | 6 +++--- tests/periph_timer/main.c | 1 + tests/periph_uart_mode/main.c | 4 +++- 9 files changed, 19 insertions(+), 24 deletions(-) diff --git a/tests/bench_msg_pingpong/main.c b/tests/bench_msg_pingpong/main.c index 334038f6ff..bdbfe8208f 100644 --- a/tests/bench_msg_pingpong/main.c +++ b/tests/bench_msg_pingpong/main.c @@ -23,6 +23,7 @@ #include #include "macros/units.h" #include "thread.h" +#include "clk.h" #include "msg.h" #include "xtimer.h" @@ -80,10 +81,8 @@ int main(void) } printf("{ \"result\" : %"PRIu32, n); -#ifdef CLOCK_CORECLOCK printf(", \"ticks\" : %"PRIu32, - (uint32_t)((TEST_DURATION_US/US_PER_MS) * (CLOCK_CORECLOCK/KHZ(1)))/n); -#endif + (uint32_t)((TEST_DURATION_US/US_PER_MS) * (coreclk()/KHZ(1)))/n); puts(" }"); return 0; diff --git a/tests/bench_mutex_pingpong/main.c b/tests/bench_mutex_pingpong/main.c index 356719fc37..bed44025f3 100644 --- a/tests/bench_mutex_pingpong/main.c +++ b/tests/bench_mutex_pingpong/main.c @@ -21,6 +21,7 @@ #include #include "macros/units.h" +#include "clk.h" #include "mutex.h" #include "thread.h" #include "xtimer.h" @@ -79,10 +80,8 @@ int main(void) } printf("{ \"result\" : %"PRIu32, n); -#ifdef CLOCK_CORECLOCK printf(", \"ticks\" : %"PRIu32, - (uint32_t)((TEST_DURATION/US_PER_MS) * (CLOCK_CORECLOCK/KHZ(1)))/n); -#endif + (uint32_t)((TEST_DURATION/US_PER_MS) * (coreclk()/KHZ(1)))/n); puts(" }"); return 0; diff --git a/tests/bench_sched_nop/main.c b/tests/bench_sched_nop/main.c index 0e3381985f..19bab00baa 100644 --- a/tests/bench_sched_nop/main.c +++ b/tests/bench_sched_nop/main.c @@ -20,6 +20,7 @@ #include #include "macros/units.h" +#include "clk.h" #include "thread.h" #include "xtimer.h" @@ -53,10 +54,8 @@ int main(void) } printf("{ \"result\" : %"PRIu32, n); -#ifdef CLOCK_CORECLOCK printf(", \"ticks\" : %"PRIu32, - (uint32_t)((TEST_DURATION/US_PER_MS) * (CLOCK_CORECLOCK/KHZ(1)))/n); -#endif + (uint32_t)((TEST_DURATION/US_PER_MS) * (coreclk()/KHZ(1)))/n); puts(" }"); return 0; diff --git a/tests/bench_thread_flags_pingpong/main.c b/tests/bench_thread_flags_pingpong/main.c index 0653cdff19..84b3ac80a2 100644 --- a/tests/bench_thread_flags_pingpong/main.c +++ b/tests/bench_thread_flags_pingpong/main.c @@ -20,6 +20,7 @@ #include #include "macros/units.h" +#include "clk.h" #include "thread.h" #include "thread_flags.h" @@ -76,10 +77,8 @@ int main(void) } printf("{ \"result\" : %"PRIu32, n); -#ifdef CLOCK_CORECLOCK printf(", \"ticks\" : %"PRIu32, - (uint32_t)((TEST_DURATION/US_PER_MS) * (CLOCK_CORECLOCK/KHZ(1)))/n); -#endif + (uint32_t)((TEST_DURATION/US_PER_MS) * (coreclk()/KHZ(1)))/n); puts(" }"); return 0; diff --git a/tests/bench_thread_yield_pingpong/main.c b/tests/bench_thread_yield_pingpong/main.c index 10c758a053..aa590f064b 100644 --- a/tests/bench_thread_yield_pingpong/main.c +++ b/tests/bench_thread_yield_pingpong/main.c @@ -21,6 +21,7 @@ #include #include "macros/units.h" +#include "clk.h" #include "thread.h" #include "xtimer.h" @@ -73,10 +74,8 @@ int main(void) } printf("{ \"result\" : %"PRIu32, n); -#ifdef CLOCK_CORECLOCK printf(", \"ticks\" : %"PRIu32, - (uint32_t)((TEST_DURATION/US_PER_MS) * (CLOCK_CORECLOCK/KHZ(1)))/n); -#endif + (uint32_t)((TEST_DURATION/US_PER_MS) * (coreclk()/KHZ(1)))/n); puts(" }"); return 0; diff --git a/tests/leds/main.c b/tests/leds/main.c index 0c1bb366b1..417637e1b8 100644 --- a/tests/leds/main.c +++ b/tests/leds/main.c @@ -21,14 +21,11 @@ #include #include +#include "clk.h" #include "board.h" #include "periph_conf.h" -#ifdef CLOCK_CORECLOCK -#define DELAY_SHORT (CLOCK_CORECLOCK / 50) -#else -#define DELAY_SHORT (500000UL) -#endif +#define DELAY_SHORT (coreclk() / 50) #define DELAY_LONG (DELAY_SHORT * 4) void dumb_delay(uint32_t delay) diff --git a/tests/periph_timer/Makefile b/tests/periph_timer/Makefile index 5e5af7f83b..baeb9a7d94 100644 --- a/tests/periph_timer/Makefile +++ b/tests/periph_timer/Makefile @@ -37,7 +37,7 @@ BOARDS_TIMER_32kHz := \ stk3700 \ # -BOARDS_TIMER_CLOCK_CORECLOCK := \ +BOARDS_TIMER_SYSCLK := \ cc2538dk \ openmote-b \ openmote-cc2538 \ @@ -52,8 +52,8 @@ else ifneq (,$(filter $(BOARDS_TIMER_250kHz),$(BOARD))) TIMER_SPEED ?= 250000 else ifneq (,$(filter $(BOARDS_TIMER_32kHz),$(BOARD))) TIMER_SPEED ?= 32768 -else ifneq (,$(filter $(BOARDS_TIMER_CLOCK_CORECLOCK),$(BOARD))) - TIMER_SPEED ?= CLOCK_CORECLOCK +else ifneq (,$(filter $(BOARDS_TIMER_SYSCLK),$(BOARD))) + TIMER_SPEED ?= coreclk\(\) endif TIMER_SPEED ?= 1000000 diff --git a/tests/periph_timer/main.c b/tests/periph_timer/main.c index 1cd0a9e39b..ac398f636c 100644 --- a/tests/periph_timer/main.c +++ b/tests/periph_timer/main.c @@ -22,6 +22,7 @@ #include #include +#include "clk.h" #include "periph/timer.h" /** diff --git a/tests/periph_uart_mode/main.c b/tests/periph_uart_mode/main.c index 33e16f944b..61075a1616 100644 --- a/tests/periph_uart_mode/main.c +++ b/tests/periph_uart_mode/main.c @@ -22,6 +22,7 @@ #include #include +#include "clk.h" #include "board.h" #include "periph/uart.h" #include "periph_conf.h" @@ -69,7 +70,8 @@ static void _delay(void) * compilers would detect that the loop is only wasting CPU cycles and * optimize it out - but here the wasting of CPU cycles is desired. */ - for (volatile uint32_t i = 0; i < CLOCK_CORECLOCK / 20; i++) { } + uint32_t loops = coreclk() / 20; + for (volatile uint32_t i = 0; i < loops; i++) { } } } From 16baad825d1620db8a3b4ebe6d035cf806add6ba Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:55:45 +0100 Subject: [PATCH 05/10] drivers/ws281x: use coreclk() instead of CLOCK_CORECLOCK --- drivers/ws281x/atmega.c | 219 +++++++++++++++++++++------------------- 1 file changed, 114 insertions(+), 105 deletions(-) diff --git a/drivers/ws281x/atmega.c b/drivers/ws281x/atmega.c index 4b9f439750..5c42df686e 100644 --- a/drivers/ws281x/atmega.c +++ b/drivers/ws281x/atmega.c @@ -23,6 +23,7 @@ #include #include +#include "clk.h" #include "atmega_gpio.h" #include "ws281x.h" #include "ws281x_params.h" @@ -94,114 +95,122 @@ void ws281x_write_buffer(ws281x_t *dev, const void *buf, size_t size) mask_off = port_state & ~(1 << atmega_pin_num(dev->params.pin)); } -#if (CLOCK_CORECLOCK >= 7500000U) && (CLOCK_CORECLOCK <= 8500000U) - const uint8_t port_num = atmega_port_num(dev->params.pin); - switch (port_num) { - case PORT_B: - while (pos < end) { - uint8_t cnt = 8; - uint8_t data = *pos; - while (cnt > 0) { - __asm__ volatile( /* Cycles | 1 | 0 */ - "out %[port], %[on]" "\n\t" /* 1 | x | x */ - "sbrs %[data], 7" "\n\t" /* 1 | x | x */ - "out %[port], %[off]" "\n\t" /* 1 | - | x */ - "rjmp .+0" "\n\t" /* 2 | x | x */ - "out %[port], %[off]" "\n\t" /* 1 | x | x */ - /* Total CPU cycles for high period: - * 5 cycles for bit 1, 2 cycles for bit 0 */ - : [data] "+r" (data) - : [port] "I" (_SFR_IO_ADDR(PORTB)), - [on] "r" (mask_on), - [off] "r" (mask_off) - ); - cnt--; - data <<= 1; + if (coreclk() >= 7500000U && coreclk() <= 8500000U) { + const uint8_t port_num = atmega_port_num(dev->params.pin); + switch (port_num) { + case PORT_B: + while (pos < end) { + uint8_t cnt = 8; + uint8_t data = *pos; + while (cnt > 0) { + __asm__ volatile( /* Cycles | 1 | 0 */ + "out %[port], %[on]" "\n\t" /* 1 | x | x */ + "sbrs %[data], 7" "\n\t" /* 1 | x | x */ + "out %[port], %[off]" "\n\t" /* 1 | - | x */ + "rjmp .+0" "\n\t" /* 2 | x | x */ + "out %[port], %[off]" "\n\t" /* 1 | x | x */ + /* Total CPU cycles for high period: + * 5 cycles for bit 1, 2 cycles for bit 0 */ + : [data] "+r" (data) + : [port] "I" (_SFR_IO_ADDR(PORTB)), + [on] "r" (mask_on), + [off] "r" (mask_off) + ); + cnt--; + data <<= 1; + } + pos++; } - pos++; - } - break; - case PORT_C: - while (pos < end) { - uint8_t cnt = 8; - uint8_t data = *pos; - while (cnt > 0) { - __asm__ volatile( /* Cycles | 1 | 0 */ - "out %[port], %[on]" "\n\t" /* 1 | x | x */ - "sbrs %[data], 7" "\n\t" /* 1 | x | x */ - "out %[port], %[off]" "\n\t" /* 1 | - | x */ - "rjmp .+0" "\n\t" /* 2 | x | x */ - "out %[port], %[off]" "\n\t" /* 1 | x | x */ - /* Total CPU cycles for high period: - * 5 cycles for bit 1, 2 cycles for bit 0 */ - : [data] "+r" (data) - : [port] "I" (_SFR_IO_ADDR(PORTC)), - [on] "r" (mask_on), - [off] "r" (mask_off) - ); - cnt--; - data <<= 1; + break; +#ifdef PORT_C + case PORT_C: + while (pos < end) { + uint8_t cnt = 8; + uint8_t data = *pos; + while (cnt > 0) { + __asm__ volatile( /* Cycles | 1 | 0 */ + "out %[port], %[on]" "\n\t" /* 1 | x | x */ + "sbrs %[data], 7" "\n\t" /* 1 | x | x */ + "out %[port], %[off]" "\n\t" /* 1 | - | x */ + "rjmp .+0" "\n\t" /* 2 | x | x */ + "out %[port], %[off]" "\n\t" /* 1 | x | x */ + /* Total CPU cycles for high period: + * 5 cycles for bit 1, 2 cycles for bit 0 */ + : [data] "+r" (data) + : [port] "I" (_SFR_IO_ADDR(PORTC)), + [on] "r" (mask_on), + [off] "r" (mask_off) + ); + cnt--; + data <<= 1; + } + pos++; } - pos++; - } - break; - case PORT_D: - while (pos < end) { - uint8_t cnt = 8; - uint8_t data = *pos; - while (cnt > 0) { - __asm__ volatile( /* Cycles | 1 | 0 */ - "out %[port], %[on]" "\n\t" /* 1 | x | x */ - "sbrs %[data], 7" "\n\t" /* 1 | x | x */ - "out %[port], %[off]" "\n\t" /* 1 | - | x */ - "rjmp .+0" "\n\t" /* 2 | x | x */ - "out %[port], %[off]" "\n\t" /* 1 | x | x */ - /* Total CPU cycles for high period: - * 5 cycles for bit 1, 2 cycles for bit 0 */ - : [data] "+r" (data) - : [port] "I" (_SFR_IO_ADDR(PORTD)), - [on] "r" (mask_on), - [off] "r" (mask_off) - ); - cnt--; - data <<= 1; - } - pos++; - } - break; - default: - assert(0); - break; - } -#elif (CLOCK_CORECLOCK >= 15500000U) && (CLOCK_CORECLOCK <= 16500000U) - while (pos < end) { - uint8_t cnt = 8; - uint8_t data = *pos; - while (cnt > 0) { - __asm__ volatile( /* CPU Cycles | 1 | 0 */ - "st %a[port], %[on]" "\n\t" /* 2 | x | x */ - "sbrc %[data], 7" "\n\t" /* 1 | x | x */ - "rjmp .+0" "\n\t" /* 2 | x | - */ - "sbrc %[data], 7" "\n\t" /* 1 | x | x */ - "rjmp .+0" "\n\t" /* 2 | x | - */ - "sbrc %[data], 7" "\n\t" /* 1 | x | x */ - "nop" "\n\t" /* 1 | x | - */ - "st %a[port], %[off]" "\n\t" /* 2 | x | x */ - /* Total CPU cycles for high period: - * 10 cycles for bit 1, 5 cycles for bit 0 */ - : [data] "+r" (data) - : [on] "r" (mask_on), - [port] "e" (port_addr), - [off] "r" (mask_off) - ); - cnt--; - data <<= 1; - } - pos++; - } -#else -#error "No low level WS281x implementation for ATmega CPUs for your CPU clock" + break; #endif + case PORT_D: + while (pos < end) { + uint8_t cnt = 8; + uint8_t data = *pos; + while (cnt > 0) { + __asm__ volatile( /* Cycles | 1 | 0 */ + "out %[port], %[on]" "\n\t" /* 1 | x | x */ + "sbrs %[data], 7" "\n\t" /* 1 | x | x */ + "out %[port], %[off]" "\n\t" /* 1 | - | x */ + "rjmp .+0" "\n\t" /* 2 | x | x */ + "out %[port], %[off]" "\n\t" /* 1 | x | x */ + /* Total CPU cycles for high period: + * 5 cycles for bit 1, 2 cycles for bit 0 */ + : [data] "+r" (data) + : [port] "I" (_SFR_IO_ADDR(PORTD)), + [on] "r" (mask_on), + [off] "r" (mask_off) + ); + cnt--; + data <<= 1; + } + pos++; + } + break; + default: + assert(0); + break; + } + } + else if (coreclk() >= 15500000U && coreclk() <= 16500000U) { + while (pos < end) { + uint8_t cnt = 8; + uint8_t data = *pos; + while (cnt > 0) { + __asm__ volatile( /* CPU Cycles | 1 | 0 */ + "st %a[port], %[on]" "\n\t" /* 2 | x | x */ + "sbrc %[data], 7" "\n\t" /* 1 | x | x */ + "rjmp .+0" "\n\t" /* 2 | x | - */ + "sbrc %[data], 7" "\n\t" /* 1 | x | x */ + "rjmp .+0" "\n\t" /* 2 | x | - */ + "sbrc %[data], 7" "\n\t" /* 1 | x | x */ + "nop" "\n\t" /* 1 | x | - */ + "st %a[port], %[off]" "\n\t" /* 2 | x | x */ + /* Total CPU cycles for high period: + * 10 cycles for bit 1, 5 cycles for bit 0 */ + : [data] "+r" (data) + : [on] "r" (mask_on), + [port] "e" (port_addr), + [off] "r" (mask_off) + ); + cnt--; + data <<= 1; + } + pos++; + } + } + else { + /* No low level WS281x implementation for ATmega CPUs for the used CPU + * clock. Abusing the linker to give a compile time error instead of + * emitting a broken firmware */ + extern void clock_frequency_is_not_supported_by_ws281x_driver(void); + clock_frequency_is_not_supported_by_ws281x_driver(); + } } int ws281x_init(ws281x_t *dev, const ws281x_params_t *params) From 91927c988b2d665c99d1958cad35458675f514c5 Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:55:56 +0100 Subject: [PATCH 06/10] examples/blinky: use coreclk() instead of CLOCK_CORECLOCK --- examples/blinky/Makefile | 6 ------ examples/blinky/main.c | 4 +++- 2 files changed, 3 insertions(+), 7 deletions(-) diff --git a/examples/blinky/Makefile b/examples/blinky/Makefile index 904e9ead10..2d58be2d6b 100644 --- a/examples/blinky/Makefile +++ b/examples/blinky/Makefile @@ -19,9 +19,3 @@ QUIET ?= 1 FEATURES_OPTIONAL += periph_timer include $(RIOTBASE)/Makefile.include - -ifeq (native,$(BOARD)) - # For the native board, CLOCK_CORECLOCK is undefined. But we need - # a valid number to get the example compiled - CFLAGS += -DCLOCK_CORECLOCK=1000000000 -endif diff --git a/examples/blinky/main.c b/examples/blinky/main.c index 038cd27d6d..ac90ccbc15 100644 --- a/examples/blinky/main.c +++ b/examples/blinky/main.c @@ -20,6 +20,7 @@ #include +#include "clk.h" #include "board.h" #include "periph_conf.h" #include "timex.h" @@ -41,7 +42,8 @@ static void delay(void) * compilers would detect that the loop is only wasting CPU cycles and * optimize it out - but here the wasting of CPU cycles is desired. */ - for (volatile uint32_t i = 0; i < CLOCK_CORECLOCK / 20; i++) { } + uint32_t loops = coreclk() / 20; + for (volatile uint32_t i = 0; i < loops; i++) { } } } From 084e33d504e5ceb61fd4c5cf7133739c8cd1c4a5 Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Sun, 5 Dec 2021 11:56:04 +0100 Subject: [PATCH 07/10] pkg/openwsn: use coreclk() instead of CLOCK_CORECLOCK --- pkg/openwsn/contrib/leds.c | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/pkg/openwsn/contrib/leds.c b/pkg/openwsn/contrib/leds.c index b064978ade..2edd8bf7eb 100644 --- a/pkg/openwsn/contrib/leds.c +++ b/pkg/openwsn/contrib/leds.c @@ -23,6 +23,7 @@ #include "leds.h" #include "openwsn_leds.h" +#include "clk.h" #include "board.h" #include "periph/gpio.h" @@ -86,7 +87,8 @@ static void _blink_checked(gpio_t pin) /* toggle for ~10s if ztimer is used */ for (uint8_t i = 0; i < 100; i++) { _toggle_checked(pin); - for (uint32_t i = 0; i < (CLOCK_CORECLOCK / 50); i++) { + uint32_t runs = coreclk() / 50; + for (uint32_t i = 0; i < runs; i++) { /* Make sure for loop is not optimized out */ __asm__ (""); } From 2e74e5e3b9ffe0d325b43520d54f900db1951dfc Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Mon, 6 Dec 2021 10:58:34 +0100 Subject: [PATCH 08/10] cpu/esp32: include sdk_conf.h in periph_cpu.h This allows access to CLOCK_CORECLOCK from periph_conf.h which is used by clk.h --- cpu/esp32/include/periph_cpu.h | 1 + 1 file changed, 1 insertion(+) diff --git a/cpu/esp32/include/periph_cpu.h b/cpu/esp32/include/periph_cpu.h index a7f3773125..1a3fd36898 100644 --- a/cpu/esp32/include/periph_cpu.h +++ b/cpu/esp32/include/periph_cpu.h @@ -20,6 +20,7 @@ #define PERIPH_CPU_H #include +#include "sdk_conf.h" #ifdef __cplusplus extern "C" { From c9207e81a3bfb7c2c733136a647dfd5ba7fe304c Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Mon, 6 Dec 2021 12:00:42 +0100 Subject: [PATCH 09/10] cpu/esp8266: include cpu_conf.h in periph_cpu.h --- cpu/esp8266/include/periph_cpu.h | 1 + 1 file changed, 1 insertion(+) diff --git a/cpu/esp8266/include/periph_cpu.h b/cpu/esp8266/include/periph_cpu.h index 01836bee74..63905996e1 100644 --- a/cpu/esp8266/include/periph_cpu.h +++ b/cpu/esp8266/include/periph_cpu.h @@ -23,6 +23,7 @@ #include #include "eagle_soc.h" +#include "cpu_conf.h" #ifdef __cplusplus extern "C" { From d755c3ae4a65cb88ec9e9a92f2fdca0ba0a818f7 Mon Sep 17 00:00:00 2001 From: Alexandre Abadie Date: Mon, 6 Dec 2021 12:03:15 +0100 Subject: [PATCH 10/10] boards/qn9080dk: move CLOCK_CORECLOCK definition to periph_conf.h --- boards/qn9080dk/include/board.h | 10 ---------- boards/qn9080dk/include/periph_conf.h | 11 +++++++++++ 2 files changed, 11 insertions(+), 10 deletions(-) diff --git a/boards/qn9080dk/include/board.h b/boards/qn9080dk/include/board.h index 734b1e7d31..814914674c 100644 --- a/boards/qn9080dk/include/board.h +++ b/boards/qn9080dk/include/board.h @@ -28,16 +28,6 @@ extern "C" { #endif -/** - * @name Clock configuration - * @{ - */ -#ifndef CLOCK_CORECLOCK -/* Using 32MHz internal oscillator as default clock source */ -#define CLOCK_CORECLOCK (32000000ul) -#endif -/** @} */ - /** * @name LED configuration * @{ diff --git a/boards/qn9080dk/include/periph_conf.h b/boards/qn9080dk/include/periph_conf.h index d80c4e7bbe..a36ca750b8 100644 --- a/boards/qn9080dk/include/periph_conf.h +++ b/boards/qn9080dk/include/periph_conf.h @@ -21,6 +21,7 @@ #include +#include "macros/units.h" #include "cpu.h" #include "periph_cpu.h" @@ -28,6 +29,16 @@ extern "C" { #endif +/** + * @name Clock configuration + * @{ + */ +#ifndef CLOCK_CORECLOCK +/* Using 32MHz internal oscillator as default clock source */ +#define CLOCK_CORECLOCK MHZ(32) /**< System core clock frequency in Hz */ +#endif +/** @} */ + /** * @name ADC configuration *