mirror of
https://github.com/Keychron/qmk_firmware.git
synced 2024-11-25 18:06:34 +06:00
311 lines
8.1 KiB
C
311 lines
8.1 KiB
C
/*
|
|
Copyright 2011 Jun Wako <wakojun@gmail.com>
|
|
Copyright 2016 Ethan Apodaca <papodaca@gmail.com>
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
#include "action.h"
|
|
#include "print.h"
|
|
#include "util.h"
|
|
#include "debug.h"
|
|
#include "xt.h"
|
|
#include "matrix.h"
|
|
|
|
static void matrix_make(uint8_t code);
|
|
static void matrix_break(uint8_t code);
|
|
|
|
static uint8_t matrix[MATRIX_ROWS];
|
|
|
|
#define ROW(code) (code >> 3)
|
|
#define COL(code) (code & 0x07)
|
|
|
|
__attribute__ ((weak))
|
|
void matrix_init_kb(void) {
|
|
matrix_init_user();
|
|
}
|
|
|
|
__attribute__ ((weak))
|
|
void matrix_scan_kb(void) {
|
|
matrix_scan_user();
|
|
}
|
|
|
|
__attribute__ ((weak))
|
|
void matrix_init_user(void) {
|
|
}
|
|
|
|
__attribute__ ((weak))
|
|
void matrix_scan_user(void) { }
|
|
|
|
void matrix_init(void) {
|
|
debug_enable = true;
|
|
xt_host_init();
|
|
|
|
// initialize matrix state: all keys off
|
|
for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
|
|
matrix[i] = 0x00;
|
|
}
|
|
|
|
matrix_init_kb();
|
|
}
|
|
|
|
// convert E0-escaped codes into unused area
|
|
static uint8_t move_e0code(uint8_t code) {
|
|
switch(code) {
|
|
// Original IBM XT keyboard has these keys
|
|
case 0x37: return 0x54; // Print Screen
|
|
case 0x46: return 0x55; // Ctrl + Pause
|
|
case 0x1C: return 0x6F; // Keypad Enter
|
|
case 0x35: return 0x7F; // Keypad /
|
|
|
|
// Any XT keyboard with these keys?
|
|
// http://download.microsoft.com/download/1/6/1/161ba512-40e2-4cc9-843a-923143f3456c/translate.pdf
|
|
// https://download.microsoft.com/download/1/6/1/161ba512-40e2-4cc9-843a-923143f3456c/scancode.doc
|
|
case 0x5B: return 0x5A; // Left GUI
|
|
case 0x5C: return 0x5B; // Right GUI
|
|
case 0x5D: return 0x5C; // Application
|
|
case 0x5E: return 0x5D; // Power(not used)
|
|
case 0x5F: return 0x5E; // Sleep(not used)
|
|
case 0x63: return 0x5F; // Wake (not used)
|
|
case 0x48: return 0x60; // Up
|
|
case 0x4B: return 0x61; // Left
|
|
case 0x50: return 0x62; // Down
|
|
case 0x4D: return 0x63; // Right
|
|
case 0x52: return 0x71; // Insert
|
|
case 0x53: return 0x72; // Delete
|
|
case 0x47: return 0x74; // Home
|
|
case 0x4F: return 0x75; // End
|
|
case 0x49: return 0x77; // Home
|
|
case 0x51: return 0x78; // End
|
|
case 0x1D: return 0x7A; // Right Ctrl
|
|
case 0x38: return 0x7C; // Right Alt
|
|
}
|
|
return 0x00;
|
|
}
|
|
|
|
uint8_t matrix_scan(void) {
|
|
static enum {
|
|
XT_STATE_INIT,
|
|
XT_STATE_E0,
|
|
// Pause: E1 1D 45, E1 9D C5
|
|
XT_STATE_E1,
|
|
XT_STATE_E1_1D,
|
|
XT_STATE_E1_9D,
|
|
} state = XT_STATE_INIT;
|
|
|
|
uint8_t code = xt_host_recv();
|
|
|
|
if (!code) {
|
|
return 0;
|
|
}
|
|
|
|
xprintf("%02X ", code);
|
|
|
|
switch (state) {
|
|
case XT_STATE_INIT:
|
|
switch (code) {
|
|
case 0xE0:
|
|
state = XT_STATE_E0;
|
|
break;
|
|
case 0xE1:
|
|
state = XT_STATE_E1;
|
|
break;
|
|
default:
|
|
if (code < 0x80) {
|
|
matrix_make(code);
|
|
} else {
|
|
matrix_break(code & 0x7F);
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
case XT_STATE_E0:
|
|
switch (code) {
|
|
case 0x2A:
|
|
case 0xAA:
|
|
case 0x36:
|
|
case 0xB6:
|
|
//ignore fake shift
|
|
state = XT_STATE_INIT;
|
|
break;
|
|
default:
|
|
if (code < 0x80) {
|
|
matrix_make(move_e0code(code));
|
|
} else {
|
|
matrix_break(move_e0code(code & 0x7F));
|
|
}
|
|
state = XT_STATE_INIT;
|
|
break;
|
|
}
|
|
break;
|
|
case XT_STATE_E1:
|
|
switch (code) {
|
|
case 0x1D:
|
|
state = XT_STATE_E1_1D;
|
|
break;
|
|
case 0x9D:
|
|
state = XT_STATE_E1_9D;
|
|
break;
|
|
default:
|
|
state = XT_STATE_INIT;
|
|
break;
|
|
}
|
|
break;
|
|
case XT_STATE_E1_1D:
|
|
switch (code) {
|
|
case 0x45:
|
|
matrix_make(0x55);
|
|
break;
|
|
default:
|
|
state = XT_STATE_INIT;
|
|
break;
|
|
}
|
|
break;
|
|
case XT_STATE_E1_9D:
|
|
switch (code) {
|
|
case 0x45:
|
|
matrix_break(0x55);
|
|
break;
|
|
default:
|
|
state = XT_STATE_INIT;
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
state = XT_STATE_INIT;
|
|
}
|
|
|
|
matrix_scan_kb();
|
|
|
|
return 1;
|
|
}
|
|
|
|
inline
|
|
uint8_t matrix_get_row(uint8_t row) {
|
|
return matrix[row];
|
|
}
|
|
|
|
inline static void matrix_make(uint8_t code) {
|
|
if (!matrix_is_on(ROW(code), COL(code))) {
|
|
matrix[ROW(code)] |= 1 << COL(code);
|
|
}
|
|
}
|
|
|
|
inline static void matrix_break(uint8_t code) {
|
|
if (matrix_is_on(ROW(code), COL(code))) {
|
|
matrix[ROW(code)] &= ~(1 << COL(code));
|
|
}
|
|
}
|
|
|
|
void matrix_clear(void) {
|
|
for (uint8_t i=0; i < MATRIX_ROWS; i++) matrix[i] = 0x00;
|
|
}
|
|
|
|
bool matrix_is_on(uint8_t row, uint8_t col) {
|
|
return (matrix_get_row(row) & (1 << col));
|
|
}
|
|
|
|
#if (MATRIX_COLS <= 8)
|
|
# define print_matrix_header() print("\nr/c 01234567\n")
|
|
# define print_matrix_row(row) print_bin_reverse8(matrix_get_row(row))
|
|
#elif (MATRIX_COLS <= 16)
|
|
# define print_matrix_header() print("\nr/c 0123456789ABCDEF\n")
|
|
# define print_matrix_row(row) print_bin_reverse16(matrix_get_row(row))
|
|
#elif (MATRIX_COLS <= 32)
|
|
# define print_matrix_header() print("\nr/c 0123456789ABCDEF0123456789ABCDEF\n")
|
|
# define print_matrix_row(row) print_bin_reverse32(matrix_get_row(row))
|
|
#endif
|
|
|
|
void matrix_print(void) {
|
|
print_matrix_header();
|
|
|
|
for (uint8_t row = 0; row < MATRIX_ROWS; row++) {
|
|
print_hex8(row);
|
|
print(": ");
|
|
print_matrix_row(row);
|
|
print("\n");
|
|
}
|
|
}
|
|
|
|
/*
|
|
XT Scancodes
|
|
============
|
|
- http://download.microsoft.com/download/1/6/1/161ba512-40e2-4cc9-843a-923143f3456c/translate.pdf
|
|
- https://download.microsoft.com/download/1/6/1/161ba512-40e2-4cc9-843a-923143f3456c/scancode.doc
|
|
|
|
01-53: Normal codes used in original XT keyboard
|
|
54-7F: Not used in original XT keyboard
|
|
|
|
0 1 2 3 4 5 6 7 8 9 A B C D E F
|
|
50 - - - - * * x x x x * * * o o o
|
|
60 * * * * x x x x x x x x x x x *
|
|
70 x * * x * * x * * x * x * x x *
|
|
|
|
-: codes existed in original XT keyboard
|
|
*: E0-escaped codes converted into unused code area(internal use in TMK)
|
|
x: Non-espcaped codes(not used in real keyboards probably, for CodeSet2-CodeSet1 translation purpose)
|
|
o: reserved
|
|
|
|
Usage in TMK:
|
|
|
|
00 (reserved) DO NOT USE
|
|
54 PrintScr*
|
|
55 Pause*
|
|
56 Euro2
|
|
57 F11
|
|
58 F12
|
|
59 Keypad =
|
|
5A LGUI*
|
|
5B RGUI*
|
|
5C APP*
|
|
5D (reserved)
|
|
5E (reserved)
|
|
5F (reserved)
|
|
60 cursor*
|
|
61 cursor*
|
|
62 cursor*
|
|
63 cursor*
|
|
64 F13
|
|
65 F14
|
|
66 F15
|
|
67 F16
|
|
68 F17
|
|
69 F18
|
|
6A F19
|
|
6B F20
|
|
6C F21
|
|
6D F22
|
|
6E F23
|
|
6F Keypad Enter*
|
|
70 KANA
|
|
71 nav*
|
|
72 nav*
|
|
73 RO
|
|
74 nav*
|
|
75 nav*
|
|
76 F24
|
|
77 nav*
|
|
78 nav*
|
|
79 HENKAN
|
|
7A RCTL*
|
|
7B MUHENKAN
|
|
7C RALT*
|
|
7D JPY
|
|
7E Keypad ,
|
|
7F Keypad / *
|
|
|
|
*/
|