From 57ca4e178996e3220c6e9b41a9090b693f0c3c00 Mon Sep 17 00:00:00 2001 From: hujun5 Date: Sat, 28 Dec 2024 11:33:42 +0800 Subject: [PATCH] up_rtc_gettime: add spinlock to protect up_rtc_gettime reason: We have removed the critical section protection for the up_rtc_gettime function in common code. Signed-off-by: hujun5 --- arch/arm/src/cxd56xx/cxd56_rtc.c | 21 +++++++++---- arch/arm/src/cxd56xx/cxd56_rtc.h | 1 + arch/arm/src/kinetis/kinetis_rtc.c | 13 +++++--- arch/arm/src/max326xx/max32660/max32660_rtc.c | 5 ++++ arch/arm/src/s32k1xx/s32k1xx_rtc.c | 15 +++++++--- arch/arm/src/sam34/sam_rtc.c | 12 ++++++-- arch/arm/src/stm32/stm32_eth.c | 16 ++++++++-- arch/arm/src/stm32/stm32_rtcounter.c | 23 +++++++------- arch/renesas/src/rx65n/rx65n_rtc.c | 30 ++++++++++++------- arch/x86_64/src/intel64/intel64_rtc.c | 9 ++++++ 10 files changed, 108 insertions(+), 37 deletions(-) diff --git a/arch/arm/src/cxd56xx/cxd56_rtc.c b/arch/arm/src/cxd56xx/cxd56_rtc.c index db33c24c32..2e6cf7c007 100644 --- a/arch/arm/src/cxd56xx/cxd56_rtc.c +++ b/arch/arm/src/cxd56xx/cxd56_rtc.c @@ -399,10 +399,14 @@ time_t up_rtc_time(void) #ifdef CONFIG_RTC_HIRES int up_rtc_gettime(struct timespec *tp) { + irqstate_t flags; uint64_t count; - count = cxd56_rtc_count(); + flags = spin_lock_irqsave(&g_rtc_lock); + + count = cxd56_rtc_count_nolock(); count += g_rtc_save->offset; + spin_unlock_irqrestore(&g_rtc_lock, flags); /* Then we can save the time in seconds and fractional seconds. */ @@ -477,21 +481,28 @@ int up_rtc_settime(const struct timespec *tp) * ****************************************************************************/ -uint64_t cxd56_rtc_count(void) +uint64_t cxd56_rtc_count_nolock(void) { uint64_t val; - irqstate_t flags; /* The pre register is latched with reading the post rtcounter register, * so these registers always have to been read in the below order, * 1st post -> 2nd pre, and should be operated in atomic. */ - flags = spin_lock_irqsave(&g_rtc_lock); - val = (uint64_t)getreg32(CXD56_RTC0_RTPOSTCNT) << 15; val |= getreg32(CXD56_RTC0_RTPRECNT); + return val; +} + +uint64_t cxd56_rtc_count(void) +{ + uint64_t val; + irqstate_t flags; + + flags = spin_lock_irqsave(&g_rtc_lock); + val = cxd56_rtc_count_nolock(); spin_unlock_irqrestore(&g_rtc_lock, flags); return val; diff --git a/arch/arm/src/cxd56xx/cxd56_rtc.h b/arch/arm/src/cxd56xx/cxd56_rtc.h index 33854e40b3..0860736fa0 100644 --- a/arch/arm/src/cxd56xx/cxd56_rtc.h +++ b/arch/arm/src/cxd56xx/cxd56_rtc.h @@ -92,6 +92,7 @@ extern "C" ****************************************************************************/ uint64_t cxd56_rtc_count(void); +uint64_t cxd56_rtc_count_nolock(void); /**************************************************************************** * Name: cxd56_rtc_almcount diff --git a/arch/arm/src/kinetis/kinetis_rtc.c b/arch/arm/src/kinetis/kinetis_rtc.c index bc9e53a699..7bb69d9331 100644 --- a/arch/arm/src/kinetis/kinetis_rtc.c +++ b/arch/arm/src/kinetis/kinetis_rtc.c @@ -28,6 +28,7 @@ #include #include #include +#include #include #include @@ -62,6 +63,10 @@ * Private Data ****************************************************************************/ +#ifdef CONFIG_RTC_HIRES +static spinlock_t g_rtc_lock = SP_UNLOCKED; +#endif + #ifdef CONFIG_RTC_ALARM static alarmcb_t g_alarmcb; static bool rtc_irq_state = false; @@ -436,7 +441,7 @@ int up_rtc_gettime(struct timespec *tp) * wrapped-around. */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); do { prescaler = getreg32(KINETIS_RTC_TPR); @@ -445,7 +450,7 @@ int up_rtc_gettime(struct timespec *tp) } while (prescaler > prescaler2); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); /* Build seconds + nanoseconds from seconds and prescaler register */ @@ -479,7 +484,7 @@ int up_rtc_settime(const struct timespec *tp) seconds = tp->tv_sec; prescaler = tp->tv_nsec / (1000000000 / CONFIG_RTC_FREQUENCY); - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); putreg32(0, KINETIS_RTC_SR); /* Disable counter */ @@ -488,7 +493,7 @@ int up_rtc_settime(const struct timespec *tp) putreg32(RTC_SR_TCE, KINETIS_RTC_SR); /* Re-enable counter */ - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return OK; } diff --git a/arch/arm/src/max326xx/max32660/max32660_rtc.c b/arch/arm/src/max326xx/max32660/max32660_rtc.c index db37e63430..e2b8e1fa0f 100644 --- a/arch/arm/src/max326xx/max32660/max32660_rtc.c +++ b/arch/arm/src/max326xx/max32660/max32660_rtc.c @@ -403,11 +403,14 @@ time_t up_rtc_time(void) #ifdef CONFIG_RTC_HIRES int up_rtc_gettime(struct timespec *tp) { + irqstate_t flags; uint64_t tmp; uint32_t sec; uint32_t ssec; uint32_t verify; + flags = spin_lock_irqsave(&g_rtc_lock); + /* Read the time handling rollover to full seconds */ do @@ -418,6 +421,8 @@ int up_rtc_gettime(struct timespec *tp) } while (verify != sec); + spin_unlock_irqrestore(&g_rtc_lock, flags); + /* Format as a tm */ tmp = ((uint64_t)ssec * NSEC_PER_SEC) / 256; diff --git a/arch/arm/src/s32k1xx/s32k1xx_rtc.c b/arch/arm/src/s32k1xx/s32k1xx_rtc.c index 136a8edd4d..5da7913efa 100644 --- a/arch/arm/src/s32k1xx/s32k1xx_rtc.c +++ b/arch/arm/src/s32k1xx/s32k1xx_rtc.c @@ -37,6 +37,7 @@ #include #include #include +#include #include @@ -60,6 +61,12 @@ volatile bool g_rtc_enabled = false; +/**************************************************************************** + * Private Data + ****************************************************************************/ + +static spinlock_t g_rtc_lock = SP_UNLOCKED; + /**************************************************************************** * Private Functions ****************************************************************************/ @@ -247,7 +254,7 @@ int up_rtc_gettime(struct timespec *tp) * wrapped-around. */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); do { prescaler = getreg32(S32K1XX_RTC_TPR); @@ -256,7 +263,7 @@ int up_rtc_gettime(struct timespec *tp) } while (prescaler > prescaler2); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); /* Build seconds + nanoseconds from seconds and prescaler register */ @@ -296,7 +303,7 @@ int up_rtc_settime(const struct timespec *ts) prescaler = 0; #endif - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); s32k1xx_rtc_disable(); @@ -305,7 +312,7 @@ int up_rtc_settime(const struct timespec *ts) s32k1xx_rtc_enable(); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return OK; } diff --git a/arch/arm/src/sam34/sam_rtc.c b/arch/arm/src/sam34/sam_rtc.c index d6d60f0b1a..a375830e31 100644 --- a/arch/arm/src/sam34/sam_rtc.c +++ b/arch/arm/src/sam34/sam_rtc.c @@ -34,6 +34,7 @@ #include #include #include +#include #include @@ -72,6 +73,8 @@ * Private Data ****************************************************************************/ +static spinlock_t g_rtc_lock = SP_UNLOCKED; + /* Callback to use when the alarm expires */ #ifdef CONFIG_RTC_ALARM @@ -654,7 +657,8 @@ int sam_rtc_setalarm(const struct timespec *tp, alarmcb_t callback) /* Is there already something waiting on the ALARM? */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); + sched_lock(); if (g_alarmcb == NULL) { /* No.. Save the callback function pointer */ @@ -732,7 +736,8 @@ int sam_rtc_setalarm(const struct timespec *tp, alarmcb_t callback) ret = OK; } - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); + sched_unlock(); return ret; } #endif @@ -759,11 +764,13 @@ int up_rtc_gettime(struct timespec *tp) { /* This is a hack to emulate a high resolution rtc using the rtt */ + irqstate_t flags; uint32_t rtc_cal; uint32_t rtc_tim; uint32_t rtt_val; struct tm t; + flags = spin_lock_irqsave(&g_rtc_lock); do { rtc_cal = getreg32(SAM_RTC_CALR); @@ -774,6 +781,7 @@ int up_rtc_gettime(struct timespec *tp) rtc_tim != getreg32(SAM_RTC_TIMR) || rtt_val != getreg32(SAM_RTT_VR)); + spin_unlock_irqrestore(&g_rtc_lock, flags); t.tm_sec = rtc_bcd2bin((rtc_tim & RTC_TIMR_SEC_MASK) >> RTC_TIMR_SEC_SHIFT); t.tm_min = rtc_bcd2bin((rtc_tim & RTC_TIMR_MIN_MASK) >> diff --git a/arch/arm/src/stm32/stm32_eth.c b/arch/arm/src/stm32/stm32_eth.c index 4aa2c5d1f1..591b84cb40 100644 --- a/arch/arm/src/stm32/stm32_eth.c +++ b/arch/arm/src/stm32/stm32_eth.c @@ -48,6 +48,7 @@ #include #include #include +#include #if defined(CONFIG_NET_PKT) # include @@ -651,6 +652,7 @@ static uint8_t g_alloc[STM32_ETH_NFREEBUFFERS * static struct stm32_ethmac_s g_stm32ethmac[STM32_NETHERNET]; #ifdef CONFIG_STM32_ETH_PTP_RTC_HIRES +static spinlock_t g_rtc_lock = SP_UNLOCKED; volatile bool g_rtc_enabled; static struct timespec g_stm32_eth_ptp_basetime; #endif @@ -3778,10 +3780,10 @@ static void stm32_eth_ptp_convert_rxtime(struct stm32_ethmac_s *priv) /* Sample PTP and CLOCK_REALTIME close to each other */ - flags = enter_critical_section(); clock_gettime(CLOCK_REALTIME, &realtime); + flags = spin_lock_irqsave(&g_rtc_lock); ptptime = stm32_eth_ptp_gettime(); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); /* Compute how much time has elapsed since packet reception * and add that to current time. @@ -4308,7 +4310,10 @@ int up_rtc_initialize(void) int up_rtc_gettime(struct timespec *tp) { + irqstate_t flags; uint64_t timestamp; + + flags = spin_lock_irqsave(&g_rtc_lock); timestamp = stm32_eth_ptp_gettime(); if (timestamp == 0) @@ -4317,12 +4322,14 @@ int up_rtc_gettime(struct timespec *tp) * Normally we shouldn't end up here because g_rtc_enabled is false. */ + spin_unlock_irqrestore(&g_rtc_lock, flags); DEBUGASSERT(!g_rtc_enabled); return -EBUSY; } ptp_to_timespec(timestamp, tp); clock_timespec_add(tp, &g_stm32_eth_ptp_basetime, tp); + spin_unlock_irqrestore(&g_rtc_lock, flags); return OK; } @@ -4346,6 +4353,9 @@ int up_rtc_settime(const struct timespec *tp) { struct timespec ptptime; uint64_t timestamp; + irqstate_t flags; + + flags = spin_lock_irqsave(&g_rtc_lock); timestamp = stm32_eth_ptp_gettime(); if (timestamp == 0) @@ -4354,6 +4364,7 @@ int up_rtc_settime(const struct timespec *tp) * Normally we shouldn't end up here because g_rtc_enabled is false. */ + spin_unlock_irqrestore(&g_rtc_lock, flags); DEBUGASSERT(!g_rtc_enabled); return -EBUSY; } @@ -4365,6 +4376,7 @@ int up_rtc_settime(const struct timespec *tp) ptp_to_timespec(timestamp, &ptptime); clock_timespec_subtract(tp, &ptptime, &g_stm32_eth_ptp_basetime); + spin_unlock_irqrestore(&g_rtc_lock, flags); return OK; } diff --git a/arch/arm/src/stm32/stm32_rtcounter.c b/arch/arm/src/stm32/stm32_rtcounter.c index 441c86c892..2d97a6f33f 100644 --- a/arch/arm/src/stm32/stm32_rtcounter.c +++ b/arch/arm/src/stm32/stm32_rtcounter.c @@ -46,6 +46,7 @@ #include #include #include +#include #include #include @@ -151,6 +152,8 @@ struct rtc_regvals_s * Private Data ****************************************************************************/ +static spinlock_t g_rtc_lock = SP_UNLOCKED; + /* Callback to use when the alarm expires */ #ifdef CONFIG_RTC_ALARM @@ -524,7 +527,7 @@ time_t up_rtc_time(void) * interrupts will prevent suspensions and interruptions: */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* And the following loop will handle any clock rollover events that may * happen between samples. Most of the time (like 99.9%), the following @@ -546,7 +549,7 @@ time_t up_rtc_time(void) */ while (cntl < tmp); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); /* Okay.. the samples should be as close together in time as possible and * we can be assured that no clock rollover occurred between the samples. @@ -593,7 +596,7 @@ int up_rtc_gettime(struct timespec *tp) * interrupts will prevent suspensions and interruptions: */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* And the following loop will handle any clock rollover events that may * happen between samples. Most of the time (like 99.9%), the following @@ -616,7 +619,7 @@ int up_rtc_gettime(struct timespec *tp) */ while (cntl < tmp); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); /* Okay.. the samples should be as close together in time as possible and * we can be assured that no clock rollover occurred between the samples. @@ -664,7 +667,7 @@ int up_rtc_settime(const struct timespec *tp) /* Enable write access to the backup domain */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); stm32_pwr_enablebkp(true); /* Then write the broken out values to the RTC counter and BKP overflow @@ -682,7 +685,7 @@ int up_rtc_settime(const struct timespec *tp) #endif stm32_pwr_enablebkp(false); - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return OK; } @@ -709,7 +712,7 @@ int stm32_rtc_setalarm(const struct timespec *tp, alarmcb_t callback) uint16_t cr; int ret = -EBUSY; - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* Is there already something waiting on the ALARM? */ @@ -743,7 +746,7 @@ int stm32_rtc_setalarm(const struct timespec *tp, alarmcb_t callback) ret = OK; } - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return ret; } @@ -769,7 +772,7 @@ int stm32_rtc_cancelalarm(void) irqstate_t flags; int ret = -ENODATA; - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); if (g_alarmcb != NULL) { @@ -789,7 +792,7 @@ int stm32_rtc_cancelalarm(void) ret = OK; } - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return ret; } diff --git a/arch/renesas/src/rx65n/rx65n_rtc.c b/arch/renesas/src/rx65n/rx65n_rtc.c index 5585fa3d60..2e5f6b4497 100644 --- a/arch/renesas/src/rx65n/rx65n_rtc.c +++ b/arch/renesas/src/rx65n/rx65n_rtc.c @@ -35,6 +35,7 @@ #include #include #include +#include #include #include #include "renesas_internal.h" @@ -103,6 +104,8 @@ struct prd_cbinfo_s * Private Data ****************************************************************************/ +static spinlock_t g_rtc_lock = SP_UNLOCKED; + #ifdef CONFIG_RTC_ALARM /* Callback to use when an EXTI is activated */ @@ -461,6 +464,7 @@ int up_rtc_initialize(void) #if defined(CONFIG_RTC_HIRES) int up_rtc_gettime(struct timespec *tp) { + irqstate_t flags; uint8_t weekcnt; uint8_t daycnt; uint8_t monthcnt; @@ -476,6 +480,9 @@ int up_rtc_gettime(struct timespec *tp) uint8_t regval; struct tm t; + flags = spin_lock_irqsave(&g_rtc_lock); + sched_lock(); + if (RTC.RCR2.BIT.START == 0) { RTC.RCR2.BIT.START = 1; @@ -541,6 +548,9 @@ int up_rtc_gettime(struct timespec *tp) UNUSED(hrcnt); UNUSED(mincnt); UNUSED(seccnt); + + spin_unlock_irqrestore(&g_rtc_lock, flags); + sched_unlock(); return OK; } #endif @@ -916,7 +926,7 @@ int rx65n_rtc_setalarm(struct alm_setalarm_s *alminfo) /* Is there already something waiting on the ALARM? */ - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* Save the callback info */ @@ -1051,8 +1061,6 @@ int rx65n_rtc_setalarm(struct alm_setalarm_s *alminfo) dummy_word = RTC.RYRAR.WORD; } - rtc_dumpregs("New alarm setting"); - /* Enable RTC ALARM interrupt */ RTC.RCR1.BIT.AIE = 1U; @@ -1068,7 +1076,9 @@ int rx65n_rtc_setalarm(struct alm_setalarm_s *alminfo) /* Set Priority of ALM interrupt */ IPR(RTC, ALM) = _0F_RTC_PRIORITY_LEVEL15; - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); + + rtc_dumpregs("New alarm setting"); UNUSED(dummy_byte); UNUSED(dummy_word); return OK; @@ -1082,7 +1092,7 @@ int rx65n_rtc_setperiodic(const struct timespec *period, irqstate_t flags; volatile uint8_t regval; uint8_t prd; - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* No.. Save the callback function pointer */ @@ -1116,7 +1126,7 @@ int rx65n_rtc_setperiodic(const struct timespec *period, /* Set PRD priority level */ IPR(RTC, PRD) = _0F_RTC_PRIORITY_LEVEL15; - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return OK; } #endif @@ -1125,7 +1135,7 @@ int rx65n_rtc_setperiodic(const struct timespec *period, void rx65n_rtc_set_carry(carrycb_t callback) { irqstate_t flags; - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* No.. Save the callback function pointer */ @@ -1148,7 +1158,7 @@ void rx65n_rtc_set_carry(carrycb_t callback) IPR(PERIB, INTB176) = 15; RTC.RCR1.BIT.CIE = 1U; - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); } #endif @@ -1158,7 +1168,7 @@ int rx65n_rtc_cancelalarm(void) irqstate_t flags; int ret = -ENODATA; - flags = enter_critical_section(); + flags = spin_lock_irqsave(&g_rtc_lock); /* Cancel the global callback function */ @@ -1176,7 +1186,7 @@ int rx65n_rtc_cancelalarm(void) rx65n_putreg(0x0, RX65N_RTC_RYRAR); ret = OK; - leave_critical_section(flags); + spin_unlock_irqrestore(&g_rtc_lock, flags); return ret; } diff --git a/arch/x86_64/src/intel64/intel64_rtc.c b/arch/x86_64/src/intel64/intel64_rtc.c index 6ed1da0080..86a0810359 100644 --- a/arch/x86_64/src/intel64/intel64_rtc.c +++ b/arch/x86_64/src/intel64/intel64_rtc.c @@ -28,6 +28,7 @@ #include #include #include +#include #include #include @@ -64,6 +65,10 @@ * Private Data ****************************************************************************/ +#ifdef CONFIG_RTC_HIRES +static spinlock_t g_rtc_lock = SP_UNLOCKED; +#endif + /**************************************************************************** * Public Data ****************************************************************************/ @@ -85,14 +90,18 @@ static unsigned long rtc_last; static unsigned long rtc_read(void) { uint64_t tmr = rdtscp(); + irqstate_t flags; + flags = spin_lock_irqsave(&g_rtc_lock); if (tmr < rtc_last) { tmr += (0xffffffffffffffffull - rtc_last); } rtc_last = tmr; + spin_unlock_irqrestore(&g_rtc_lock, flags); tmr = (tmr / (g_x86_64_timer_freq / 1000000ul)) * 1000l; + return tmr; }