#include #include "dynamic_keymap.h" #include "raw_hid.h" #include "rgb_matrix.h" #include "version.h" enum Command { // Probe for System76 EC protocol CMD_PROBE = 1, // Read board string CMD_BOARD = 2, // Read version string CMD_VERSION = 3, // Reset to bootloader CMD_RESET = 6, // Get keyboard map index CMD_KEYMAP_GET = 9, // Set keyboard map index CMD_KEYMAP_SET = 10, // Get LED value by index CMD_LED_GET_VALUE = 11, // Set LED value by index CMD_LED_SET_VALUE = 12, // Get LED color by index CMD_LED_GET_COLOR = 13, // Set LED color by index CMD_LED_SET_COLOR = 14, // Get LED matrix mode and speed CMD_LED_GET_MODE = 15, // Set LED matrix mode and speed CMD_LED_SET_MODE = 16, // Get currently pressed keys CMD_MATRIX_GET = 17, // Save LED settings to ROM CMD_LED_SAVE = 18, }; #define CMD_LED_INDEX_ALL 0xFF static bool keymap_get(uint8_t layer, uint8_t output, uint8_t input, uint16_t *value) { if (layer < dynamic_keymap_get_layer_count()) { if (output < MATRIX_ROWS) { if (input < MATRIX_COLS) { *value = dynamic_keymap_get_keycode(layer, output, input); return true; } } } return false; } static bool keymap_set(uint8_t layer, uint8_t output, uint8_t input, uint16_t value) { if (layer < dynamic_keymap_get_layer_count()) { if (output < MATRIX_ROWS) { if (input < MATRIX_COLS) { dynamic_keymap_set_keycode(layer, output, input, value); return true; } } } return false; } static bool bootloader_reset = false; static bool bootloader_unlocked = false; void system76_ec_unlock(void) { rgb_matrix_mode_noeeprom(RGB_MATRIX_CUSTOM_unlocked); bootloader_unlocked = true; } #if defined(RGB_MATRIX_CUSTOM_KB) enum Mode { MODE_SOLID_COLOR = 0, MODE_PER_KEY, MODE_CYCLE_ALL, MODE_CYCLE_LEFT_RIGHT, MODE_CYCLE_UP_DOWN, MODE_CYCLE_OUT_IN, MODE_CYCLE_OUT_IN_DUAL, MODE_RAINBOW_MOVING_CHEVRON, MODE_CYCLE_PINWHEEL, MODE_CYCLE_SPIRAL, MODE_RAINDROPS, MODE_SPLASH, MODE_MULTISPLASH, MODE_ACTIVE_KEYS, MODE_LAST, }; static enum rgb_matrix_effects mode_map[] = { RGB_MATRIX_SOLID_COLOR, RGB_MATRIX_CUSTOM_raw_rgb, RGB_MATRIX_CYCLE_ALL, RGB_MATRIX_CYCLE_LEFT_RIGHT, RGB_MATRIX_CYCLE_UP_DOWN, RGB_MATRIX_CYCLE_OUT_IN, RGB_MATRIX_CYCLE_OUT_IN_DUAL, RGB_MATRIX_RAINBOW_MOVING_CHEVRON, RGB_MATRIX_CYCLE_PINWHEEL, RGB_MATRIX_CYCLE_SPIRAL, RGB_MATRIX_RAINDROPS, RGB_MATRIX_SPLASH, RGB_MATRIX_MULTISPLASH, RGB_MATRIX_CUSTOM_active_keys, }; _Static_assert(sizeof(mode_map) == MODE_LAST, "mode_map_length"); RGB raw_rgb_data[DRIVER_LED_TOTAL]; rgb_config_t layer_rgb[DYNAMIC_KEYMAP_LAYER_COUNT] = { // Layer 1 { .enable = 1, .mode = RGB_MATRIX_STARTUP_MODE, .hsv = { .h = RGB_MATRIX_STARTUP_HUE, .s = RGB_MATRIX_STARTUP_SAT, .v = RGB_MATRIX_STARTUP_VAL, }, .speed = RGB_MATRIX_STARTUP_SPD, }, // Layer 2 { .enable = 1, .mode = RGB_MATRIX_CUSTOM_active_keys, .hsv = { .h = RGB_MATRIX_STARTUP_HUE, .s = RGB_MATRIX_STARTUP_SAT, .v = RGB_MATRIX_STARTUP_VAL, }, .speed = RGB_MATRIX_STARTUP_SPD, }, // Layer 3 { .enable = 1, .mode = RGB_MATRIX_CUSTOM_active_keys, .hsv = { .h = RGB_MATRIX_STARTUP_HUE, .s = RGB_MATRIX_STARTUP_SAT, .v = RGB_MATRIX_STARTUP_VAL, }, .speed = RGB_MATRIX_STARTUP_SPD, }, // Layer 4 { .enable = 1, .mode = RGB_MATRIX_CUSTOM_active_keys, .hsv = { .h = RGB_MATRIX_STARTUP_HUE, .s = RGB_MATRIX_STARTUP_SAT, .v = RGB_MATRIX_STARTUP_VAL, }, .speed = RGB_MATRIX_STARTUP_SPD, }, }; // Read or write EEPROM data with checks for being inside System76 EC region static bool system76_ec_eeprom_op(void * buf, uint16_t size, uint16_t offset, bool write) { uint16_t addr = SYSTEM76_EC_EEPROM_ADDR + offset; uint16_t end = addr + size; if ( (end > addr) && // check for overflow and zero size (addr >= SYSTEM76_EC_EEPROM_ADDR) && (end <= (SYSTEM76_EC_EEPROM_ADDR + SYSTEM76_EC_EEPROM_SIZE)) ) { if (write) { eeprom_update_block( (const void *)buf, (void *)addr, size ); } else { eeprom_read_block( (void *)buf, (const void *)addr, size ); } return true; } else { return false; } } // Read or write EEPROM RGB parameters void system76_ec_rgb_eeprom(bool write) { uint16_t layer_rgb_size = sizeof(layer_rgb); system76_ec_eeprom_op( (void *)layer_rgb, layer_rgb_size, 0, write ); system76_ec_eeprom_op( (void *)raw_rgb_data, sizeof(raw_rgb_data), layer_rgb_size, write ); } // Update RGB parameters on layer change void system76_ec_rgb_layer(layer_state_t layer_state) { if (!bootloader_unlocked) { uint8_t layer = get_highest_layer(layer_state); if (layer < DYNAMIC_KEYMAP_LAYER_COUNT) { rgb_matrix_config = layer_rgb[layer]; } } } #endif // defined(RGB_MATRIX_CUSTOM_KB) void raw_hid_receive(uint8_t *data, uint8_t length) { // Error response by default, set to success by commands data[1] = 1; switch (data[0]) { case CMD_PROBE: // Signature data[2] = 0x76; data[3] = 0xEC; // Version data[4] = 0x01; data[1] = 0; break; case CMD_BOARD: strncpy((char *)&data[2], QMK_KEYBOARD, length - 2); data[1] = 0; break; case CMD_VERSION: strncpy((char *)&data[2], QMK_VERSION, length - 2); data[1] = 0; break; case CMD_RESET: if (bootloader_unlocked) { data[1] = 0; bootloader_reset = true; } break; case CMD_KEYMAP_GET: { uint16_t value = 0; if (keymap_get(data[2], data[3], data[4], &value)) { data[5] = (uint8_t)value; data[6] = (uint8_t)(value >> 8); data[1] = 0; } } break; case CMD_KEYMAP_SET: { uint16_t value = ((uint16_t)data[5]) | (((uint16_t)data[6]) << 8); if (keymap_set(data[2], data[3], data[4], value)) { data[1] = 0; } } break; #if defined(RGB_MATRIX_CUSTOM_KB) case CMD_LED_GET_VALUE: if (!bootloader_unlocked) { uint8_t index = data[2]; for(uint8_t layer = 0; layer < DYNAMIC_KEYMAP_LAYER_COUNT; layer++) { if (index == (0xF0 | layer)) { data[3] = layer_rgb[layer].hsv.v; data[4] = RGB_MATRIX_MAXIMUM_BRIGHTNESS; data[1] = 0; break; } } } break; case CMD_LED_SET_VALUE: if (!bootloader_unlocked) { uint8_t index = data[2]; uint8_t value = data[3]; if (value >= RGB_MATRIX_MAXIMUM_BRIGHTNESS) { value = RGB_MATRIX_MAXIMUM_BRIGHTNESS; } for(uint8_t layer = 0; layer < DYNAMIC_KEYMAP_LAYER_COUNT; layer++) { if (index == (0xF0 | layer)) { layer_rgb[layer].hsv.v = value; data[1] = 0; system76_ec_rgb_layer(layer_state); break; } } } break; case CMD_LED_GET_COLOR: if (!bootloader_unlocked) { uint8_t index = data[2]; if (index < DRIVER_LED_TOTAL) { data[3] = raw_rgb_data[index].r; data[4] = raw_rgb_data[index].g; data[5] = raw_rgb_data[index].b; data[1] = 0; } else { for(uint8_t layer = 0; layer < DYNAMIC_KEYMAP_LAYER_COUNT; layer++) { if (index == (0xF0 | layer)) { data[3] = layer_rgb[layer].hsv.h; data[4] = layer_rgb[layer].hsv.s; data[5] = 0; data[1] = 0; break; } } } } break; case CMD_LED_SET_COLOR: if (!bootloader_unlocked) { uint8_t index = data[2]; RGB rgb = { .r = data[3], .g = data[4], .b = data[5], }; if (index < DRIVER_LED_TOTAL) { raw_rgb_data[index] = rgb; data[1] = 0; } else { for(uint8_t layer = 0; layer < DYNAMIC_KEYMAP_LAYER_COUNT; layer++) { if (index == (0xF0 | layer)) { layer_rgb[layer].hsv.h = rgb.r; layer_rgb[layer].hsv.s = rgb.g; // Ignore rgb.b data[1] = 0; system76_ec_rgb_layer(layer_state); break; } } } } break; case CMD_LED_GET_MODE: if (!bootloader_unlocked) { uint8_t layer = data[2]; if (layer < DYNAMIC_KEYMAP_LAYER_COUNT) { enum rgb_matrix_effects mode = layer_rgb[layer].mode; for (uint8_t i = 0; i < MODE_LAST; i++) { if (mode_map[i] == mode) { data[3] = i; data[4] = layer_rgb[layer].speed; data[1] = 0; break; } } } } break; case CMD_LED_SET_MODE: if (!bootloader_unlocked) { uint8_t layer = data[2]; uint8_t mode = data[3]; uint8_t speed = data[4]; if (layer < DYNAMIC_KEYMAP_LAYER_COUNT && mode < MODE_LAST) { layer_rgb[layer].mode = mode_map[mode]; layer_rgb[layer].speed = speed; data[1] = 0; system76_ec_rgb_layer(layer_state); } } break; case CMD_LED_SAVE: if (!bootloader_unlocked) { system76_ec_rgb_eeprom(true); data[1] = 0; } break; #endif // defined(RGB_MATRIX_CUSTOM_KB) case CMD_MATRIX_GET: { //TODO: improve performance? data[2] = matrix_rows(); data[3] = matrix_cols(); uint8_t byte = 4; uint8_t bit = 0; for (uint8_t row = 0; row < matrix_rows(); row++) { for (uint8_t col = 0; col < matrix_cols(); col++) { if (byte < length) { if (matrix_is_on(row, col)) { data[byte] |= (1 << bit); } else { data[byte] &= ~(1 << bit); } } bit++; if (bit >= 8) { byte++; bit = 0; } } } data[1] = 0; } break; } raw_hid_send(data, length); if (bootloader_reset) { // Give host time to read response wait_ms(100); // Jump to the bootloader bootloader_jump(); } }