2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
27 #include "build/atomic.h"
29 #include "drivers/io.h"
30 #include "drivers/light_led.h"
31 #include "drivers/nvic.h"
32 #include "drivers/resource.h"
33 #include "drivers/sound_beeper.h"
38 #if defined(STM32F3) || defined(STM32F4) || defined(STM32F7) || defined(STM32H7)
39 // See "RM CoreSight Architecture Specification"
40 // B2.3.10 "LSR and LAR, Software Lock Status Register and Software Lock Access Register"
41 // "E1.2.11 LAR, Lock Access Register"
43 #define DWT_LAR_UNLOCK_VALUE 0xC5ACCE55
47 // cycles per microsecond
48 static uint32_t usTicks
= 0;
49 // current uptime for 1kHz systick timer. will rollover after 49 days. hopefully we won't care.
50 static volatile uint32_t sysTickUptime
= 0;
51 static volatile uint32_t sysTickValStamp
= 0;
52 // cached value of RCC->CSR
53 uint32_t cachedRccCsrValue
;
54 static uint32_t cpuClockFrequency
= 0;
56 void cycleCounterInit(void)
58 #if defined(USE_HAL_DRIVER)
59 cpuClockFrequency
= HAL_RCC_GetSysClockFreq();
61 RCC_ClocksTypeDef clocks
;
62 RCC_GetClocksFreq(&clocks
);
63 cpuClockFrequency
= clocks
.SYSCLK_Frequency
;
65 usTicks
= cpuClockFrequency
/ 1000000;
67 CoreDebug
->DEMCR
|= CoreDebug_DEMCR_TRCENA_Msk
;
69 #if defined(DWT_LAR_UNLOCK_VALUE)
71 ITM
->LAR
= DWT_LAR_UNLOCK_VALUE
;
72 #elif defined(STM32F7)
73 DWT
->LAR
= DWT_LAR_UNLOCK_VALUE
;
74 #elif defined(STM32F3) || defined(STM32F4)
75 // Note: DWT_Type does not contain LAR member.
77 __O
uint32_t *DWTLAR
= (uint32_t *)(DWT_BASE
+ 0x0FB0);
78 *(DWTLAR
) = DWT_LAR_UNLOCK_VALUE
;
83 DWT
->CTRL
|= DWT_CTRL_CYCCNTENA_Msk
;
88 static volatile int sysTickPending
= 0;
90 void SysTick_Handler(void)
92 ATOMIC_BLOCK(NVIC_PRIO_MAX
) {
94 sysTickValStamp
= SysTick
->VAL
;
96 (void)(SysTick
->CTRL
);
99 // used by the HAL for some timekeeping and timeouts, should always be 1ms
104 // Return system uptime in microseconds (rollover in 70minutes)
106 uint32_t microsISR(void)
108 register uint32_t ms
, pending
, cycle_cnt
;
110 ATOMIC_BLOCK(NVIC_PRIO_MAX
) {
111 cycle_cnt
= SysTick
->VAL
;
113 if (SysTick
->CTRL
& SysTick_CTRL_COUNTFLAG_Msk
) {
115 // Record it for multiple calls within the same rollover period
116 // (Will be cleared when serviced).
117 // Note that multiple rollovers are not considered.
121 // Read VAL again to ensure the value is read after the rollover.
123 cycle_cnt
= SysTick
->VAL
;
127 pending
= sysTickPending
;
130 return ((ms
+ pending
) * 1000) + (usTicks
* 1000 - cycle_cnt
) / usTicks
;
133 uint32_t micros(void)
135 register uint32_t ms
, cycle_cnt
;
137 // Call microsISR() in interrupt and elevated (non-zero) BASEPRI context
139 if ((SCB
->ICSR
& SCB_ICSR_VECTACTIVE_Msk
) || (__get_BASEPRI())) {
145 cycle_cnt
= SysTick
->VAL
;
146 } while (ms
!= sysTickUptime
|| cycle_cnt
> sysTickValStamp
);
148 return (ms
* 1000) + (usTicks
* 1000 - cycle_cnt
) / usTicks
;
151 uint32_t getCycleCounter(void)
156 int32_t clockCyclesToMicros(int32_t clockCycles
)
158 return clockCycles
/ usTicks
;
161 // Note that this conversion is signed as this is used for periods rather than absolute timestamps
162 int32_t clockCyclesTo10thMicros(int32_t clockCycles
)
164 return 10 * clockCycles
/ (int32_t)usTicks
;
167 uint32_t clockMicrosToCycles(uint32_t micros
)
169 return micros
* usTicks
;
172 // Return system uptime in milliseconds (rollover in 49 days)
173 uint32_t millis(void)
175 return sysTickUptime
;
179 void delayMicroseconds(uint32_t us
)
181 uint32_t now
= micros();
182 while (micros() - now
< us
);
185 void delayMicroseconds(uint32_t us
)
187 uint32_t elapsed
= 0;
188 uint32_t lastCount
= SysTick
->VAL
;
191 register uint32_t current_count
= SysTick
->VAL
;
194 // measure the time elapsed since the last time we checked
195 elapsed
+= current_count
- lastCount
;
196 lastCount
= current_count
;
198 // convert to microseconds
199 elapsed_us
= elapsed
/ usTicks
;
200 if (elapsed_us
>= us
)
203 // reduce the delay by the elapsed time
206 // keep fractional microseconds for the next iteration
212 void delay(uint32_t ms
)
215 delayMicroseconds(1000);
218 static void indicate(uint8_t count
, uint16_t duration
)
238 void indicateFailure(failureMode_e mode
, int codeRepeatsRemaining
)
240 while (codeRepeatsRemaining
--) {
241 indicate(WARNING_FLASH_COUNT
, WARNING_FLASH_DURATION_MS
);
243 delay(WARNING_PAUSE_DURATION_MS
);
245 indicate(mode
+ 1, WARNING_CODE_DURATION_LONG_MS
);
251 void failureMode(failureMode_e mode
)
253 indicateFailure(mode
, 10);
258 systemResetToBootloader(BOOTLOADER_REQUEST_ROM
);
262 void initialiseMemorySections(void)
265 /* Load functions into ITCM RAM */
266 extern uint8_t tcm_code_start
;
267 extern uint8_t tcm_code_end
;
268 extern uint8_t tcm_code
;
269 memcpy(&tcm_code_start
, &tcm_code
, (size_t) (&tcm_code_end
- &tcm_code_start
));
273 /* Load functions into RAM */
274 extern uint8_t ccm_code_start
;
275 extern uint8_t ccm_code_end
;
276 extern uint8_t ccm_code
;
277 memcpy(&ccm_code_start
, &ccm_code
, (size_t) (&ccm_code_end
- &ccm_code_start
));
281 /* Load FAST_DATA variable initializers into DTCM RAM */
282 extern uint8_t _sfastram_data
;
283 extern uint8_t _efastram_data
;
284 extern uint8_t _sfastram_idata
;
285 memcpy(&_sfastram_data
, &_sfastram_idata
, (size_t) (&_efastram_data
- &_sfastram_data
));
290 void initialiseD2MemorySections(void)
292 /* Load DMA_DATA variable intializers into D2 RAM */
293 extern uint8_t _sdmaram_bss
;
294 extern uint8_t _edmaram_bss
;
295 extern uint8_t _sdmaram_data
;
296 extern uint8_t _edmaram_data
;
297 extern uint8_t _sdmaram_idata
;
298 bzero(&_sdmaram_bss
, (size_t) (&_edmaram_bss
- &_sdmaram_bss
));
299 memcpy(&_sdmaram_data
, &_sdmaram_idata
, (size_t) (&_edmaram_data
- &_sdmaram_data
));
303 static void unusedPinInit(IO_t io
)
305 if (IOGetOwner(io
) == OWNER_FREE
) {
306 IOConfigGPIO(io
, IOCFG_IPU
);
310 void unusedPinsInit(void)
312 IOTraversePins(unusedPinInit
);