// Copyright 2020 Christopher Courtney, aka Drashna Jael're (@drashna) // SPDX-License-Identifier: GPL-2.0-or-later #include "drashna.h" #ifdef __AVR__ # include #endif userspace_config_t userspace_config; /** * @brief Handle registering a keycode, with optional modifer based on timed event * * @param code keycode to send to host * @param mod_code modifier to send with code, if held for tapping term or longer * @param pressed the press/release event (can use "record->event.pressed" for this) * @return true exits function * @return false exits function */ bool mod_key_press_timer(uint16_t code, uint16_t mod_code, bool pressed) { static uint16_t this_timer; mod_key_press(code, mod_code, pressed, this_timer); return false; } /** * @brief Handle registation of keycode, with optional modifier based on custom timer * * @param code keycode to send to host * @param mod_code modifier keycode to send with code, if held for tapping term or longer * @param pressed the press/release event * @param this_timer custom timer to use * @return true * @return false */ bool mod_key_press(uint16_t code, uint16_t mod_code, bool pressed, uint16_t this_timer) { if (pressed) { this_timer = timer_read(); } else { if (timer_elapsed(this_timer) < TAPPING_TERM) { tap_code(code); } else { register_code(mod_code); tap_code(code); unregister_code(mod_code); } } return false; } /** * @brief Performs exact match for modifier values * * @param value the modifer varible (get_mods/get_oneshot_mods/get_weak_mods) * @param mask the modifier mask to check for * @return true Has the exact modifiers specifed * @return false Does not have the exact modifiers specified */ bool hasAllBitsInMask(uint8_t value, uint8_t mask) { value &= 0xF; mask &= 0xF; return (value & mask) == mask; } /** * @brief Tap keycode, with no mods * * @param kc keycode to use */ void tap_code16_nomods(uint16_t kc) { uint8_t temp_mod = get_mods(); clear_mods(); clear_oneshot_mods(); tap_code16(kc); set_mods(temp_mod); } #ifdef I2C_SCANNER_ENABLE # include "i2c_master.h" # include "debug.h" # ifndef I2C_SCANNER_TIMEOUT # define I2C_SCANNER_TIMEOUT 50 # endif i2c_status_t i2c_start_bodge(uint8_t address, uint16_t timeout) { i2c_start(address); // except on ChibiOS where the only way is do do "something" uint8_t data = 0; return i2c_readReg(address, 0, &data, sizeof(data), I2C_SCANNER_TIMEOUT); } # define i2c_start i2c_start_bodge void do_scan(void) { uint8_t nDevices = 0; dprintf("Scanning...\n"); for (uint8_t address = 1; address < 127; address++) { // The i2c_scanner uses the return value of // i2c_start to see if a device did acknowledge to the address. i2c_status_t error = i2c_start(address << 1, I2C_SCANNER_TIMEOUT); if (error == I2C_STATUS_SUCCESS) { i2c_stop(); xprintf(" I2C device found at address 0x%02X\n", I2C_SCANNER_TIMEOUT); nDevices++; } else { // dprintf(" Unknown error (%u) at address 0x%02X\n", error, address); } } if (nDevices == 0) xprintf("No I2C devices found\n"); else xprintf("done\n"); } uint16_t scan_timer = 0; void matrix_scan_i2c(void) { if (timer_elapsed(scan_timer) > 5000) { do_scan(); scan_timer = timer_read(); } } void keyboard_post_init_i2c(void) { i2c_init(); scan_timer = timer_read(); } #endif void bootmagic_lite(void) { bool perform_reset = false; // We need multiple scans because debouncing can't be turned off. matrix_scan(); #if defined(DEBOUNCE) && DEBOUNCE > 0 wait_ms(DEBOUNCE * 2); #else wait_ms(30); #endif matrix_scan(); // If the configured key (commonly Esc) is held down on power up, // reset the EEPROM valid state and jump to bootloader. // This isn't very generalized, but we need something that doesn't // rely on user's keymaps in firmware or EEPROM. uint8_t row = BOOTMAGIC_LITE_ROW, col = BOOTMAGIC_LITE_COLUMN; #if defined(BOOTMAGIC_LITE_EEPROM_ROW) && defined(BOOTMAGIC_LITE_EEPROM_COLUMN) uint8_t row_e = BOOTMAGIC_LITE_EEPROM_ROW, col_e = BOOTMAGIC_LITE_EEPROM_COLUMN; #endif #if defined(SPLIT_KEYBOARD) && defined(BOOTMAGIC_LITE_ROW_RIGHT) && defined(BOOTMAGIC_LITE_COLUMN_RIGHT) if (!is_keyboard_left()) { row = BOOTMAGIC_LITE_ROW_RIGHT; col = BOOTMAGIC_LITE_COLUMN_RIGHT; #if defined(BOOTMAGIC_LITE_EEPROM_ROW) && defined(BOOTMAGIC_LITE_EEPROM_COLUMN) && defined(BOOTMAGIC_LITE_EEPROM_ROW_RIGHT) && defined(BOOTMAGIC_LITE_EEPROM_COLUMN_RIGHT) row_e = BOOTMAGIC_LITE_EEPROM_ROW_RIGHT; col_e = BOOTMAGIC_LITE_EEPROM_COLUMN_RIGHT; # endif } #endif #if defined(BOOTMAGIC_LITE_EEPROM_ROW) && defined(BOOTMAGIC_LITE_EEPROM_COLUMN) if (matrix_get_row(row_e) & (1 << col_e)) { eeconfig_disable(); perform_reset = true; } #endif if (matrix_get_row(row) & (1 << col)) { perform_reset = true; } #ifdef STM32F411xE if (!readPin(A0)) { perform_reset = true; } #endif if (perform_reset) { bootloader_jump(); } }