matrix.c 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. /* Copyright 2018 Maarten Dekkers <maartenwut@gmail.com>
  2. *
  3. * This program is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  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. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include <stdint.h>
  17. #include <stdbool.h>
  18. #if defined(__AVR__)
  19. #include <avr/io.h>
  20. #endif
  21. #include "wait.h"
  22. #include "print.h"
  23. #include "debug.h"
  24. #include "util.h"
  25. #include "matrix.h"
  26. #include "timer.h"
  27. /* Set 0 if debouncing isn't needed */
  28. #ifndef DEBOUNCING_DELAY
  29. # define DEBOUNCING_DELAY 5
  30. #endif
  31. #define COL_SHIFTER ((uint32_t)1)
  32. static uint16_t debouncing_time;
  33. static bool debouncing = false;
  34. static const uint8_t row_pins[MATRIX_ROWS] = MATRIX_ROW_PINS;
  35. /* matrix state(1:on, 0:off) */
  36. static matrix_row_t matrix[MATRIX_ROWS];
  37. static matrix_row_t matrix_debouncing[MATRIX_ROWS];
  38. static void init_rows(void);
  39. static bool read_rows_on_col(matrix_row_t current_matrix[], uint8_t current_col);
  40. static void unselect_cols(void);
  41. static void select_col(uint8_t col);
  42. inline
  43. uint8_t matrix_rows(void) {
  44. return MATRIX_ROWS;
  45. }
  46. inline
  47. uint8_t matrix_cols(void) {
  48. return MATRIX_COLS;
  49. }
  50. void matrix_init(void) {
  51. unselect_cols();
  52. init_rows();
  53. // initialize matrix state: all keys off
  54. for (uint8_t i=0; i < MATRIX_ROWS; i++) {
  55. matrix[i] = 0;
  56. matrix_debouncing[i] = 0;
  57. }
  58. matrix_init_quantum();
  59. }
  60. uint8_t matrix_scan(void)
  61. {
  62. // Set col, read rows
  63. for (uint8_t current_col = 0; current_col < MATRIX_COLS; current_col++) {
  64. bool matrix_changed = read_rows_on_col(matrix_debouncing, current_col);
  65. if (matrix_changed) {
  66. debouncing = true;
  67. debouncing_time = timer_read();
  68. }
  69. }
  70. if (debouncing && (timer_elapsed(debouncing_time) > DEBOUNCING_DELAY)) {
  71. for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
  72. matrix[i] = matrix_debouncing[i];
  73. }
  74. debouncing = false;
  75. }
  76. matrix_scan_quantum();
  77. return 1;
  78. }
  79. inline
  80. bool matrix_is_on(uint8_t row, uint8_t col)
  81. {
  82. return (matrix[row] & ((matrix_row_t)1<col));
  83. }
  84. inline
  85. matrix_row_t matrix_get_row(uint8_t row)
  86. {
  87. return matrix[row];
  88. }
  89. void matrix_print(void)
  90. {
  91. print("\nr/c 0123456789ABCDEFGHIJKLMNOPQRSTUV \n");
  92. for (uint8_t row = 0; row < MATRIX_ROWS; row++) {
  93. phex(row); print(": ");
  94. print_bin_reverse32(matrix_get_row(row));
  95. print("\n");
  96. }
  97. }
  98. uint8_t matrix_key_count(void)
  99. {
  100. uint8_t count = 0;
  101. for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
  102. count += bitpop32(matrix[i]);
  103. }
  104. return count;
  105. }
  106. static bool read_rows_on_col(matrix_row_t current_matrix[], uint8_t current_col)
  107. {
  108. bool matrix_changed = false;
  109. // Select col and wait for col selecton to stabilize
  110. select_col(current_col);
  111. wait_us(30);
  112. // For each row...
  113. for(uint8_t row_index = 0; row_index < MATRIX_ROWS; row_index++)
  114. {
  115. // Store last value of row prior to reading
  116. matrix_row_t last_row_value = current_matrix[row_index];
  117. // Check row pin state
  118. // Use the otherwise unused row: 3, col: 0 for caps lock
  119. if (row_index == 2 && current_col == 2) {
  120. // Pin E2 uses active low
  121. if ((_SFR_IO8(E2 >> 4) & _BV(E2 & 0xF)) == 0)
  122. {
  123. // Pin LO, set col bit
  124. current_matrix[row_index] |= (COL_SHIFTER << current_col);
  125. }
  126. else
  127. {
  128. // Pin HI, clear col bit
  129. current_matrix[row_index] &= ~(COL_SHIFTER << current_col);
  130. }
  131. }
  132. else {
  133. if ((_SFR_IO8(row_pins[row_index] >> 4) & _BV(row_pins[row_index] & 0xF)))
  134. {
  135. // Pin HI, set col bit
  136. current_matrix[row_index] |= (COL_SHIFTER << current_col);
  137. }
  138. else
  139. {
  140. // Pin LO, clear col bit
  141. current_matrix[row_index] &= ~(COL_SHIFTER << current_col);
  142. }
  143. }
  144. // Determine if the matrix changed state
  145. if ((last_row_value != current_matrix[row_index]) && !(matrix_changed))
  146. {
  147. matrix_changed = true;
  148. }
  149. }
  150. // Unselect cols
  151. unselect_cols();
  152. return matrix_changed;
  153. }
  154. /* Row pin configuration
  155. * row: 0 1 2 3 4
  156. * pin: D0 D1 D2 D3 D5
  157. *
  158. * Caps lock uses its own pin E2
  159. */
  160. static void init_rows(void) {
  161. DDRD &= ~((1<<0)| (1<<1) | (1<<2) | (1<<3) | (1<<5)); // IN
  162. PORTD &= ~((1<<0)| (1<<1) | (1<<2) | (1<<3) | (1<<5)); // LO
  163. DDRE &= ~(1<<2); // IN
  164. PORTE |= (1<<2); // HI
  165. }
  166. /* Columns 0 - 16
  167. * col 0: B5
  168. * col 1: B6
  169. * These columns use a 74HC237D 3 to 8 bit demultiplexer.
  170. * A B C GL1
  171. * col / pin: PF0 PF1 PC7 PC6
  172. * 2: 0 0 0 1
  173. * 3: 1 0 0 1
  174. * 4: 0 1 0 1
  175. * 5: 1 1 0 1
  176. * 6: 0 0 1 1
  177. * 7: 1 0 1 1
  178. * 8: 0 1 1 1
  179. * 9: 1 1 1 1
  180. * col 10: E6
  181. * col 11: B0
  182. * col 12: B7
  183. * col 13: D4
  184. * col 14: D6
  185. * col 15: D7
  186. * col 16: B4
  187. */
  188. static void unselect_cols(void) {
  189. DDRB |= (1<<5) | (1<<6) | (1<<0) | (1<<7) | (1<<4); // OUT
  190. PORTB &= ~((1<<5) | (1<<6) | (1<<0) | (1<<7) | (1<<4)); // LO
  191. DDRD |= (1<<4) | (1<<6) | (1<<7); // OUT
  192. PORTD &= ~((1<<4) | (1<<6) | (1<<7)); // LO
  193. DDRE |= (1<<6); // OUT
  194. PORTE &= ~((1<<6)); // LO
  195. DDRF |= (1<<0) | (1<<1); // OUT
  196. PORTF &= ~((1<<0) | (1<<1)); // LO
  197. DDRC |= (1<<7) | (1<<6); // OUT
  198. PORTC &= ~((1<<7) | (1<<6)); // LO
  199. }
  200. static void select_col(uint8_t col)
  201. {
  202. switch (col) {
  203. case 0:
  204. PORTB |= (1<<5); // HI
  205. break;
  206. case 1:
  207. PORTB |= (1<<6); // HI
  208. break;
  209. case 2:
  210. PORTC |= (1<<6); // HI
  211. break;
  212. case 3:
  213. PORTC |= (1<<6); // HI
  214. PORTF |= (1<<0); // HI
  215. break;
  216. case 4:
  217. PORTC |= (1<<6); // HI
  218. PORTF |= (1<<1); // HI
  219. break;
  220. case 5:
  221. PORTC |= (1<<6); // HI
  222. PORTF |= (1<<0); // HI
  223. PORTF |= (1<<1); // HI
  224. break;
  225. case 6:
  226. PORTC |= (1<<6); // HI
  227. PORTC |= (1<<7); // HI
  228. break;
  229. case 7:
  230. PORTC |= (1<<6); // HI
  231. PORTF |= (1<<0); // HI
  232. PORTC |= (1<<7); // HI
  233. break;
  234. case 8:
  235. PORTC |= (1<<6); // HI
  236. PORTF |= (1<<1); // HI
  237. PORTC |= (1<<7); // HI
  238. break;
  239. case 9:
  240. PORTC |= (1<<6); // HI
  241. PORTF |= (1<<0); // HI
  242. PORTF |= (1<<1); // HI
  243. PORTC |= (1<<7); // HI
  244. break;
  245. case 10:
  246. PORTE |= (1<<6); // HI
  247. break;
  248. case 11:
  249. PORTB |= (1<<0); // HI
  250. break;
  251. case 12:
  252. PORTB |= (1<<7); // HI
  253. break;
  254. case 13:
  255. PORTD |= (1<<4); // HI
  256. break;
  257. case 14:
  258. PORTD |= (1<<6); // HI
  259. break;
  260. case 15:
  261. PORTD |= (1<<7); // HI
  262. break;
  263. case 16:
  264. PORTB |= (1<<4); // HI
  265. break;
  266. }
  267. }