2019-11-08 10:17:08 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 Foundries.io Ltd
|
|
|
|
* Copyright (c) 2019 STMicroelectronics
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2023-08-28 13:15:43 +02:00
|
|
|
#include <zephyr/init.h>
|
2019-11-08 10:17:08 +01:00
|
|
|
#include <soc.h>
|
2020-11-20 18:41:15 +01:00
|
|
|
#include <stm32_ll_lptim.h>
|
2020-12-01 16:14:28 +01:00
|
|
|
#include <stm32_ll_bus.h>
|
|
|
|
#include <stm32_ll_rcc.h>
|
|
|
|
#include <stm32_ll_pwr.h>
|
2021-01-05 21:21:16 +01:00
|
|
|
#include <stm32_ll_system.h>
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/drivers/clock_control.h>
|
|
|
|
#include <zephyr/drivers/clock_control/stm32_clock_control.h>
|
|
|
|
#include <zephyr/drivers/timer/system_timer.h>
|
|
|
|
#include <zephyr/sys_clock.h>
|
2022-10-04 15:33:53 +02:00
|
|
|
#include <zephyr/irq.h>
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/spinlock.h>
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
#define DT_DRV_COMPAT st_stm32_lptim
|
|
|
|
|
|
|
|
#if DT_NUM_INST_STATUS_OKAY(DT_DRV_COMPAT) > 1
|
|
|
|
#error Only one LPTIM instance should be enabled
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define LPTIM (LPTIM_TypeDef *) DT_INST_REG_ADDR(0)
|
|
|
|
|
2022-08-05 15:59:38 +02:00
|
|
|
#if DT_INST_NUM_CLOCKS(0) == 1
|
2022-08-22 16:50:07 +02:00
|
|
|
#warning Kconfig for LPTIM source clock (LSI/LSE) is deprecated, use device tree.
|
2022-08-05 15:59:38 +02:00
|
|
|
static const struct stm32_pclken lptim_clk[] = {
|
|
|
|
STM32_CLOCK_INFO(0, DT_DRV_INST(0)),
|
|
|
|
/* Use Kconfig to configure source clocks fields */
|
|
|
|
/* Fortunately, values are consistent across enabled series */
|
|
|
|
#ifdef CONFIG_STM32_LPTIM_CLOCK_LSI
|
|
|
|
{.bus = STM32_SRC_LSI, .enr = LPTIM1_SEL(1)}
|
|
|
|
#else
|
|
|
|
{.bus = STM32_SRC_LSE, .enr = LPTIM1_SEL(3)}
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
#else
|
2022-08-05 14:08:47 +02:00
|
|
|
static const struct stm32_pclken lptim_clk[] = STM32_DT_INST_CLOCKS(0);
|
2022-08-05 15:59:38 +02:00
|
|
|
#endif
|
|
|
|
|
2022-08-17 15:37:22 +02:00
|
|
|
static const struct device *const clk_ctrl = DEVICE_DT_GET(STM32_CLOCK_CONTROL_NODE);
|
2022-08-05 14:08:47 +02:00
|
|
|
|
2019-11-08 10:17:08 +01:00
|
|
|
/*
|
|
|
|
* Assumptions and limitations:
|
|
|
|
*
|
2022-08-05 11:34:49 +02:00
|
|
|
* - system clock based on an LPTIM instance, clocked by LSI or LSE
|
2022-06-21 15:42:56 +02:00
|
|
|
* - prescaler is set to a 2^value from 1 (division of the LPTIM source clock by 1)
|
|
|
|
* to 128 (division of the LPTIM source clock by 128)
|
2022-08-05 11:34:49 +02:00
|
|
|
* - using LPTIM AutoReload capability to trig the IRQ (timeout irq)
|
2019-11-08 10:17:08 +01:00
|
|
|
* - when timeout irq occurs the counter is already reset
|
2022-08-05 15:59:38 +02:00
|
|
|
* - the maximum timeout duration is reached with the lptim_time_base value
|
2022-06-21 15:42:56 +02:00
|
|
|
* - with prescaler of 1, the max timeout (LPTIM_TIMEBASE) is 2 seconds:
|
|
|
|
* 0xFFFF / (LSE freq (32768Hz) / 1)
|
|
|
|
* - with prescaler of 128, the max timeout (LPTIM_TIMEBASE) is 256 seconds:
|
|
|
|
* 0xFFFF / (LSE freq (32768Hz) / 128)
|
2019-11-08 10:17:08 +01:00
|
|
|
*/
|
|
|
|
|
2022-05-11 17:29:32 +02:00
|
|
|
static uint32_t lptim_clock_freq = KHZ(32);
|
2022-08-05 15:59:38 +02:00
|
|
|
static int32_t lptim_time_base;
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2022-05-11 17:29:32 +02:00
|
|
|
/* The prescaler given by the DTS and to apply to the lptim_clock_freq */
|
2022-06-21 15:42:56 +02:00
|
|
|
#define LPTIM_CLOCK_RATIO DT_PROP(DT_DRV_INST(0), st_prescaler)
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2022-06-21 15:42:56 +02:00
|
|
|
/* Minimum nb of clock cycles to have to set autoreload register correctly */
|
2020-06-03 14:07:26 +02:00
|
|
|
#define LPTIM_GUARD_VALUE 2
|
|
|
|
|
2019-11-08 10:17:08 +01:00
|
|
|
/* A 32bit value cannot exceed 0xFFFFFFFF/LPTIM_TIMEBASE counting cycles.
|
|
|
|
* This is for example about of 65000 x 2000ms when clocked by LSI
|
|
|
|
*/
|
2020-05-27 18:26:57 +02:00
|
|
|
static uint32_t accumulated_lptim_cnt;
|
2022-06-23 11:37:48 +02:00
|
|
|
/* Next autoreload value to set */
|
|
|
|
static uint32_t autoreload_next;
|
|
|
|
/* Indicate if the autoreload register is ready for a write */
|
|
|
|
static bool autoreload_ready = true;
|
2019-11-08 10:17:08 +01:00
|
|
|
|
|
|
|
static struct k_spinlock lock;
|
|
|
|
|
2022-09-13 10:45:28 +02:00
|
|
|
/* For tick accuracy, a specific tick to freq ratio is expected */
|
|
|
|
/* This check assumes LSI@32KHz or LSE@32768Hz */
|
|
|
|
#if !defined(CONFIG_STM32_LPTIM_TICK_FREQ_RATIO_OVERRIDE)
|
|
|
|
#if (((DT_CLOCKS_CELL_BY_IDX(DT_DRV_INST(0), 1, bus) == STM32_SRC_LSI) && \
|
|
|
|
(CONFIG_SYS_CLOCK_TICKS_PER_SEC != 4000)) || \
|
|
|
|
((DT_CLOCKS_CELL_BY_IDX(DT_DRV_INST(0), 1, bus) == STM32_SRC_LSE) && \
|
|
|
|
(CONFIG_SYS_CLOCK_TICKS_PER_SEC != 4096)))
|
|
|
|
#warning Advised tick freq is 4096 for LSE / 4000 for LSI
|
|
|
|
#endif
|
|
|
|
#endif /* !CONFIG_STM32_LPTIM_TICK_FREQ_RATIO_OVERRIDE */
|
|
|
|
|
2023-01-13 12:07:43 +01:00
|
|
|
static inline bool arrm_state_get(void)
|
|
|
|
{
|
|
|
|
return (LL_LPTIM_IsActiveFlag_ARRM(LPTIM) && LL_LPTIM_IsEnabledIT_ARRM(LPTIM));
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static void lptim_irq_handler(const struct device *unused)
|
2019-11-08 10:17:08 +01:00
|
|
|
{
|
|
|
|
|
|
|
|
ARG_UNUSED(unused);
|
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
uint32_t autoreload = LL_LPTIM_GetAutoReload(LPTIM);
|
2022-06-23 11:37:48 +02:00
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
if ((LL_LPTIM_IsActiveFlag_ARROK(LPTIM) != 0)
|
|
|
|
&& LL_LPTIM_IsEnabledIT_ARROK(LPTIM) != 0) {
|
|
|
|
LL_LPTIM_ClearFlag_ARROK(LPTIM);
|
2022-06-23 11:37:48 +02:00
|
|
|
if ((autoreload_next > 0) && (autoreload_next != autoreload)) {
|
|
|
|
/* the new autoreload value change, we set it */
|
|
|
|
autoreload_ready = false;
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_SetAutoReload(LPTIM, autoreload_next);
|
2022-06-23 11:37:48 +02:00
|
|
|
} else {
|
|
|
|
autoreload_ready = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-13 12:07:43 +01:00
|
|
|
if (arrm_state_get()) {
|
2019-11-08 10:17:08 +01:00
|
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
|
|
|
2021-02-25 21:33:15 +01:00
|
|
|
/* do not change ARR yet, sys_clock_announce will do */
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_ClearFLAG_ARRM(LPTIM);
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2020-05-08 11:39:46 +02:00
|
|
|
/* increase the total nb of autoreload count
|
2021-03-12 18:46:52 +01:00
|
|
|
* used in the sys_clock_cycle_get_32() function.
|
2019-11-08 10:17:08 +01:00
|
|
|
*/
|
2022-06-23 11:37:48 +02:00
|
|
|
autoreload++;
|
2020-05-08 11:39:46 +02:00
|
|
|
|
|
|
|
accumulated_lptim_cnt += autoreload;
|
2019-11-08 10:17:08 +01:00
|
|
|
|
|
|
|
k_spin_unlock(&lock, key);
|
|
|
|
|
|
|
|
/* announce the elapsed time in ms (count register is 16bit) */
|
2020-05-27 18:26:57 +02:00
|
|
|
uint32_t dticks = (autoreload
|
2019-11-08 10:17:08 +01:00
|
|
|
* CONFIG_SYS_CLOCK_TICKS_PER_SEC)
|
2022-08-05 15:59:38 +02:00
|
|
|
/ lptim_clock_freq;
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2021-02-25 21:33:15 +01:00
|
|
|
sys_clock_announce(IS_ENABLED(CONFIG_TICKLESS_KERNEL)
|
2020-05-19 08:52:33 +02:00
|
|
|
? dticks : (dticks > 0));
|
2019-11-08 10:17:08 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-23 11:37:48 +02:00
|
|
|
static void lptim_set_autoreload(uint32_t arr)
|
|
|
|
{
|
|
|
|
/* Update autoreload register */
|
|
|
|
autoreload_next = arr;
|
|
|
|
|
|
|
|
if (!autoreload_ready)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* The ARR register ready, we could set it directly */
|
2022-08-05 11:34:49 +02:00
|
|
|
if ((arr > 0) && (arr != LL_LPTIM_GetAutoReload(LPTIM))) {
|
2022-06-23 11:37:48 +02:00
|
|
|
/* The new autoreload value change, we set it */
|
|
|
|
autoreload_ready = false;
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_ClearFlag_ARROK(LPTIM);
|
|
|
|
LL_LPTIM_SetAutoReload(LPTIM, arr);
|
2022-06-23 11:37:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-18 16:34:28 +02:00
|
|
|
static inline uint32_t z_clock_lptim_getcounter(void)
|
|
|
|
{
|
|
|
|
uint32_t lp_time;
|
|
|
|
uint32_t lp_time_prev_read;
|
|
|
|
|
|
|
|
/* It should be noted that to read reliably the content
|
|
|
|
* of the LPTIM_CNT register, two successive read accesses
|
|
|
|
* must be performed and compared
|
|
|
|
*/
|
2022-08-05 11:34:49 +02:00
|
|
|
lp_time = LL_LPTIM_GetCounter(LPTIM);
|
2020-06-18 16:34:28 +02:00
|
|
|
do {
|
|
|
|
lp_time_prev_read = lp_time;
|
2022-08-05 11:34:49 +02:00
|
|
|
lp_time = LL_LPTIM_GetCounter(LPTIM);
|
2020-06-18 16:34:28 +02:00
|
|
|
} while (lp_time != lp_time_prev_read);
|
|
|
|
return lp_time;
|
|
|
|
}
|
|
|
|
|
2021-02-25 21:33:15 +01:00
|
|
|
void sys_clock_set_timeout(int32_t ticks, bool idle)
|
2019-11-08 10:17:08 +01:00
|
|
|
{
|
2022-08-05 11:34:49 +02:00
|
|
|
/* new LPTIM AutoReload value to set (aligned on Kernel ticks) */
|
2020-05-27 18:26:57 +02:00
|
|
|
uint32_t next_arr = 0;
|
2023-07-18 08:58:54 +02:00
|
|
|
int err;
|
2019-11-08 10:17:08 +01:00
|
|
|
|
|
|
|
ARG_UNUSED(idle);
|
|
|
|
|
2020-05-11 13:12:52 +02:00
|
|
|
if (!IS_ENABLED(CONFIG_TICKLESS_KERNEL)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-02 00:41:27 +02:00
|
|
|
if (ticks == K_TICKS_FOREVER) {
|
2023-03-28 08:24:07 +02:00
|
|
|
clock_control_off(clk_ctrl, (clock_control_subsys_t) &lptim_clk[0]);
|
2019-11-08 10:17:08 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-04 16:57:12 +02:00
|
|
|
/* if LPTIM clock was previously stopped, it must now be restored */
|
2023-07-18 08:58:54 +02:00
|
|
|
err = clock_control_on(clk_ctrl, (clock_control_subsys_t) &lptim_clk[0]);
|
2020-06-04 16:57:12 +02:00
|
|
|
|
2023-07-18 08:58:54 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-08 10:17:08 +01:00
|
|
|
/* passing ticks==1 means "announce the next tick",
|
|
|
|
* ticks value of zero (or even negative) is legal and
|
|
|
|
* treated identically: it simply indicates the kernel would like the
|
|
|
|
* next tick announcement as soon as possible.
|
|
|
|
*/
|
2022-08-05 15:59:38 +02:00
|
|
|
ticks = CLAMP(ticks - 1, 1, lptim_time_base);
|
2019-11-08 10:17:08 +01:00
|
|
|
|
|
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
|
|
|
|
|
|
/* read current counter value (cannot exceed 16bit) */
|
|
|
|
|
2020-06-18 16:34:28 +02:00
|
|
|
uint32_t lp_time = z_clock_lptim_getcounter();
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
uint32_t autoreload = LL_LPTIM_GetAutoReload(LPTIM);
|
2020-06-03 14:07:26 +02:00
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
if (LL_LPTIM_IsActiveFlag_ARRM(LPTIM)
|
2020-06-03 14:07:26 +02:00
|
|
|
|| ((autoreload - lp_time) < LPTIM_GUARD_VALUE)) {
|
|
|
|
/* interrupt happens or happens soon.
|
|
|
|
* It's impossible to set autoreload value.
|
|
|
|
*/
|
|
|
|
k_spin_unlock(&lock, key);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-08 10:17:08 +01:00
|
|
|
/* calculate the next arr value (cannot exceed 16bit)
|
|
|
|
* adjust the next ARR match value to align on Ticks
|
|
|
|
* from the current counter value to first next Tick
|
|
|
|
*/
|
|
|
|
next_arr = (((lp_time * CONFIG_SYS_CLOCK_TICKS_PER_SEC)
|
2022-08-05 15:59:38 +02:00
|
|
|
/ lptim_clock_freq) + 1) * lptim_clock_freq
|
2019-11-08 10:17:08 +01:00
|
|
|
/ (CONFIG_SYS_CLOCK_TICKS_PER_SEC);
|
2022-08-05 15:59:38 +02:00
|
|
|
next_arr = next_arr + ((uint32_t)(ticks) * lptim_clock_freq)
|
2022-05-11 17:29:32 +02:00
|
|
|
/ CONFIG_SYS_CLOCK_TICKS_PER_SEC;
|
|
|
|
/* if the lptim_clock_freq < one ticks/sec, then next_arr must be > 0 */
|
2019-11-08 10:17:08 +01:00
|
|
|
|
|
|
|
/* maximise to TIMEBASE */
|
2022-08-05 15:59:38 +02:00
|
|
|
if (next_arr > lptim_time_base) {
|
|
|
|
next_arr = lptim_time_base;
|
2019-11-08 10:17:08 +01:00
|
|
|
}
|
2020-06-03 14:07:26 +02:00
|
|
|
/* The new autoreload value must be LPTIM_GUARD_VALUE clock cycles
|
|
|
|
* after current lptim to make sure we don't miss
|
|
|
|
* an autoreload interrupt
|
2020-05-11 15:53:19 +02:00
|
|
|
*/
|
2020-06-03 14:07:26 +02:00
|
|
|
else if (next_arr < (lp_time + LPTIM_GUARD_VALUE)) {
|
|
|
|
next_arr = lp_time + LPTIM_GUARD_VALUE;
|
2020-05-11 15:53:19 +02:00
|
|
|
}
|
2022-05-11 17:29:32 +02:00
|
|
|
/* with slow lptim_clock_freq, LPTIM_GUARD_VALUE of 1 is enough */
|
|
|
|
next_arr = next_arr - 1;
|
2020-06-03 13:55:57 +02:00
|
|
|
|
2022-06-23 11:37:48 +02:00
|
|
|
/* Update autoreload register */
|
|
|
|
lptim_set_autoreload(next_arr);
|
2019-11-08 10:17:08 +01:00
|
|
|
|
|
|
|
k_spin_unlock(&lock, key);
|
|
|
|
}
|
|
|
|
|
2023-01-13 12:07:43 +01:00
|
|
|
static uint32_t sys_clock_lp_time_get(void)
|
|
|
|
{
|
|
|
|
uint32_t lp_time;
|
|
|
|
|
|
|
|
do {
|
|
|
|
/* In case of counter roll-over, add the autoreload value,
|
|
|
|
* because the irq has not yet been handled
|
|
|
|
*/
|
|
|
|
if (arrm_state_get()) {
|
|
|
|
lp_time = LL_LPTIM_GetAutoReload(LPTIM) + 1;
|
|
|
|
lp_time += z_clock_lptim_getcounter();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
lp_time = z_clock_lptim_getcounter();
|
|
|
|
|
|
|
|
/* Check if the flag ARRM wasn't be set during the process */
|
|
|
|
} while (arrm_state_get());
|
|
|
|
|
|
|
|
return lp_time;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-02-25 21:33:15 +01:00
|
|
|
uint32_t sys_clock_elapsed(void)
|
2019-11-08 10:17:08 +01:00
|
|
|
{
|
|
|
|
if (!IS_ENABLED(CONFIG_TICKLESS_KERNEL)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
|
|
|
2023-01-13 12:07:43 +01:00
|
|
|
uint32_t lp_time = sys_clock_lp_time_get();
|
2020-05-11 15:53:19 +02:00
|
|
|
|
2019-11-08 10:17:08 +01:00
|
|
|
k_spin_unlock(&lock, key);
|
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
/* gives the value of LPTIM counter (ms)
|
2019-11-08 10:17:08 +01:00
|
|
|
* since the previous 'announce'
|
|
|
|
*/
|
2022-08-05 15:59:38 +02:00
|
|
|
uint64_t ret = ((uint64_t)lp_time * CONFIG_SYS_CLOCK_TICKS_PER_SEC) / lptim_clock_freq;
|
2019-11-08 10:17:08 +01:00
|
|
|
|
2020-05-27 18:26:57 +02:00
|
|
|
return (uint32_t)(ret);
|
2019-11-08 10:17:08 +01:00
|
|
|
}
|
|
|
|
|
2021-03-12 18:46:52 +01:00
|
|
|
uint32_t sys_clock_cycle_get_32(void)
|
2019-11-08 10:17:08 +01:00
|
|
|
{
|
|
|
|
/* just gives the accumulated count in a number of hw cycles */
|
|
|
|
|
|
|
|
k_spinlock_key_t key = k_spin_lock(&lock);
|
|
|
|
|
2023-01-13 12:07:43 +01:00
|
|
|
uint32_t lp_time = sys_clock_lp_time_get();
|
2020-05-11 15:53:19 +02:00
|
|
|
|
2019-11-08 10:17:08 +01:00
|
|
|
lp_time += accumulated_lptim_cnt;
|
|
|
|
|
2020-05-11 15:53:19 +02:00
|
|
|
/* convert lptim count in a nb of hw cycles with precision */
|
2022-08-05 15:59:38 +02:00
|
|
|
uint64_t ret = ((uint64_t)lp_time * sys_clock_hw_cycles_per_sec()) / lptim_clock_freq;
|
2020-05-11 15:53:19 +02:00
|
|
|
|
2019-11-08 10:17:08 +01:00
|
|
|
k_spin_unlock(&lock, key);
|
|
|
|
|
|
|
|
/* convert in hw cycles (keeping 32bit value) */
|
2020-05-27 18:26:57 +02:00
|
|
|
return (uint32_t)(ret);
|
2019-11-08 10:17:08 +01:00
|
|
|
}
|
2021-11-04 12:51:39 +01:00
|
|
|
|
2023-02-06 15:48:11 +01:00
|
|
|
/* Wait for the IER register of the stm32U5 ready, after any bit write operation */
|
|
|
|
void stm32_lptim_wait_ready(void)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_SOC_SERIES_STM32U5X
|
|
|
|
while (LL_LPTIM_IsActiveFlag_DIEROK(LPTIM) == 0) {
|
|
|
|
}
|
|
|
|
LL_LPTIM_ClearFlag_DIEROK(LPTIM);
|
|
|
|
#else
|
|
|
|
/* Empty : not relevant */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
init: remove the need for a dummy device pointer in SYS_INIT functions
The init infrastructure, found in `init.h`, is currently used by:
- `SYS_INIT`: to call functions before `main`
- `DEVICE_*`: to initialize devices
They are all sorted according to an initialization level + a priority.
`SYS_INIT` calls are really orthogonal to devices, however, the required
function signature requires a `const struct device *dev` as a first
argument. The only reason for that is because the same init machinery is
used by devices, so we have something like:
```c
struct init_entry {
int (*init)(const struct device *dev);
/* only set by DEVICE_*, otherwise NULL */
const struct device *dev;
}
```
As a result, we end up with such weird/ugly pattern:
```c
static int my_init(const struct device *dev)
{
/* always NULL! add ARG_UNUSED to avoid compiler warning */
ARG_UNUSED(dev);
...
}
```
This is really a result of poor internals isolation. This patch proposes
a to make init entries more flexible so that they can accept sytem
initialization calls like this:
```c
static int my_init(void)
{
...
}
```
This is achieved using a union:
```c
union init_function {
/* for SYS_INIT, used when init_entry.dev == NULL */
int (*sys)(void);
/* for DEVICE*, used when init_entry.dev != NULL */
int (*dev)(const struct device *dev);
};
struct init_entry {
/* stores init function (either for SYS_INIT or DEVICE*)
union init_function init_fn;
/* stores device pointer for DEVICE*, NULL for SYS_INIT. Allows
* to know which union entry to call.
*/
const struct device *dev;
}
```
This solution **does not increase ROM usage**, and allows to offer clean
public APIs for both SYS_INIT and DEVICE*. Note that however, init
machinery keeps a coupling with devices.
**NOTE**: This is a breaking change! All `SYS_INIT` functions will need
to be converted to the new signature. See the script offered in the
following commit.
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
init: convert SYS_INIT functions to the new signature
Conversion scripted using scripts/utils/migrate_sys_init.py.
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
manifest: update projects for SYS_INIT changes
Update modules with updated SYS_INIT calls:
- hal_ti
- lvgl
- sof
- TraceRecorderSource
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
tests: devicetree: devices: adjust test
Adjust test according to the recently introduced SYS_INIT
infrastructure.
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
tests: kernel: threads: adjust SYS_INIT call
Adjust to the new signature: int (*init_fn)(void);
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
2022-10-19 09:33:44 +02:00
|
|
|
static int sys_clock_driver_init(void)
|
2021-11-04 12:51:39 +01:00
|
|
|
{
|
2022-06-21 15:42:56 +02:00
|
|
|
uint32_t count_per_tick;
|
2022-08-23 15:16:51 +02:00
|
|
|
int err;
|
|
|
|
|
2022-08-05 14:08:47 +02:00
|
|
|
if (!device_is_ready(clk_ctrl)) {
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable LPTIM bus clock */
|
2023-03-28 08:24:07 +02:00
|
|
|
err = clock_control_on(clk_ctrl, (clock_control_subsys_t) &lptim_clk[0]);
|
2022-08-23 15:16:51 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
2022-08-05 14:08:47 +02:00
|
|
|
|
2021-11-04 12:51:39 +01:00
|
|
|
#if defined(LL_APB1_GRP1_PERIPH_LPTIM1)
|
|
|
|
LL_APB1_GRP1_ReleaseReset(LL_APB1_GRP1_PERIPH_LPTIM1);
|
|
|
|
#elif defined(LL_APB3_GRP1_PERIPH_LPTIM1)
|
|
|
|
LL_SRDAMR_GRP1_EnableAutonomousClock(LL_SRDAMR_GRP1_PERIPH_LPTIM1AMEN);
|
2023-03-01 16:14:55 +01:00
|
|
|
#elif defined(LL_APB7_GRP1_PERIPH_LPTIM1)
|
|
|
|
LL_APB7_GRP1_ReleaseReset(LL_APB7_GRP1_PERIPH_LPTIM1);
|
2021-11-04 12:51:39 +01:00
|
|
|
#endif
|
|
|
|
|
2022-08-05 15:59:38 +02:00
|
|
|
/* Enable LPTIM clock source */
|
2022-08-23 15:16:51 +02:00
|
|
|
err = clock_control_configure(clk_ctrl,
|
2023-03-28 08:24:07 +02:00
|
|
|
(clock_control_subsys_t) &lptim_clk[1],
|
2022-08-23 15:16:51 +02:00
|
|
|
NULL);
|
|
|
|
if (err < 0) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
2021-11-04 12:51:39 +01:00
|
|
|
|
2022-08-05 15:59:38 +02:00
|
|
|
/* Get LPTIM clock freq */
|
2023-03-28 08:24:07 +02:00
|
|
|
err = clock_control_get_rate(clk_ctrl, (clock_control_subsys_t) &lptim_clk[1],
|
2022-08-05 15:59:38 +02:00
|
|
|
&lptim_clock_freq);
|
2021-11-04 12:51:39 +01:00
|
|
|
|
2022-08-23 15:16:51 +02:00
|
|
|
if (err < 0) {
|
|
|
|
return -EIO;
|
|
|
|
}
|
2022-08-23 14:15:34 +02:00
|
|
|
#if defined(CONFIG_SOC_SERIES_STM32L0X)
|
|
|
|
/* Driver only supports freqs up to 32768Hz. On L0, LSI freq is 37KHz,
|
|
|
|
* which will overflow the LPTIM counter.
|
|
|
|
* Previous LPTIM configuration using device tree was doing forcing this
|
|
|
|
* with a Kconfig default. Impact is that time is 1.13 faster than reality.
|
|
|
|
* Following lines reproduce this behavior in order not to change behavior.
|
|
|
|
* This issue will be fixed by implementation LPTIM prescaler support.
|
|
|
|
*/
|
|
|
|
if (lptim_clk[1].bus == STM32_SRC_LSI) {
|
|
|
|
lptim_clock_freq = KHZ(32);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-12-14 11:08:21 +01:00
|
|
|
/* Set LPTIM time base based on clock source freq */
|
2022-08-05 15:59:38 +02:00
|
|
|
if (lptim_clock_freq == KHZ(32)) {
|
|
|
|
lptim_time_base = 0xF9FF;
|
|
|
|
} else if (lptim_clock_freq == 32768) {
|
|
|
|
lptim_time_base = 0xFFFF;
|
|
|
|
} else {
|
|
|
|
return -EIO;
|
2021-11-04 12:51:39 +01:00
|
|
|
}
|
|
|
|
|
2022-12-14 11:08:21 +01:00
|
|
|
if (IS_ENABLED(DT_PROP(DT_DRV_INST(0), st_static_prescaler))) {
|
|
|
|
/*
|
|
|
|
* LPTIM of the stm32, like stm32U5, which has a clock source x2.
|
|
|
|
* A full 16bit LPTIM counter is counting 4s at 2 * 1/32768 (with LSE)
|
|
|
|
* Time base = (4s * freq) - 1
|
|
|
|
*/
|
|
|
|
lptim_clock_freq = lptim_clock_freq / 2;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Else, a full 16bit LPTIM counter is counting 2s at 1/32768 (with LSE)
|
|
|
|
* Time base = (2s * freq) - 1
|
|
|
|
*/
|
|
|
|
|
2022-06-21 15:42:56 +02:00
|
|
|
/* Actual lptim clock freq when the clock source is reduced by the prescaler */
|
|
|
|
lptim_clock_freq = lptim_clock_freq / LPTIM_CLOCK_RATIO;
|
|
|
|
|
2021-11-04 12:51:39 +01:00
|
|
|
/* Clear the event flag and possible pending interrupt */
|
2022-08-05 11:34:49 +02:00
|
|
|
IRQ_CONNECT(DT_INST_IRQN(0),
|
|
|
|
DT_INST_IRQ(0, priority),
|
2021-11-04 12:51:39 +01:00
|
|
|
lptim_irq_handler, 0, 0);
|
2022-08-05 11:34:49 +02:00
|
|
|
irq_enable(DT_INST_IRQN(0));
|
2021-11-04 12:51:39 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_SOC_SERIES_STM32WLX
|
2022-08-05 11:34:49 +02:00
|
|
|
/* Enable the LPTIM wakeup EXTI line */
|
2021-11-04 12:51:39 +01:00
|
|
|
LL_EXTI_EnableIT_0_31(LL_EXTI_LINE_29);
|
|
|
|
#endif
|
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
/* configure the LPTIM counter */
|
|
|
|
LL_LPTIM_SetClockSource(LPTIM, LL_LPTIM_CLK_SOURCE_INTERNAL);
|
2022-06-21 15:42:56 +02:00
|
|
|
/* the LPTIM clock freq is affected by the prescaler */
|
|
|
|
LL_LPTIM_SetPrescaler(LPTIM, (__CLZ(__RBIT(LPTIM_CLOCK_RATIO)) << LPTIM_CFGR_PRESC_Pos));
|
2023-03-01 16:14:55 +01:00
|
|
|
#if defined(CONFIG_SOC_SERIES_STM32U5X) || \
|
|
|
|
defined(CONFIG_SOC_SERIES_STM32WBAX)
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_OC_SetPolarity(LPTIM, LL_LPTIM_CHANNEL_CH1,
|
2021-11-04 12:51:39 +01:00
|
|
|
LL_LPTIM_OUTPUT_POLARITY_REGULAR);
|
|
|
|
#else
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_SetPolarity(LPTIM, LL_LPTIM_OUTPUT_POLARITY_REGULAR);
|
2021-11-04 12:51:39 +01:00
|
|
|
#endif
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_SetUpdateMode(LPTIM, LL_LPTIM_UPDATE_MODE_IMMEDIATE);
|
|
|
|
LL_LPTIM_SetCounterMode(LPTIM, LL_LPTIM_COUNTER_MODE_INTERNAL);
|
|
|
|
LL_LPTIM_DisableTimeout(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
/* counting start is initiated by software */
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_TrigSw(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
|
2023-03-01 16:14:55 +01:00
|
|
|
#if defined(CONFIG_SOC_SERIES_STM32U5X) || \
|
|
|
|
defined(CONFIG_SOC_SERIES_STM32WBAX)
|
2022-08-05 11:34:49 +02:00
|
|
|
/* Enable the LPTIM before proceeding with configuration */
|
|
|
|
LL_LPTIM_Enable(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_DisableIT_CC1(LPTIM);
|
2023-02-06 15:48:11 +01:00
|
|
|
stm32_lptim_wait_ready();
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_ClearFLAG_CC1(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
#else
|
2022-08-05 11:34:49 +02:00
|
|
|
/* LPTIM interrupt set-up before enabling */
|
2021-11-04 12:51:39 +01:00
|
|
|
/* no Compare match Interrupt */
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_DisableIT_CMPM(LPTIM);
|
|
|
|
LL_LPTIM_ClearFLAG_CMPM(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Autoreload match Interrupt */
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_EnableIT_ARRM(LPTIM);
|
2023-02-06 15:48:11 +01:00
|
|
|
stm32_lptim_wait_ready();
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_ClearFLAG_ARRM(LPTIM);
|
2023-02-06 15:48:11 +01:00
|
|
|
|
2021-11-04 12:51:39 +01:00
|
|
|
/* ARROK bit validates the write operation to ARR register */
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_EnableIT_ARROK(LPTIM);
|
2023-02-06 15:48:11 +01:00
|
|
|
stm32_lptim_wait_ready();
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_ClearFlag_ARROK(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
|
|
|
|
accumulated_lptim_cnt = 0;
|
|
|
|
|
2023-03-01 16:14:55 +01:00
|
|
|
#if !defined(CONFIG_SOC_SERIES_STM32U5X) && \
|
|
|
|
!defined(CONFIG_SOC_SERIES_STM32WBAX)
|
2022-08-05 11:34:49 +02:00
|
|
|
/* Enable the LPTIM counter */
|
|
|
|
LL_LPTIM_Enable(LPTIM);
|
2021-11-04 12:51:39 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Set the Autoreload value once the timer is enabled */
|
|
|
|
if (IS_ENABLED(CONFIG_TICKLESS_KERNEL)) {
|
2022-08-05 11:34:49 +02:00
|
|
|
/* LPTIM is triggered on a LPTIM_TIMEBASE period */
|
2022-08-05 15:59:38 +02:00
|
|
|
lptim_set_autoreload(lptim_time_base);
|
2021-11-04 12:51:39 +01:00
|
|
|
} else {
|
2022-06-21 15:42:56 +02:00
|
|
|
/* nb of LPTIM counter unit per kernel tick (depends on lptim clock prescaler) */
|
|
|
|
count_per_tick = (lptim_clock_freq / CONFIG_SYS_CLOCK_TICKS_PER_SEC);
|
2022-08-05 11:34:49 +02:00
|
|
|
/* LPTIM is triggered on a Tick period */
|
2022-06-21 15:42:56 +02:00
|
|
|
lptim_set_autoreload(count_per_tick - 1);
|
2021-11-04 12:51:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Start the LPTIM counter in continuous mode */
|
2022-08-05 11:34:49 +02:00
|
|
|
LL_LPTIM_StartCounter(LPTIM, LL_LPTIM_OPERATING_MODE_CONTINUOUS);
|
2021-11-04 12:51:39 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_DEBUG
|
2022-08-05 11:34:49 +02:00
|
|
|
/* stop LPTIM during DEBUG */
|
2021-11-04 12:51:39 +01:00
|
|
|
#if defined(LL_DBGMCU_APB1_GRP1_LPTIM1_STOP)
|
|
|
|
LL_DBGMCU_APB1_GRP1_FreezePeriph(LL_DBGMCU_APB1_GRP1_LPTIM1_STOP);
|
|
|
|
#elif defined(LL_DBGMCU_APB3_GRP1_LPTIM1_STOP)
|
|
|
|
LL_DBGMCU_APB3_GRP1_FreezePeriph(LL_DBGMCU_APB3_GRP1_LPTIM1_STOP);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SYS_INIT(sys_clock_driver_init, PRE_KERNEL_2,
|
|
|
|
CONFIG_SYSTEM_CLOCK_INIT_PRIORITY);
|