logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

matrix.c (15768B)


  1. /*
  2. Copyright 2012-2018 Jun Wako, Jack Humbert, Yiancar
  3. Copyright 2019 Evy Dekkers
  4. This program is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation, either version 2 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. #include "wait.h"
  16. #include "util.h"
  17. #include "matrix.h"
  18. #include "debounce.h"
  19. #ifdef DIRECT_PINS
  20. static pin_t direct_pins[MATRIX_ROWS][MATRIX_COLS] = DIRECT_PINS;
  21. #elif (DIODE_DIRECTION == ROW2COL) || (DIODE_DIRECTION == COL2ROW)
  22. static const pin_t row_pins[MATRIX_ROWS] = MATRIX_ROW_PINS;
  23. //static const pin_t col_pins[MATRIX_COLS] = MATRIX_COL_PINS;
  24. #endif
  25. // matrix code
  26. #ifdef DIRECT_PINS
  27. static void init_pins(void) {
  28. for (int row = 0; row < MATRIX_ROWS; row++) {
  29. for (int col = 0; col < MATRIX_COLS; col++) {
  30. pin_t pin = direct_pins[row][col];
  31. if (pin != NO_PIN) {
  32. gpio_set_pin_input_high(pin);
  33. }
  34. }
  35. }
  36. }
  37. static bool read_cols_on_row(matrix_row_t current_matrix[], uint8_t current_row) {
  38. matrix_row_t last_row_value = current_matrix[current_row];
  39. current_matrix[current_row] = 0;
  40. for (uint8_t col_index = 0; col_index < MATRIX_COLS; col_index++) {
  41. pin_t pin = direct_pins[current_row][col_index];
  42. if (pin != NO_PIN) {
  43. current_matrix[current_row] |= gpio_read_pin(pin) ? 0 : (MATRIX_ROW_SHIFTER << col_index);
  44. }
  45. }
  46. return (last_row_value != current_matrix[current_row]);
  47. }
  48. #elif (DIODE_DIRECTION == COL2ROW)
  49. static void select_row(uint8_t row) {
  50. gpio_set_pin_output(row_pins[row]);
  51. gpio_write_pin_low(row_pins[row]);
  52. }
  53. static void unselect_row(uint8_t row) { gpio_set_pin_input_high(row_pins[row]); }
  54. static void unselect_rows(void) {
  55. for (uint8_t x = 0; x < MATRIX_ROWS; x++) {
  56. gpio_set_pin_input_high(row_pins[x]);
  57. }
  58. }
  59. static void init_pins(void) {
  60. unselect_rows();
  61. for (uint8_t x = 0; x < MATRIX_COLS; x++) {
  62. gpio_set_pin_input_high(col_pins[x]);
  63. }
  64. }
  65. static bool read_cols_on_row(matrix_row_t current_matrix[], uint8_t current_row) {
  66. // Store last value of row prior to reading
  67. matrix_row_t last_row_value = current_matrix[current_row];
  68. // Clear data in matrix row
  69. current_matrix[current_row] = 0;
  70. // Select row and wait for row selecton to stabilize
  71. select_row(current_row);
  72. wait_us(30);
  73. // For each col...
  74. for (uint8_t col_index = 0; col_index < MATRIX_COLS; col_index++) {
  75. // Select the col pin to read (active low)
  76. uint8_t pin_state = gpio_read_pin(col_pins[col_index]);
  77. // Populate the matrix row with the state of the col pin
  78. current_matrix[current_row] |= pin_state ? 0 : (MATRIX_ROW_SHIFTER << col_index);
  79. }
  80. // Unselect row
  81. unselect_row(current_row);
  82. return (last_row_value != current_matrix[current_row]);
  83. }
  84. #elif (DIODE_DIRECTION == ROW2COL)
  85. /* Cols 0 - 16
  86. * These columns use two 74HC138 3 to 8 bit demultiplexer. B4, C7 is the enable pin, must be set high (1) to use it.
  87. *
  88. * col / pin: PA0 PA1 PA2 PB4 PC7 PC2 PC3 PC5
  89. * 0: 0 ── 0 ── 0 1 ── 0 0 0 0
  90. * ────────────────────────────────────────────
  91. * 1: 0 ── 0 ── 1 1 ── 0 0 0 0
  92. * ────────────────────────────────────────────
  93. * 2: 0 ── 1 ── 0 1 ── 0 0 0 0
  94. * ────────────────────────────────────────────
  95. * 3: 0 ── 1 ── 1 1 ── 0 0 0 0
  96. * ────────────────────────────────────────────
  97. * 4: 1 ── 0 ── 0 1 ── 0 0 0 0
  98. * ────────────────────────────────────────────
  99. * 5: 1 ── 0 ── 1 1 ── 0 0 0 0
  100. * ────────────────────────────────────────────
  101. * 6: 1 ── 1 ── 0 1 ── 0 0 0 0
  102. * ────────────────────────────────────────────
  103. * 7: 1 ── 1 ── 1 1 ── 0 0 0 0
  104. * ────────────────────────────────────────────
  105. * 8: 1 ── 1 ── 1 0 ── 1 0 0 0
  106. * ────────────────────────────────────────────
  107. * 9: 0 ── 0 ── 0 0 ── 1 0 0 0
  108. * ────────────────────────────────────────────
  109. *10: 0 ── 0 ── 1 0 ── 1 0 0 0
  110. * ────────────────────────────────────────────
  111. *11: 0 ── 1 ── 0 0 ── 1 0 0 0
  112. * ────────────────────────────────────────────
  113. *12: 0 ── 1 ── 1 0 ── 1 0 0 0
  114. * ────────────────────────────────────────────
  115. *13: 1 ── 0 ── 0 0 ── 1 0 0 0
  116. * ────────────────────────────────────────────
  117. *14: 1 ── 0 ── 1 0 ── 1 0 0 0
  118. * ────────────────────────────────────────────
  119. *15: 1 ── 1 ── 0 0 ── 1 0 0 0
  120. * ────────────────────────────────────────────
  121. *16: 0 ── 0 ── 0 0 ── 0 1 0 0
  122. * ────────────────────────────────────────────
  123. *17: 0 ── 0 ── 0 0 ── 0 0 1 0
  124. * ────────────────────────────────────────────
  125. *18: 0 ── 0 ── 0 0 ── 0 0 0 1
  126. *
  127. */
  128. static void select_col(uint8_t col) {
  129. switch (col) {
  130. case 0:
  131. gpio_write_pin_low(A0);
  132. gpio_write_pin_low(A1);
  133. gpio_write_pin_low(A2);
  134. gpio_write_pin_high(B4);
  135. break;
  136. case 1:
  137. gpio_write_pin_low(A0);
  138. gpio_write_pin_low(A1);
  139. gpio_write_pin_high(A2);
  140. gpio_write_pin_high(B4);
  141. break;
  142. case 2:
  143. gpio_write_pin_low(A0);
  144. gpio_write_pin_high(A1);
  145. gpio_write_pin_low(A2);
  146. gpio_write_pin_high(B4);
  147. break;
  148. case 3:
  149. gpio_write_pin_low(A0);
  150. gpio_write_pin_high(A1);
  151. gpio_write_pin_high(A2);
  152. gpio_write_pin_high(B4);
  153. break;
  154. case 4:
  155. gpio_write_pin_high(A0);
  156. gpio_write_pin_low(A1);
  157. gpio_write_pin_low(A2);
  158. gpio_write_pin_high(B4);
  159. break;
  160. case 5:
  161. gpio_write_pin_high(A0);
  162. gpio_write_pin_low(A1);
  163. gpio_write_pin_high(A2);
  164. gpio_write_pin_high(B4);
  165. break;
  166. case 6:
  167. gpio_write_pin_high(A0);
  168. gpio_write_pin_high(A1);
  169. gpio_write_pin_low(A2);
  170. gpio_write_pin_high(B4);
  171. break;
  172. case 7:
  173. gpio_write_pin_high(A0);
  174. gpio_write_pin_high(A1);
  175. gpio_write_pin_high(A2);
  176. gpio_write_pin_high(B4);
  177. break;
  178. case 8:
  179. gpio_write_pin_high(A0);
  180. gpio_write_pin_high(A1);
  181. gpio_write_pin_high(A2);
  182. gpio_write_pin_high(C7);
  183. break;
  184. case 9:
  185. gpio_write_pin_low(A0);
  186. gpio_write_pin_low(A1);
  187. gpio_write_pin_low(A2);
  188. gpio_write_pin_high(C7);
  189. break;
  190. case 10:
  191. gpio_write_pin_low(A0);
  192. gpio_write_pin_low(A1);
  193. gpio_write_pin_high(A2);
  194. gpio_write_pin_high(C7);
  195. break;
  196. case 11:
  197. gpio_write_pin_low(A0);
  198. gpio_write_pin_high(A1);
  199. gpio_write_pin_low(A2);
  200. gpio_write_pin_high(C7);
  201. break;
  202. case 12:
  203. gpio_write_pin_low(A0);
  204. gpio_write_pin_high(A1);
  205. gpio_write_pin_high(A2);
  206. gpio_write_pin_high(C7);
  207. break;
  208. case 13:
  209. gpio_write_pin_high(A0);
  210. gpio_write_pin_low(A1);
  211. gpio_write_pin_low(A2);
  212. gpio_write_pin_high(C7);
  213. break;
  214. case 14:
  215. gpio_write_pin_high(A0);
  216. gpio_write_pin_low(A1);
  217. gpio_write_pin_high(A2);
  218. gpio_write_pin_high(C7);
  219. break;
  220. case 15:
  221. gpio_write_pin_high(A0);
  222. gpio_write_pin_high(A1);
  223. gpio_write_pin_low(A2);
  224. gpio_write_pin_high(C7);
  225. break;
  226. case 16:
  227. gpio_write_pin_low(C2);
  228. break;
  229. case 17:
  230. gpio_write_pin_low(C3);
  231. break;
  232. case 18:
  233. gpio_write_pin_low(C5);
  234. break;
  235. }
  236. }
  237. static void unselect_col(uint8_t col) {
  238. switch (col) {
  239. case 0:
  240. gpio_write_pin_high(A0);
  241. gpio_write_pin_high(A1);
  242. gpio_write_pin_high(A2);
  243. gpio_write_pin_low(B4);
  244. break;
  245. case 1:
  246. gpio_write_pin_high(A0);
  247. gpio_write_pin_high(A1);
  248. gpio_write_pin_low(A2);
  249. gpio_write_pin_low(B4);
  250. break;
  251. case 2:
  252. gpio_write_pin_high(A0);
  253. gpio_write_pin_low(A1);
  254. gpio_write_pin_high(A2);
  255. gpio_write_pin_low(B4);
  256. break;
  257. case 3:
  258. gpio_write_pin_high(A0);
  259. gpio_write_pin_low(A1);
  260. gpio_write_pin_low(A2);
  261. gpio_write_pin_low(B4);
  262. break;
  263. case 4:
  264. gpio_write_pin_low(A0);
  265. gpio_write_pin_high(A1);
  266. gpio_write_pin_high(A2);
  267. gpio_write_pin_low(B4);
  268. break;
  269. case 5:
  270. gpio_write_pin_low(A0);
  271. gpio_write_pin_high(A1);
  272. gpio_write_pin_low(A2);
  273. gpio_write_pin_low(B4);
  274. break;
  275. case 6:
  276. gpio_write_pin_low(A0);
  277. gpio_write_pin_low(A1);
  278. gpio_write_pin_high(A2);
  279. gpio_write_pin_low(B4);
  280. break;
  281. case 7:
  282. gpio_write_pin_low(A0);
  283. gpio_write_pin_low(A1);
  284. gpio_write_pin_low(A2);
  285. gpio_write_pin_low(B4);
  286. break;
  287. case 8:
  288. gpio_write_pin_low(A0);
  289. gpio_write_pin_low(A1);
  290. gpio_write_pin_low(A2);
  291. gpio_write_pin_low(C7);
  292. break;
  293. case 9:
  294. gpio_write_pin_high(A0);
  295. gpio_write_pin_high(A1);
  296. gpio_write_pin_high(A2);
  297. gpio_write_pin_low(C7);
  298. break;
  299. case 10:
  300. gpio_write_pin_high(A0);
  301. gpio_write_pin_high(A1);
  302. gpio_write_pin_low(A2);
  303. gpio_write_pin_low(C7);
  304. break;
  305. case 11:
  306. gpio_write_pin_high(A0);
  307. gpio_write_pin_low(A1);
  308. gpio_write_pin_high(A2);
  309. gpio_write_pin_low(C7);
  310. break;
  311. case 12:
  312. gpio_write_pin_high(A0);
  313. gpio_write_pin_low(A1);
  314. gpio_write_pin_low(A2);
  315. gpio_write_pin_low(C7);
  316. break;
  317. case 13:
  318. gpio_write_pin_low(A0);
  319. gpio_write_pin_high(A1);
  320. gpio_write_pin_high(A2);
  321. gpio_write_pin_low(C7);
  322. break;
  323. case 14:
  324. gpio_write_pin_low(A0);
  325. gpio_write_pin_high(A1);
  326. gpio_write_pin_low(A2);
  327. gpio_write_pin_low(C7);
  328. break;
  329. case 15:
  330. gpio_write_pin_low(A0);
  331. gpio_write_pin_low(A1);
  332. gpio_write_pin_high(A2);
  333. gpio_write_pin_low(C7);
  334. break;
  335. case 16:
  336. gpio_write_pin_high(C2);
  337. break;
  338. case 17:
  339. gpio_write_pin_high(C3);
  340. break;
  341. case 18:
  342. gpio_write_pin_high(C5);
  343. break;
  344. }
  345. }
  346. static void unselect_cols(void) {
  347. //Native
  348. gpio_write_pin_high(C2);
  349. gpio_write_pin_high(C3);
  350. gpio_write_pin_high(C5);
  351. //Demultiplexer
  352. gpio_write_pin_low(B4);
  353. gpio_write_pin_low(C7);
  354. gpio_write_pin_high(A0);
  355. gpio_write_pin_high(A1);
  356. gpio_write_pin_high(A2);
  357. }
  358. static void init_pins(void) {
  359. unselect_cols();
  360. for (uint8_t x = 0; x < MATRIX_ROWS; x++) {
  361. gpio_set_pin_input_high(row_pins[x]);
  362. }
  363. gpio_set_pin_output(A0);
  364. gpio_set_pin_output(A1);
  365. gpio_set_pin_output(A2);
  366. gpio_set_pin_output(B4);
  367. gpio_set_pin_output(C7);
  368. gpio_set_pin_output(C2);
  369. gpio_set_pin_output(C3);
  370. gpio_set_pin_output(C5);
  371. }
  372. static bool read_rows_on_col(matrix_row_t current_matrix[], uint8_t current_col) {
  373. bool matrix_changed = false;
  374. // Select col and wait for col selecton to stabilize
  375. select_col(current_col);
  376. wait_us(30);
  377. // For each row...
  378. for (uint8_t row_index = 0; row_index < MATRIX_ROWS; row_index++) {
  379. // Store last value of row prior to reading
  380. matrix_row_t last_row_value = current_matrix[row_index];
  381. // Check row pin state
  382. if (gpio_read_pin(row_pins[row_index]) == 0) {
  383. // Pin LO, set col bit
  384. current_matrix[row_index] |= (MATRIX_ROW_SHIFTER << current_col);
  385. } else {
  386. // Pin HI, clear col bit
  387. current_matrix[row_index] &= ~(MATRIX_ROW_SHIFTER << current_col);
  388. }
  389. // Determine if the matrix changed state
  390. if ((last_row_value != current_matrix[row_index]) && !(matrix_changed)) {
  391. matrix_changed = true;
  392. }
  393. }
  394. // Unselect col
  395. unselect_col(current_col);
  396. return matrix_changed;
  397. }
  398. #endif
  399. void matrix_init_custom(void) {
  400. // initialize key pins
  401. init_pins();
  402. }
  403. bool matrix_scan_custom(matrix_row_t current_matrix[]) {
  404. bool changed = false;
  405. #if defined(DIRECT_PINS) || (DIODE_DIRECTION == COL2ROW)
  406. // Set row, read cols
  407. for (uint8_t current_row = 0; current_row < MATRIX_ROWS; current_row++) {
  408. changed |= read_cols_on_row(current_matrix, current_row);
  409. }
  410. #elif (DIODE_DIRECTION == ROW2COL)
  411. // Set col, read rows
  412. for (uint8_t current_col = 0; current_col < MATRIX_COLS; current_col++) {
  413. changed |= read_rows_on_col(current_matrix, current_col);
  414. }
  415. #endif
  416. return changed;
  417. }