 3f5dc47296
			
		
	
	
		3f5dc47296
		
			
		
	
	
	
	
		
			
			* Use polled waiting on platforms that support it Due to context switching overhead waiting a very short amount of time on a sleeping thread is often not accurate and in fact not usable for timing critical usage i.e. in a driver. Thus we use polled waiting for ranges in the us range on platforms that support it instead. The fallback is the thread sleeping mechanism. This includes: * ARM platforms with CYCCNT register (ARMv7, ARMv8) this is incremented at CPU clock frequency * GD32VF103 RISC-V port with CSR_MCYCLE register this is incremented at CPU clock frequency * RP2040 ARMv6 port which uses the integrated timer peripheral which is incremented with a fixed 1MHz frequency * Use wait_us() instead of chSysPolledDelayX ...as it is powered by busy waiting now. * Add chibios waiting methods test bench
		
			
				
	
	
		
			56 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			56 lines
		
	
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // Copyright 2022 Stefan Kerkmann
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "quantum.h"
 | |
| #include "hal.h"
 | |
| #include "bootloader.h"
 | |
| #include "pico/bootrom.h"
 | |
| 
 | |
| #if !defined(RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED)
 | |
| #    define RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED_MASK 0U
 | |
| #else
 | |
| #    define RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED_MASK (1U << RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED)
 | |
| #endif
 | |
| 
 | |
| __attribute__((weak)) void mcu_reset(void) {
 | |
|     NVIC_SystemReset();
 | |
| }
 | |
| void bootloader_jump(void) {
 | |
|     reset_usb_boot(RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED_MASK, 0U);
 | |
| }
 | |
| 
 | |
| void enter_bootloader_mode_if_requested(void) {}
 | |
| 
 | |
| #if defined(RP2040_BOOTLOADER_DOUBLE_TAP_RESET)
 | |
| #    if !defined(RP2040_BOOTLOADER_DOUBLE_TAP_RESET_TIMEOUT)
 | |
| #        define RP2040_BOOTLOADER_DOUBLE_TAP_RESET_TIMEOUT 200U
 | |
| #    endif
 | |
| 
 | |
| // Needs to be located in a RAM section that is never initialized on boot to
 | |
| // preserve its value on reset
 | |
| static volatile uint32_t __attribute__((section(".ram0.bootloader_magic"))) magic_location;
 | |
| const uint32_t magic_token = 0xCAFEB0BA;
 | |
| 
 | |
| // We can not use the __early_init / enter_bootloader_mode_if_requested hook as
 | |
| // we depend on an already initialized system with usable memory regions and
 | |
| // populated function pointer tables to the optimized math functions in the
 | |
| // bootrom. This function is called just prior to main.
 | |
| void __late_init(void) {
 | |
|     // All clocks have to be enabled before jumping to the bootloader function,
 | |
|     // otherwise the bootrom will be stuck infinitely.
 | |
|     clocks_init();
 | |
| 
 | |
|     if (magic_location != magic_token) {
 | |
|         magic_location = magic_token;
 | |
|         // ChibiOS is not initialized at this point, so sleeping is only
 | |
|         // possible via busy waiting.
 | |
|         wait_us(RP2040_BOOTLOADER_DOUBLE_TAP_RESET_TIMEOUT * 1000U);
 | |
|         magic_location = 0;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     magic_location = 0;
 | |
|     reset_usb_boot(RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED_MASK, 0U);
 | |
| }
 | |
| 
 | |
| #endif
 |