You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

189 lines
4.4 KiB

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