summaryrefslogtreecommitdiff
path: root/hhkb/matrix.c
diff options
context:
space:
mode:
Diffstat (limited to 'hhkb/matrix.c')
-rw-r--r--hhkb/matrix.c91
1 files changed, 91 insertions, 0 deletions
diff --git a/hhkb/matrix.c b/hhkb/matrix.c
new file mode 100644
index 0000000000..3034a63612
--- /dev/null
+++ b/hhkb/matrix.c
@@ -0,0 +1,91 @@
+/*
+ * scan matrix
+ */
+#include <avr/io.h>
+#include <util/delay.h>
+#include "keymap.h"
+#include "matrix.h"
+#include "print.h"
+
+// matrix is active low. (key on: 0/key off: 1)
+//
+// HHKB has no ghost and no bounce.
+// row: HC4051 select input channel(0-8)
+// PB0, PB1, PB2(A, B, C)
+// col: LS145 select low output line(0-8)
+// PB3, PB4, PB5, PB6(A, B, C, D)
+// use D as ENABLE: (enable: 0/unenable: 1)
+// key: KEY: (on: 0/ off:1)
+// UNKNOWN: unknown whether input or output
+// PE6,PE7(KEY, UNKNOWN)
+#define COL_ENABLE (1<<6)
+#define KEY_SELELCT(ROW, COL) (PORTB = COL_ENABLE|(((COL)&0x07)<<3)|((ROW)&0x07))
+#define KEY_ENABLE (PORTB &= ~COL_ENABLE)
+#define KEY_UNABLE (PORTB |= COL_ENABLE)
+#define KEY_ON ((PINE&(1<<6)) ? false : true)
+
+// matrix state buffer
+uint8_t *matrix;
+uint8_t *matrix_prev;
+static uint8_t _matrix0[MATRIX_ROWS];
+static uint8_t _matrix1[MATRIX_ROWS];
+
+
+// this must be called once before matrix_scan.
+void matrix_init(void)
+{
+ // row & col output(PB0-6)
+ DDRB = 0xFF;
+ PORTB = KEY_SELELCT(0, 0);
+ // KEY & VALID input with pullup(PE6,7)
+ DDRE = 0x3F;
+ PORTE = 0xC0;
+
+ // initialize matrix state: all keys off
+ for (int i=0; i < MATRIX_ROWS; i++) _matrix0[i] = 0xFF;
+ for (int i=0; i < MATRIX_ROWS; i++) _matrix1[i] = 0xFF;
+ matrix = _matrix0;
+ matrix_prev = _matrix1;
+}
+
+uint8_t matrix_scan(void)
+{
+ uint8_t *tmp;
+
+ tmp = matrix_prev;
+ matrix_prev = matrix;
+ matrix = tmp;
+
+ for (int row = 0; row < MATRIX_ROWS; row++) {
+ for (int col = 0; col < MATRIX_COLS; col++) {
+ KEY_SELELCT(row, col);
+ _delay_us(50); // from logic analyzer chart
+ KEY_ENABLE;
+ _delay_us(10); // from logic analyzer chart
+ if (KEY_ON) {
+ matrix[row] &= ~(1<<col);
+ } else {
+ matrix[row] |= (1<<col);
+ }
+ KEY_UNABLE;
+ _delay_us(150); // from logic analyzer chart
+ }
+ }
+ return 1;
+}
+
+bool matrix_is_modified(void) {
+ for (int i=0; i <MATRIX_ROWS; i++) {
+ if (matrix[i] != matrix_prev[i])
+ return true;
+ }
+ return false;
+}
+
+bool matrix_has_ghost(void) {
+ return false;
+}
+
+bool matrix_has_ghost_in_row(uint8_t row) {
+ return false;
+}