mirror of
https://github.com/Keychron/qmk_firmware.git
synced 2024-11-22 16:37:58 +06:00
236 lines
5.7 KiB
C
236 lines
5.7 KiB
C
// Copyright 2022 QMK
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "gpio.h"
|
|
#include "matrix.h"
|
|
#include "mcp23018.h"
|
|
#include "util.h"
|
|
#include "wait.h"
|
|
#include "debug.h"
|
|
|
|
#define I2C_ADDR 0x20
|
|
|
|
static uint8_t mcp23018_errors = 0;
|
|
|
|
static void expander_init(void) {
|
|
mcp23018_init(I2C_ADDR);
|
|
}
|
|
|
|
static void expander_init_cols(void) {
|
|
mcp23018_errors += !mcp23018_set_config(I2C_ADDR, mcp23018_PORTA, ALL_INPUT);
|
|
mcp23018_errors += !mcp23018_set_config(I2C_ADDR, mcp23018_PORTB, ALL_INPUT);
|
|
}
|
|
|
|
static void expander_select_row(uint8_t row) {
|
|
if (mcp23018_errors) {
|
|
// wait to mimic i2c interactions
|
|
wait_us(100);
|
|
return;
|
|
}
|
|
|
|
mcp23018_errors += !mcp23018_set_config(I2C_ADDR, mcp23018_PORTB, ~(1 << (row + 1)));
|
|
}
|
|
|
|
static void expander_unselect_row(uint8_t row) {
|
|
// No need to unselect row as the next `select_row` will blank everything anyway
|
|
}
|
|
|
|
static void expander_unselect_rows(void) {
|
|
if (mcp23018_errors) {
|
|
return;
|
|
}
|
|
|
|
mcp23018_errors += !mcp23018_set_config(I2C_ADDR, mcp23018_PORTB, ALL_INPUT);
|
|
}
|
|
|
|
static matrix_row_t expander_read_row(void) {
|
|
if (mcp23018_errors) {
|
|
return 0;
|
|
}
|
|
|
|
uint8_t ret = 0xFF;
|
|
mcp23018_errors += !mcp23018_read_pins(I2C_ADDR, mcp23018_PORTA, &ret);
|
|
|
|
ret = bitrev(~ret);
|
|
ret = ((ret & 0b11111000) >> 1) | (ret & 0b00000011);
|
|
|
|
return ((uint16_t)ret) << 7;
|
|
}
|
|
|
|
static void expander_scan(void) {
|
|
if (!mcp23018_errors) {
|
|
return;
|
|
}
|
|
|
|
static uint16_t mcp23018_reset_loop = 0;
|
|
if (++mcp23018_reset_loop > 0x1FFF) {
|
|
// tuned to about 5s given the current scan rate
|
|
dprintf("trying to reset mcp23018\n");
|
|
mcp23018_reset_loop = 0;
|
|
mcp23018_errors = 0;
|
|
expander_unselect_rows();
|
|
expander_init_cols();
|
|
}
|
|
}
|
|
|
|
/* Column pin configuration
|
|
*
|
|
* Pro Micro: 6 5 4 3 2 1 0
|
|
* PD3 PD2 PD4 PC6 PD7 PE6 PB4
|
|
*
|
|
* Expander: 13 12 11 10 9 8 7
|
|
*/
|
|
static void init_cols(void) {
|
|
// Pro Micro
|
|
gpio_set_pin_input_high(E6);
|
|
gpio_set_pin_input_high(D2);
|
|
gpio_set_pin_input_high(D3);
|
|
gpio_set_pin_input_high(D4);
|
|
gpio_set_pin_input_high(D7);
|
|
gpio_set_pin_input_high(C6);
|
|
gpio_set_pin_input_high(B4);
|
|
|
|
// Expander
|
|
expander_init_cols();
|
|
}
|
|
|
|
static matrix_row_t read_cols(void) {
|
|
// clang-format off
|
|
return expander_read_row() |
|
|
(gpio_read_pin(D3) ? 0 : (1<<6)) |
|
|
(gpio_read_pin(D2) ? 0 : (1<<5)) |
|
|
(gpio_read_pin(D4) ? 0 : (1<<4)) |
|
|
(gpio_read_pin(C6) ? 0 : (1<<3)) |
|
|
(gpio_read_pin(D7) ? 0 : (1<<2)) |
|
|
(gpio_read_pin(E6) ? 0 : (1<<1)) |
|
|
(gpio_read_pin(B4) ? 0 : (1<<0)) ;
|
|
// clang-format on
|
|
}
|
|
|
|
/* Row pin configuration
|
|
*
|
|
* Pro Micro: 0 1 2 3 4 5
|
|
* F4 F5 F6 F7 B1 B2
|
|
*
|
|
* Expander: 0 1 2 3 4 5
|
|
*/
|
|
static void unselect_rows(void) {
|
|
// Pro Micro
|
|
gpio_set_pin_input(B1);
|
|
gpio_set_pin_input(B2);
|
|
gpio_set_pin_input(F4);
|
|
gpio_set_pin_input(F5);
|
|
gpio_set_pin_input(F6);
|
|
gpio_set_pin_input(F7);
|
|
gpio_write_pin_low(B1);
|
|
gpio_write_pin_low(B2);
|
|
gpio_write_pin_low(F4);
|
|
gpio_write_pin_low(F5);
|
|
gpio_write_pin_low(F6);
|
|
gpio_write_pin_low(F7);
|
|
|
|
// Expander
|
|
expander_unselect_rows();
|
|
}
|
|
|
|
static void unselect_row(uint8_t row) {
|
|
// Pro Micro
|
|
switch (row) {
|
|
case 0:
|
|
gpio_set_pin_input(F4);
|
|
gpio_write_pin_low(F4);
|
|
break;
|
|
case 1:
|
|
gpio_set_pin_input(F5);
|
|
gpio_write_pin_low(F5);
|
|
break;
|
|
case 2:
|
|
gpio_set_pin_input(F6);
|
|
gpio_write_pin_low(F6);
|
|
break;
|
|
case 3:
|
|
gpio_set_pin_input(F7);
|
|
gpio_write_pin_low(F7);
|
|
break;
|
|
case 4:
|
|
gpio_set_pin_input(B1);
|
|
gpio_write_pin_low(B1);
|
|
break;
|
|
case 5:
|
|
gpio_set_pin_input(B2);
|
|
gpio_write_pin_low(B2);
|
|
break;
|
|
}
|
|
|
|
// Expander
|
|
expander_unselect_row(row);
|
|
}
|
|
|
|
static void select_row(uint8_t row) {
|
|
// Pro Micro
|
|
switch (row) {
|
|
case 0:
|
|
gpio_set_pin_output(F4);
|
|
gpio_write_pin_low(F4);
|
|
break;
|
|
case 1:
|
|
gpio_set_pin_output(F5);
|
|
gpio_write_pin_low(F5);
|
|
break;
|
|
case 2:
|
|
gpio_set_pin_output(F6);
|
|
gpio_write_pin_low(F6);
|
|
break;
|
|
case 3:
|
|
gpio_set_pin_output(F7);
|
|
gpio_write_pin_low(F7);
|
|
break;
|
|
case 4:
|
|
gpio_set_pin_output(B1);
|
|
gpio_write_pin_low(B1);
|
|
break;
|
|
case 5:
|
|
gpio_set_pin_output(B2);
|
|
gpio_write_pin_low(B2);
|
|
break;
|
|
}
|
|
|
|
// Expander
|
|
expander_select_row(row);
|
|
}
|
|
|
|
static bool read_cols_on_row(matrix_row_t current_matrix[], uint8_t current_row) {
|
|
// Store last value of row prior to reading
|
|
matrix_row_t last_row_value = current_matrix[current_row];
|
|
|
|
// Clear data in matrix row
|
|
current_matrix[current_row] = 0;
|
|
|
|
// Select row and wait for row selection to stabilize
|
|
select_row(current_row);
|
|
// Skip the wait_us(30); as i2c is slow enough to debounce the io changes
|
|
|
|
current_matrix[current_row] = read_cols();
|
|
|
|
unselect_row(current_row);
|
|
|
|
return (last_row_value != current_matrix[current_row]);
|
|
}
|
|
|
|
void matrix_init_custom(void) {
|
|
expander_init();
|
|
|
|
unselect_rows();
|
|
init_cols();
|
|
}
|
|
|
|
bool matrix_scan_custom(matrix_row_t current_matrix[]) {
|
|
expander_scan();
|
|
|
|
bool changed = false;
|
|
for (uint8_t current_row = 0; current_row < MATRIX_ROWS; current_row++) {
|
|
changed |= read_cols_on_row(current_matrix, current_row);
|
|
}
|
|
return changed;
|
|
}
|