mirror of
https://github.com/qmk/qmk_firmware.git
synced 2025-07-20 06:32:01 +00:00
* Add splitkb's Elora * WIP * Make requested changes * fix missing call to keyboard init user for elora * sync vial_qmk with qmk * Add fixes * Add encoder quadrature and update files * Update Readme * Implement changes * Remove encoder driver from keyboard.json * make requested changed * Implement changes and remove myriad_task as it wasn't actually doing anything. * Remove myriad.h from matrix.c * Simplify OLED code debug * Update RGB keycodes (qmk#24484) * remove rules.mk (qmk#23281) * Add matrix state sync --------- Co-authored-by: NapOli1084 <10320176+NapOli1084@users.noreply.github.com> Co-authored-by: harveysch <126267034+harvey-splitkb@users.noreply.github.com> Co-authored-by: harvey <harvey@splitkb.com> Co-authored-by: VeyPatch <126267034+VeyPatch@users.noreply.github.com>
315 lines
9.7 KiB
C
315 lines
9.7 KiB
C
// Copyright 2024 splitkb.com (support@splitkb.com)
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include QMK_KEYBOARD_H
|
|
|
|
#include "myriad.h"
|
|
|
|
#include "i2c_master.h"
|
|
#include "analog.h"
|
|
|
|
typedef struct __attribute__((__packed__)) {
|
|
char magic_numbers[3];
|
|
uint8_t version_major;
|
|
uint8_t version_minor;
|
|
uint8_t version_patch;
|
|
uint32_t checksum;
|
|
uint16_t payload_length;
|
|
} myriad_header_t;
|
|
|
|
typedef struct __attribute__((__packed__)) {
|
|
uint16_t vendor_id;
|
|
uint16_t product_id;
|
|
uint8_t revision;
|
|
} identity_record_t;
|
|
|
|
static bool myriad_reader(uint8_t *data, uint16_t length) {
|
|
const uint8_t eeprom_address = 0x50; // 1010 000 - NOT shifted for R/W bit
|
|
const uint16_t i2c_timeout = 100; // in milliseconds
|
|
|
|
uint8_t num_pages = (length / 256) + 1;
|
|
uint8_t last_page_size = length % 256;
|
|
|
|
for (int i = 0; i < num_pages; i++) {
|
|
uint8_t reg = 0; // We always start on a page boundary, so this is always zero
|
|
uint16_t read_length;
|
|
if (i == num_pages - 1) {
|
|
read_length = last_page_size;
|
|
} else {
|
|
read_length = 256;
|
|
}
|
|
i2c_status_t s = i2c_read_register((eeprom_address + i) << 1, reg, &(data[i*256]), read_length, i2c_timeout);
|
|
if (s != I2C_STATUS_SUCCESS) { return false; }
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static bool verify_header(myriad_header_t *header) {
|
|
char magic_numbers[] = {'M', 'Y', 'R'};
|
|
uint8_t version_major = 1;
|
|
uint16_t version_minor = 0;
|
|
|
|
for (int i = 0; i < sizeof(magic_numbers); i++) {
|
|
// Check that the header starts with 'MYR', indicating that this is indeed a Myriad card.
|
|
if (header->magic_numbers[i] != magic_numbers[i]) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (header->version_major != version_major || header->version_minor > version_minor) {
|
|
// We obviously don't support cards with a different major version, because that indicates a breaking change.
|
|
// We also don't support cards with HIGHER minor version,
|
|
// as we are not guaranteed to be able to properly use all its features.
|
|
return false;
|
|
}
|
|
|
|
if (header->payload_length > (2048 - sizeof(myriad_header_t))) {
|
|
// The EEPROM chips are *at most* 16kb / 2kB large,
|
|
// and some of that is taken up by the header.
|
|
// We obviously can't have a payload which exceeds the EEPROM's size.
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// Sourced from https://en.wikipedia.org/wiki/Adler-32#Example_implementation
|
|
static bool verify_checksum(uint8_t *data, uint16_t length, uint32_t checksum) {
|
|
// Skip the header
|
|
data += sizeof(myriad_header_t);
|
|
length -= sizeof(myriad_header_t);
|
|
|
|
const uint32_t MOD_ADLER = 65521;
|
|
|
|
uint32_t a = 1, b = 0;
|
|
size_t index;
|
|
|
|
// Process each byte of the data in order
|
|
for (index = 0; index < length; ++index)
|
|
{
|
|
a = (a + data[index]) % MOD_ADLER;
|
|
b = (b + a) % MOD_ADLER;
|
|
}
|
|
uint32_t calculated = ((b << 16) | a);
|
|
|
|
return calculated == checksum;
|
|
}
|
|
|
|
// Locates a specific entry by type
|
|
// Returns the offset of the PAYLOAD.
|
|
static int16_t locate_entry(uint8_t entry_type, uint8_t entry_data_length, uint8_t *data, uint16_t minimum, uint16_t maximum) {
|
|
if (minimum < sizeof(myriad_header_t)) {
|
|
// Records must start *after* the header.
|
|
// We silently allow this so the caller can just specify `0` as minimum for the first entry.
|
|
minimum = sizeof(myriad_header_t);
|
|
}
|
|
|
|
uint16_t offset = minimum;
|
|
while (offset < maximum) {
|
|
if (data[offset] == entry_type) {
|
|
// Type matches!
|
|
if (data[offset+1] == entry_data_length) {
|
|
// We found what we are looking for, so return payload reference.
|
|
return offset+2;
|
|
} else {
|
|
// The entry is the wrong length?
|
|
return -2;
|
|
}
|
|
} else {
|
|
// No type match, so skip this one
|
|
// We skip the type byte, the length byte, and any potential data (with length stored in the length byte)
|
|
offset += 2 + data[offset+1];
|
|
}
|
|
}
|
|
|
|
// We hit the maximum and didn't find what we are looking for
|
|
return -1;
|
|
}
|
|
|
|
static bool read_card_identity(uint8_t *data, uint16_t length, identity_record_t *record) {
|
|
const uint8_t identity_type = 0x01;
|
|
const uint8_t entry_data_length = sizeof(identity_record_t);
|
|
int16_t result = locate_entry(identity_type, entry_data_length, data, 0, length);
|
|
if (result < 0) { return false; }
|
|
|
|
for (int i = 0; i < sizeof(identity_record_t); i++) {
|
|
((uint8_t*)record)[i] = data[result + i];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static myriad_card_t _detect_myriad(void) {
|
|
gpio_set_pin_input(MYRIAD_PRESENT);
|
|
wait_ms(100);
|
|
// The pin has an external pull-up, and a Myriad card shorts it to ground.
|
|
#ifndef MYRIAD_OVERRIDE_PRESENCE
|
|
if (gpio_read_pin(MYRIAD_PRESENT)) {
|
|
return NONE;
|
|
}
|
|
#endif
|
|
|
|
// Attempt to read header
|
|
myriad_header_t header;
|
|
if (!myriad_reader((uint8_t*)&header, sizeof(header))) { return INVALID; }
|
|
if (!verify_header(&header)) { return INVALID; }
|
|
|
|
// Now that we have determined that the header is valid
|
|
// and we know the payload length, read the entire thing
|
|
uint8_t data[2048]; // Guaranteed to be large enough.
|
|
uint16_t data_size = sizeof(header)+header.payload_length;
|
|
if (!myriad_reader(data, data_size)) { return INVALID; }
|
|
if (!verify_checksum(data, data_size, header.checksum)) { return INVALID; }
|
|
|
|
identity_record_t identity;
|
|
if (!read_card_identity(data, data_size, &identity)) { return INVALID; }
|
|
|
|
if (identity.vendor_id == 0x0001 && identity.product_id == 0x0001) {
|
|
return SKB_ENCODER;
|
|
} else if (identity.vendor_id == 0x0001 && identity.product_id == 0x0002) {
|
|
return SKB_JOYSTICK;
|
|
} else if (identity.vendor_id == 0x0001 && identity.product_id == 0x0003) {
|
|
return SKB_SWITCHES;
|
|
}
|
|
|
|
return UNKNOWN;
|
|
}
|
|
|
|
// Determine card presence & identity
|
|
// Does NOT initialize the card for use!
|
|
myriad_card_t detect_myriad(void) {
|
|
static myriad_card_t card = UNINITIALIZED;
|
|
|
|
if (card == UNINITIALIZED) {
|
|
i2c_init();
|
|
card = _detect_myriad();
|
|
}
|
|
|
|
return card;
|
|
}
|
|
|
|
static void myr_switches_init(void) {
|
|
gpio_set_pin_input_high(MYRIAD_GPIO1); // S4
|
|
gpio_set_pin_input_high(MYRIAD_GPIO2); // S2
|
|
gpio_set_pin_input_high(MYRIAD_GPIO3); // S1
|
|
gpio_set_pin_input_high(MYRIAD_GPIO4); // S3
|
|
}
|
|
|
|
static void myr_encoder_init(void) {
|
|
gpio_set_pin_input_high(MYRIAD_GPIO1); // Press
|
|
gpio_set_pin_input_high(MYRIAD_GPIO2); // A
|
|
gpio_set_pin_input_high(MYRIAD_GPIO3); // B
|
|
}
|
|
|
|
static uint16_t myr_joystick_timer;
|
|
static void myr_joystick_init(void) {
|
|
gpio_set_pin_input_high(MYRIAD_GPIO1); // Press
|
|
|
|
myr_joystick_timer = timer_read();
|
|
}
|
|
|
|
// Make sure any card present is ready for use
|
|
static myriad_card_t myriad_card_init(void) {
|
|
static bool initialized = false;
|
|
|
|
myriad_card_t card = detect_myriad();
|
|
if (initialized) {
|
|
return card;
|
|
}
|
|
initialized = true;
|
|
|
|
switch (card) {
|
|
case SKB_SWITCHES:
|
|
myr_switches_init();
|
|
break;
|
|
case SKB_ENCODER:
|
|
myr_encoder_init();
|
|
break;
|
|
case SKB_JOYSTICK:
|
|
myr_joystick_init();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return card;
|
|
}
|
|
|
|
bool myriad_hook_matrix(matrix_row_t current_matrix[]) {
|
|
myriad_card_t card = myriad_card_init();
|
|
uint8_t word = 0;
|
|
|
|
if (card == SKB_SWITCHES) {
|
|
word |= ((!gpio_read_pin(MYRIAD_GPIO3)) & 1) << 0;
|
|
word |= ((!gpio_read_pin(MYRIAD_GPIO2)) & 1) << 1;
|
|
word |= ((!gpio_read_pin(MYRIAD_GPIO4)) & 1) << 2;
|
|
word |= ((!gpio_read_pin(MYRIAD_GPIO1)) & 1) << 3;
|
|
} else if (card == SKB_ENCODER) {
|
|
word |= ((!gpio_read_pin(MYRIAD_GPIO1)) & 1) << 4;
|
|
} else if (card == SKB_JOYSTICK) {
|
|
word |= ((!gpio_read_pin(MYRIAD_GPIO1)) & 1) << 4;
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
// 5 bytes of on-board keys, so we are the 6th
|
|
bool matrix_has_changed = current_matrix[5] ^ word;
|
|
current_matrix[5] = word;
|
|
|
|
return matrix_has_changed;
|
|
}
|
|
|
|
static pin_t encoders_pad_a[NUM_ENCODERS_MAX_PER_SIDE];
|
|
static pin_t encoders_pad_b[NUM_ENCODERS_MAX_PER_SIDE];
|
|
|
|
uint8_t myriad_hook_encoder(uint8_t index, bool pad_b) {
|
|
if (myriad_card_init() != SKB_ENCODER) { return 0; }
|
|
// 3 onboard encoders, so we are number 4
|
|
pin_t pin = pad_b ? encoders_pad_b[index] : encoders_pad_a[index];
|
|
encoders_pad_a[3] = MYRIAD_GPIO2;
|
|
encoders_pad_b[3] = MYRIAD_GPIO3;
|
|
return gpio_read_pin(pin) ? 1 : 0;
|
|
}
|
|
|
|
report_mouse_t pointing_device_driver_get_report(report_mouse_t mouse_report) {
|
|
if (myriad_card_init() != SKB_JOYSTICK) { return mouse_report; }
|
|
|
|
if (timer_elapsed(myr_joystick_timer) < 10) {
|
|
wait_ms(2);
|
|
return mouse_report;
|
|
}
|
|
|
|
myr_joystick_timer = timer_read();
|
|
|
|
// `analogReadPin` returns 0..1023
|
|
int32_t y = (analogReadPin(MYRIAD_ADC1) - 512) * -1; // Note: axis is flipped
|
|
int32_t x = analogReadPin(MYRIAD_ADC2) - 512;
|
|
// Values are now -512..512
|
|
|
|
// Create a dead zone in the middle where the mouse doesn't move
|
|
const int16_t dead_zone = 10;
|
|
if ((y < 0 && y > -1*dead_zone) || (y > 0 && y < dead_zone)) {
|
|
y = 0;
|
|
}
|
|
if ((x < 0 && x > -1*dead_zone) || (x > 0 && x < dead_zone)) {
|
|
x = 0;
|
|
}
|
|
|
|
// quadratic movement
|
|
x = abs(x) * x / 5000;
|
|
y = abs(y) * y / 5000;
|
|
|
|
// Clamp final value to make sure we don't under/overflow
|
|
if (y < -127) { y = -127; }
|
|
if (y > 127) { y = 127; }
|
|
if (x < -127) { x = -127; }
|
|
if (x > 127) { x = 127; }
|
|
|
|
mouse_report.x = x;
|
|
mouse_report.y = y;
|
|
|
|
return mouse_report;
|
|
}
|
|
|
|
void pointing_device_driver_init(void) {
|
|
gpio_set_pin_input(MYRIAD_ADC1); // Y
|
|
gpio_set_pin_input(MYRIAD_ADC2); // X
|
|
} |