51 #include "stm32l4xx.h"
149 #if ( !defined(configUSE_TICKLESS_IDLE) || configUSE_TICKLESS_IDLE != 2 )
150 #warning Please edit FreeRTOSConfig.h to define configUSE_TICKLESS_IDLE as 2 *or* exclude this file.
153 #ifdef xPortSysTickHandler
154 #warning Please edit FreeRTOSConfig.h to eliminate the preprocessor definition for xPortSysTickHandler.
169 #ifndef configTICK_INTERRUPT_PRIORITY
170 #define configTICK_INTERRUPT_PRIORITY configLIBRARY_LOWEST_INTERRUPT_PRIORITY
177 #ifdef configTICK_USES_LSI
178 #define LPTIMSEL_Val 1
179 #define IS_REF_CLOCK_READY() (RCC->CSR & RCC_CSR_LSIRDY)
181 #define LPTIMSEL_Val 3
182 #define IS_REF_CLOCK_READY() (RCC->BDCR & RCC_BDCR_LSERDY)
190 #ifndef configLPTIM_REF_CLOCK_HZ
191 #define configLPTIM_REF_CLOCK_HZ 32768UL
203 #ifndef configLPTIM_ENABLE_PRECISION
204 #define configLPTIM_ENABLE_PRECISION 1
216 #ifndef traceTICKS_DROPPED
217 #define traceTICKS_DROPPED(x)
220 #define LPTIM_CLOCK_HZ ( configLPTIM_REF_CLOCK_HZ )
222 static TickType_t xMaximumSuppressedTicks;
226 static uint32_t ulTimerCountsForOneTick;
230 #if ( configLPTIM_ENABLE_PRECISION != 0 )
232 static int lSubcountErrorPerTick;
241 static volatile int lRunningSubcountError;
250 #if ( configTICK_ENABLE_UWTICK_PRECISION != 0 )
252 static uint32_t uwTickSuppressedSystem;
257 static volatile uint16_t usIdealCmp;
261 static volatile uint8_t isCmpWriteInProgress;
265 static volatile uint8_t isTickNowSuppressed;
278 #ifdef configLPTIM_SRC_LPTIM1
280 #define LPTIM_IRQn LPTIM1_IRQn
281 #define LPTIM_IRQHandler LPTIM1_IRQHandler
283 #ifdef configLPTIM_SRC_LPTIM2
285 #define LPTIM_IRQn LPTIM2_IRQn
286 #define LPTIM_IRQHandler LPTIM2_IRQHandler
292 #define LPTIM_IRQn LPTIM1_IRQn
293 #define LPTIM_IRQHandler LPTIM1_IRQHandler
294 #define configLPTIM_SRC_LPTIM1
299 void vPortSetupTimerInterrupt(
void )
308 #ifdef configLPTIM_SRC_LPTIM1
309 RCC->APB1ENR1 |= RCC_APB1ENR1_LPTIM1EN;
310 MODIFY_REG(RCC->CCIPR, RCC_CCIPR_LPTIM1SEL, LPTIMSEL_Val << RCC_CCIPR_LPTIM1SEL_Pos);
311 DBGMCU->APB1FZR1 |= DBGMCU_APB1FZR1_DBG_LPTIM1_STOP;
312 RCC->APB1RSTR1 |= RCC_APB1RSTR1_LPTIM1RST;
313 RCC->APB1RSTR1 &= ~RCC_APB1RSTR1_LPTIM1RST;
315 #ifdef configLPTIM_SRC_LPTIM2
316 RCC->APB1ENR2 |= RCC_APB1ENR2_LPTIM2EN;
317 MODIFY_REG(RCC->CCIPR, RCC_CCIPR_LPTIM2SEL, LPTIMSEL_Val << RCC_CCIPR_LPTIM2SEL_Pos);
318 DBGMCU->APB1FZR2 |= DBGMCU_APB1FZR2_DBG_LPTIM2_STOP;
319 RCC->APB1RSTR2 |= RCC_APB1RSTR2_LPTIM2RST;
320 RCC->APB1RSTR2 &= ~RCC_APB1RSTR2_LPTIM2RST;
324 #define EXTI_IMR1_LPTIM1 (1UL << 29)
325 #define EXTI_IMR1_LPTIM2 (1UL << 30)
326 #define EXTI_IMR1_LPTIM3 (1UL << 31)
331 #ifdef configLPTIM_SRC_LPTIM1
332 EXTI->IMR1 |= EXTI_IMR1_LPTIM1;
334 #ifdef configLPTIM_SRC_LPTIM2
335 EXTI->IMR1 |= EXTI_IMR1_LPTIM2;
337 #ifdef configLPTIM_SRC_LPTIM3
338 EXTI->IMR1 |= EXTI_IMR1_LPTIM3;
346 configASSERT(IS_REF_CLOCK_READY());
350 ulTimerCountsForOneTick = ( LPTIM_CLOCK_HZ + ( configTICK_RATE_HZ / 2 ) ) / configTICK_RATE_HZ;
351 configASSERT( ulTimerCountsForOneTick >= 4UL );
359 #if ( configLPTIM_ENABLE_PRECISION == 0 )
361 xMaximumSuppressedTicks = 65536UL / ulTimerCountsForOneTick - 1 - 1;
365 xMaximumSuppressedTicks = 65536UL * configTICK_RATE_HZ / LPTIM_CLOCK_HZ - 1 - 1;
372 #if ( LPTIM_CLOCK_HZ % configTICK_RATE_HZ_LITERAL < configTICK_RATE_HZ_LITERAL/2 )
373 #define IS_SUBCOUNT_EPT_POSITIVE 1
375 #define IS_SUBCOUNT_EPT_POSITIVE 0
378 lSubcountErrorPerTick = LPTIM_CLOCK_HZ - ( ulTimerCountsForOneTick * configTICK_RATE_HZ );
379 configASSERT( lSubcountErrorPerTick != configTICK_RATE_HZ / 2 );
386 LPTIM->IER = LPTIM_IER_CMPMIE | LPTIM_IER_CMPOKIE;
387 LPTIM->CFGR = (0 << LPTIM_CFGR_PRESC_Pos);
388 LPTIM->CR = LPTIM_CR_ENABLE;
390 LPTIM->CMP = ulTimerCountsForOneTick;
391 isCmpWriteInProgress = pdTRUE;
392 usIdealCmp = ulTimerCountsForOneTick;
393 #if ( configLPTIM_ENABLE_PRECISION != 0 )
395 lRunningSubcountError = lSubcountErrorPerTick;
398 LPTIM->CR |= LPTIM_CR_CNTSTRT;
403 NVIC_SetPriority( LPTIM_IRQn, configTICK_INTERRUPT_PRIORITY );
404 NVIC_EnableIRQ( LPTIM_IRQn );
412 void vPortSuppressTicksAndSleep( TickType_t xExpectedIdleTime )
416 if (xExpectedIdleTime > xMaximumSuppressedTicks)
418 xExpectedIdleTime = xMaximumSuppressedTicks;
425 #if ( configLPTIM_ENABLE_PRECISION != 0 )
426 uint32_t ulExtraCounts = (xExpectedIdleTime - 1UL) * LPTIM_CLOCK_HZ / configTICK_RATE_HZ;
427 int32_t lExtraError = (xExpectedIdleTime - 1UL) * LPTIM_CLOCK_HZ % configTICK_RATE_HZ;
429 uint32_t ulExtraCounts = (xExpectedIdleTime - 1UL) * ulTimerCountsForOneTick;
444 if (eTaskConfirmSleepModeStatus() == eAbortSleep)
450 #if ( configLPTIM_ENABLE_PRECISION != 0 )
456 lRunningSubcountError += lExtraError;
457 if (lRunningSubcountError > (
int)(configTICK_RATE_HZ/2))
460 lRunningSubcountError -= configTICK_RATE_HZ;
475 isTickNowSuppressed = pdTRUE;
477 #if ( configTICK_ENABLE_UWTICK_PRECISION != 0 )
479 uwTickSuppressedSystem = uwTick;
487 usIdealCmp += ulExtraCounts;
488 if (!isCmpWriteInProgress)
490 isCmpWriteInProgress = pdTRUE;
491 LPTIM->CMP = usIdealCmp == 0xFFFF ? 0 : usIdealCmp;
493 uint32_t ulExpectedEndCmp = usIdealCmp;
520 TickType_t xModifiableIdleTime = xExpectedIdleTime;
522 if (xModifiableIdleTime > 0)
544 }
while (usIdealCmp == ulExpectedEndCmp && eTaskConfirmSleepModeStatus() != eAbortSleep);
559 TickType_t xCompleteTickPeriods = xExpectedIdleTime - (TickType_t) 1;
567 portDISABLE_INTERRUPTS();
568 if (usIdealCmp == ulExpectedEndCmp)
577 uint32_t ulCurrCount;
578 do ulCurrCount = LPTIM->CNT;
while (ulCurrCount != LPTIM->CNT);
598 uint32_t ulFullCountsLeft = (uint16_t)(usIdealCmp - ulCurrCount - 1UL);
599 #if ( configLPTIM_ENABLE_PRECISION == 0 )
600 if (ulFullCountsLeft < xMaximumSuppressedTicks * ulTimerCountsForOneTick)
602 if (ulFullCountsLeft <= xMaximumSuppressedTicks * LPTIM_CLOCK_HZ / configTICK_RATE_HZ)
611 TickType_t xFullTicksLeft;
612 #if ( configLPTIM_ENABLE_PRECISION != 0 )
614 xFullTicksLeft = ulFullCountsLeft * configTICK_RATE_HZ / LPTIM_CLOCK_HZ;
615 if (xFullTicksLeft == xExpectedIdleTime)
623 xFullTicksLeft = xExpectedIdleTime - (TickType_t)1;
628 xFullTicksLeft = ulFullCountsLeft / ulTimerCountsForOneTick;
637 configASSERT( xFullTicksLeft < xExpectedIdleTime );
639 if (xFullTicksLeft != 0)
641 xCompleteTickPeriods -= xFullTicksLeft;
648 uint32_t ulFullTicksLeftAsCounts;
649 #if ( configLPTIM_ENABLE_PRECISION != 0 )
651 ulFullTicksLeftAsCounts = xFullTicksLeft * LPTIM_CLOCK_HZ / configTICK_RATE_HZ;
652 lExtraError = xFullTicksLeft * LPTIM_CLOCK_HZ % configTICK_RATE_HZ;
653 lRunningSubcountError -= lExtraError;
654 if (lRunningSubcountError < -(
int)(configTICK_RATE_HZ/2))
656 ulFullTicksLeftAsCounts++;
657 lRunningSubcountError += configTICK_RATE_HZ;
662 ulFullTicksLeftAsCounts = xFullTicksLeft * ulTimerCountsForOneTick;
666 usIdealCmp -= ulFullTicksLeftAsCounts;
667 if (!isCmpWriteInProgress)
669 isCmpWriteInProgress = pdTRUE;
670 LPTIM->CMP = usIdealCmp == 0xFFFF ? 0 : usIdealCmp;
681 isTickNowSuppressed = pdFALSE;
682 portENABLE_INTERRUPTS();
689 vTaskStepTick( xCompleteTickPeriods );
691 #if ( configTICK_ENABLE_UWTICK_PRECISION != 0 )
694 if(xCompleteTickPeriods > (uwTick - uwTickSuppressedSystem)) {
695 uwTick = uwTick + ((uint32_t)(xCompleteTickPeriods - (uwTick - uwTickSuppressedSystem)));
703 void LPTIM_IRQHandler(
void )
726 if (LPTIM->ISR & LPTIM_ISR_CMPM)
728 LPTIM->ICR = LPTIM_ICR_CMPMCF;
738 uint32_t ulCountValue;
739 do ulCountValue = LPTIM->CNT;
while (ulCountValue != LPTIM->CNT);
740 uint32_t ulCountsLate = (uint16_t)(ulCountValue - usIdealCmp);
753 if (ulCountsLate >= ulTimerCountsForOneTick &&
754 ulCountsLate < 65536UL - 1 - ulTimerCountsForOneTick &&
755 !isTickNowSuppressed)
760 traceTICKS_DROPPED( ulCountsLate / ulTimerCountsForOneTick );
761 usIdealCmp = ulCountValue;
776 if (ulCountsLate < ulTimerCountsForOneTick)
780 uint32_t ulNumCounts = ulTimerCountsForOneTick;
781 #if ( configLPTIM_ENABLE_PRECISION != 0 )
783 lRunningSubcountError += lSubcountErrorPerTick;
784 #if ( IS_SUBCOUNT_EPT_POSITIVE )
786 if (lRunningSubcountError >= (
int)(configTICK_RATE_HZ/2))
789 lRunningSubcountError -= configTICK_RATE_HZ;
794 if (lRunningSubcountError <= -(
int)(configTICK_RATE_HZ/2))
797 lRunningSubcountError += configTICK_RATE_HZ;
807 usIdealCmp += ulNumCounts;
808 if (!isCmpWriteInProgress)
810 LPTIM->CMP = usIdealCmp == 0xFFFF ? 0 : usIdealCmp;
811 isCmpWriteInProgress = pdTRUE;
818 portDISABLE_INTERRUPTS();
819 BaseType_t xWasHigherPriorityTaskWoken = xTaskIncrementTick();
820 portENABLE_INTERRUPTS();
822 portYIELD_FROM_ISR(xWasHigherPriorityTaskWoken);
830 if (LPTIM->ISR & LPTIM_ISR_CMPOK)
834 LPTIM->ICR = LPTIM_ICR_CMPOKCF;
840 if ((uint16_t)(LPTIM->CMP - usIdealCmp) > 1UL)
842 LPTIM->CMP = usIdealCmp == 0xFFFF ? 0 : usIdealCmp;
847 isCmpWriteInProgress = pdFALSE;