 d717396708
			
		
	
	
		d717396708
		
			
		
	
	
	
	
		
			
			* Disable RESET keycode because of naming conflicts * Add Pico SDK as submodule * Add RP2040 build support to QMK * Adjust USB endpoint structs for RP2040 * Add RP2040 bootloader and double-tap reset routine * Add generic and pro micro RP2040 boards * Add RP2040 onekey keyboard * Add WS2812 PIO DMA enabled driver and documentation Supports regular and open-drain output configuration. RP2040 GPIOs are sadly not 5V tolerant, so this is a bit use-less or needs extra hardware or you take the risk to fry your hardware. * Adjust SIO Driver for RP2040 * Adjust I2C Driver for RP2040 * Adjust SPI Driver for RP2040 * Add PIO serial driver and documentation * Add general RP2040 documentation * Apply suggestions from code review Co-authored-by: Nick Brassel <nick@tzarc.org> Co-authored-by: Nick Brassel <nick@tzarc.org>
		
			
				
	
	
		
			57 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			57 lines
		
	
	
	
		
			2 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. The internal timer peripheral is running
 | |
|         // at this point with a precision of 1us.
 | |
|         chSysPolledDelayX(MS2RTC(1 * MHZ, RP2040_BOOTLOADER_DOUBLE_TAP_RESET_TIMEOUT));
 | |
|         magic_location = 0;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     magic_location = 0;
 | |
|     reset_usb_boot(RP2040_BOOTLOADER_DOUBLE_TAP_RESET_LED_MASK, 0U);
 | |
| }
 | |
| 
 | |
| #endif
 |