matrix.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #include <stdint.h>
  2. #include <stdbool.h>
  3. #include <string.h>
  4. #include <hal.h>
  5. #include "timer.h"
  6. #include "wait.h"
  7. #include "print.h"
  8. #include "matrix.h"
  9. #include "i2c_master.h"
  10. #include QMK_KEYBOARD_H
  11. #ifndef DEBOUNCE
  12. #define DEBOUNCE 10
  13. #endif
  14. //#define DEBUG_MATRIX_SCAN_RATE
  15. //#ifdef DEBUG_MATRIX_SCAN_RATE
  16. //uint32_t matrix_timer;
  17. //uint32_t matrix_scan_count;
  18. //#endif
  19. static uint8_t mcp23017_reset_loop = 0;
  20. volatile matrix_row_t matrix[MATRIX_ROWS];
  21. volatile matrix_row_t raw_matrix[MATRIX_ROWS];
  22. volatile uint8_t debounce_matrix[MATRIX_ROWS * MATRIX_COLS];
  23. static matrix_row_t read_cols(uint8_t row);
  24. static void init_cols(void);
  25. static void unselect_rows(void);
  26. static void select_row(uint8_t row);
  27. static void init_rows(void);
  28. __attribute__((weak)) void matrix_init_user(void) {}
  29. __attribute__((weak)) void matrix_scan_user(void) {}
  30. __attribute__((weak)) void matrix_init_kb(void) {
  31. matrix_init_user();
  32. }
  33. __attribute__((weak)) void matrix_scan_kb(void) {
  34. matrix_scan_user();
  35. }
  36. void matrix_init(void) {
  37. mcp23017_status = init_mcp23017();
  38. (void) mcp23017_reset_loop;
  39. init_rows();
  40. unselect_rows();
  41. init_cols();
  42. for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
  43. matrix[i] = 0;
  44. raw_matrix[i] = 0;
  45. for (uint8_t j = 0; j < MATRIX_COLS; ++j) {
  46. debounce_matrix[i * MATRIX_COLS + j] = 0;
  47. }
  48. }
  49. matrix_init_quantum();
  50. }
  51. void matrix_power_up(void) {
  52. mcp23017_status = init_mcp23017();
  53. init_rows();
  54. unselect_rows();
  55. init_cols();
  56. for (uint8_t i = 0; i < MATRIX_ROWS; i++) {
  57. matrix[i] = 0;
  58. }
  59. }
  60. matrix_row_t debounce_mask(matrix_row_t rawcols, uint8_t row) {
  61. matrix_row_t result = 0;
  62. matrix_row_t change = rawcols ^raw_matrix[row];
  63. raw_matrix[row] = rawcols;
  64. for (uint8_t i = 0; i < MATRIX_COLS; ++i) {
  65. if (debounce_matrix[row * MATRIX_COLS + i]) {
  66. --debounce_matrix[row * MATRIX_COLS + i];
  67. } else {
  68. result |= (1 << i);
  69. }
  70. if (change & (1 << i)) {
  71. debounce_matrix[row * MATRIX_COLS + i] = DEBOUNCE;
  72. }
  73. }
  74. return result;
  75. }
  76. matrix_row_t debounce_read_cols(uint8_t row) {
  77. // Read the row without debouncing filtering and store it for later usage.
  78. matrix_row_t cols = read_cols(row);
  79. // Get the Debounce mask.
  80. matrix_row_t mask = debounce_mask(cols, row);
  81. // debounce the row and return the result.
  82. return (cols & mask) | (matrix[row] & ~mask);;
  83. }
  84. uint8_t matrix_scan(void) {
  85. if (mcp23017_status) {
  86. if (++mcp23017_reset_loop == 0) {
  87. mcp23017_status = init_mcp23017();
  88. if (!mcp23017_status) {
  89. ergodox_blink_all_leds();
  90. }
  91. }
  92. }
  93. for (uint8_t i = 0; i < MATRIX_ROWS_PER_SIDE; i++) {
  94. select_row(i);
  95. select_row(i + MATRIX_ROWS_PER_SIDE);
  96. matrix[i] = debounce_read_cols(i);
  97. matrix[i + MATRIX_ROWS_PER_SIDE] = debounce_read_cols(i + MATRIX_ROWS_PER_SIDE);
  98. unselect_rows();
  99. }
  100. matrix_scan_quantum();
  101. return 0;
  102. }
  103. bool matrix_is_modified(void) {
  104. return true;
  105. }
  106. inline
  107. bool matrix_is_on(uint8_t row, uint8_t col) {
  108. return (matrix[row] & (1 << col));
  109. }
  110. inline
  111. matrix_row_t matrix_get_row(uint8_t row) {
  112. return matrix[row];
  113. }
  114. void matrix_print(void) {
  115. }
  116. static matrix_row_t read_cols(uint8_t row) {
  117. if (row < MATRIX_ROWS_PER_SIDE) {
  118. uint8_t data = 0xFF;
  119. if (!mcp23017_status) {
  120. uint8_t regAddr = I2C_GPIOB;
  121. mcp23017_status = i2c_readReg(I2C_ADDR, regAddr, &data, 1, 10);
  122. }
  123. if (mcp23017_status) {
  124. return 0;
  125. }
  126. return (~data) & 0x3F;
  127. } else {
  128. uint8_t data_p = (GPIOB -> IDR);
  129. uint8_t data = data_p;
  130. return ((~data) & 0x3f);
  131. }
  132. }
  133. static void init_cols(void) {
  134. palSetPadMode(GPIOB, 0, PAL_MODE_INPUT_PULLUP);
  135. palSetPadMode(GPIOB, 1, PAL_MODE_INPUT_PULLUP);
  136. palSetPadMode(GPIOB, 2, PAL_MODE_INPUT_PULLUP);
  137. palSetPadMode(GPIOB, 3, PAL_MODE_INPUT_PULLUP);
  138. palSetPadMode(GPIOB, 4, PAL_MODE_INPUT_PULLUP);
  139. palSetPadMode(GPIOB, 5, PAL_MODE_INPUT_PULLUP);
  140. }
  141. static void init_rows(void) {
  142. palSetPadMode(GPIOB, 8, PAL_MODE_OUTPUT_PUSHPULL);
  143. palSetPadMode(GPIOB, 9, PAL_MODE_OUTPUT_PUSHPULL);
  144. palSetPadMode(GPIOB, 10, PAL_MODE_OUTPUT_PUSHPULL);
  145. palSetPadMode(GPIOB, 11, PAL_MODE_OUTPUT_PUSHPULL);
  146. palSetPadMode(GPIOB, 12, PAL_MODE_OUTPUT_PUSHPULL);
  147. palSetPadMode(GPIOB, 13, PAL_MODE_OUTPUT_PUSHPULL);
  148. palSetPadMode(GPIOB, 14, PAL_MODE_OUTPUT_PUSHPULL);
  149. }
  150. static void unselect_rows(void) {
  151. GPIOB->BSRR = 0b1111111 << 8;
  152. }
  153. static void select_row(uint8_t row) {
  154. if (row < MATRIX_ROWS_PER_SIDE) {
  155. if (!mcp23017_status) {
  156. uint8_t data = (0xFF & ~(1 << row));
  157. mcp23017_status = i2c_writeReg(I2C_ADDR, I2C_GPIOA, &data, 1, 10);
  158. }
  159. } else {
  160. GPIOB->BRR = 0x1 << (row+1);
  161. }
  162. }