transport.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. #include <string.h>
  2. #include <stddef.h>
  3. #include "config.h"
  4. #include "matrix.h"
  5. #include "quantum.h"
  6. #define ROWS_PER_HAND (MATRIX_ROWS / 2)
  7. #ifdef RGBLIGHT_ENABLE
  8. #include "rgblight.h"
  9. #endif
  10. #ifdef BACKLIGHT_ENABLE
  11. #include "backlight.h"
  12. #endif
  13. #ifdef ENCODER_ENABLE
  14. #include "encoder.h"
  15. #endif
  16. #if defined(USE_I2C) || defined(EH)
  17. #include "i2c_master.h"
  18. #include "i2c_slave.h"
  19. typedef struct _I2C_slave_buffer_t {
  20. matrix_row_t smatrix[ROWS_PER_HAND];
  21. uint8_t backlight_level;
  22. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  23. rgblight_syncinfo_t rgblight_sync;
  24. #endif
  25. #ifdef ENCODER_ENABLE
  26. uint8_t encoder_state[NUMBER_OF_ENCODERS];
  27. #endif
  28. } I2C_slave_buffer_t;
  29. static I2C_slave_buffer_t *const i2c_buffer = (I2C_slave_buffer_t *)i2c_slave_reg;
  30. #define I2C_BACKLIGHT_START offsetof(I2C_slave_buffer_t, backlight_level)
  31. #define I2C_RGB_START offsetof(I2C_slave_buffer_t, rgblight_sync)
  32. #define I2C_KEYMAP_START offsetof(I2C_slave_buffer_t, smatrix)
  33. #define I2C_ENCODER_START offsetof(I2C_slave_buffer_t, encoder_state)
  34. #define TIMEOUT 100
  35. #ifndef SLAVE_I2C_ADDRESS
  36. #define SLAVE_I2C_ADDRESS 0x32
  37. #endif
  38. // Get rows from other half over i2c
  39. bool transport_master(matrix_row_t matrix[]) {
  40. i2c_readReg(SLAVE_I2C_ADDRESS, I2C_KEYMAP_START, (void *)matrix, sizeof(i2c_buffer->smatrix), TIMEOUT);
  41. // write backlight info
  42. #ifdef BACKLIGHT_ENABLE
  43. uint8_t level = is_backlight_enabled() ? get_backlight_level() : 0;
  44. if (level != i2c_buffer->backlight_level) {
  45. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_BACKLIGHT_START, (void *)&level, sizeof(level), TIMEOUT) >= 0) {
  46. i2c_buffer->backlight_level = level;
  47. }
  48. }
  49. #endif
  50. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  51. if (rgblight_get_change_flags()) {
  52. rgblight_syncinfo_t rgblight_sync;
  53. rgblight_get_syncinfo(&rgblight_sync);
  54. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_RGB_START, (void *)&rgblight_sync, sizeof(rgblight_sync), TIMEOUT) >= 0) {
  55. rgblight_clear_change_flags();
  56. }
  57. }
  58. #endif
  59. #ifdef ENCODER_ENABLE
  60. i2c_readReg(SLAVE_I2C_ADDRESS, I2C_ENCODER_START, (void *)i2c_buffer->encoder_state, sizeof(i2c_buffer->encoder_state), TIMEOUT);
  61. encoder_update_raw(i2c_buffer->encoder_state);
  62. #endif
  63. return true;
  64. }
  65. void transport_slave(matrix_row_t matrix[]) {
  66. // Copy matrix to I2C buffer
  67. memcpy((void *)i2c_buffer->smatrix, (void *)matrix, sizeof(i2c_buffer->smatrix));
  68. // Read Backlight Info
  69. #ifdef BACKLIGHT_ENABLE
  70. backlight_set(i2c_buffer->backlight_level);
  71. #endif
  72. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  73. // Update the RGB with the new data
  74. if (i2c_buffer->rgblight_sync.status.change_flags != 0) {
  75. rgblight_update_sync(&i2c_buffer->rgblight_sync, false);
  76. i2c_buffer->rgblight_sync.status.change_flags = 0;
  77. }
  78. #endif
  79. #ifdef ENCODER_ENABLE
  80. encoder_state_raw(i2c_buffer->encoder_state);
  81. #endif
  82. }
  83. void transport_master_init(void) { i2c_init(); }
  84. void transport_slave_init(void) { i2c_slave_init(SLAVE_I2C_ADDRESS); }
  85. #else // USE_SERIAL
  86. #include "serial.h"
  87. typedef struct _Serial_s2m_buffer_t {
  88. // TODO: if MATRIX_COLS > 8 change to uint8_t packed_matrix[] for pack/unpack
  89. matrix_row_t smatrix[ROWS_PER_HAND];
  90. #ifdef ENCODER_ENABLE
  91. uint8_t encoder_state[NUMBER_OF_ENCODERS];
  92. #endif
  93. } Serial_s2m_buffer_t;
  94. typedef struct _Serial_m2s_buffer_t {
  95. #ifdef BACKLIGHT_ENABLE
  96. uint8_t backlight_level;
  97. #endif
  98. } Serial_m2s_buffer_t;
  99. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  100. // When MCUs on both sides drive their respective RGB LED chains,
  101. // it is necessary to synchronize, so it is necessary to communicate RGB
  102. // information. In that case, define RGBLIGHT_SPLIT with info on the number
  103. // of LEDs on each half.
  104. //
  105. // Otherwise, if the master side MCU drives both sides RGB LED chains,
  106. // there is no need to communicate.
  107. typedef struct _Serial_rgblight_t {
  108. rgblight_syncinfo_t rgblight_sync;
  109. } Serial_rgblight_t;
  110. volatile Serial_rgblight_t serial_rgblight = {};
  111. uint8_t volatile status_rgblight = 0;
  112. #endif
  113. volatile Serial_s2m_buffer_t serial_s2m_buffer = {};
  114. volatile Serial_m2s_buffer_t serial_m2s_buffer = {};
  115. uint8_t volatile status0 = 0;
  116. enum serial_transaction_id {
  117. GET_SLAVE_MATRIX = 0,
  118. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  119. PUT_RGBLIGHT,
  120. #endif
  121. };
  122. SSTD_t transactions[] = {
  123. [GET_SLAVE_MATRIX] =
  124. {
  125. (uint8_t *)&status0,
  126. sizeof(serial_m2s_buffer),
  127. (uint8_t *)&serial_m2s_buffer,
  128. sizeof(serial_s2m_buffer),
  129. (uint8_t *)&serial_s2m_buffer,
  130. },
  131. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  132. [PUT_RGBLIGHT] =
  133. {
  134. (uint8_t *)&status_rgblight, sizeof(serial_rgblight), (uint8_t *)&serial_rgblight, 0, NULL // no slave to master transfer
  135. },
  136. #endif
  137. };
  138. void transport_master_init(void) { soft_serial_initiator_init(transactions, TID_LIMIT(transactions)); }
  139. void transport_slave_init(void) { soft_serial_target_init(transactions, TID_LIMIT(transactions)); }
  140. #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  141. // rgblight synchronization information communication.
  142. void transport_rgblight_master(void) {
  143. if (rgblight_get_change_flags()) {
  144. rgblight_get_syncinfo((rgblight_syncinfo_t *)&serial_rgblight.rgblight_sync);
  145. if (soft_serial_transaction(PUT_RGBLIGHT) == TRANSACTION_END) {
  146. rgblight_clear_change_flags();
  147. }
  148. }
  149. }
  150. void transport_rgblight_slave(void) {
  151. if (status_rgblight == TRANSACTION_ACCEPTED) {
  152. rgblight_update_sync((rgblight_syncinfo_t *)&serial_rgblight.rgblight_sync, false);
  153. status_rgblight = TRANSACTION_END;
  154. }
  155. }
  156. #else
  157. #define transport_rgblight_master()
  158. #define transport_rgblight_slave()
  159. #endif
  160. bool transport_master(matrix_row_t matrix[]) {
  161. #ifndef SERIAL_USE_MULTI_TRANSACTION
  162. if (soft_serial_transaction() != TRANSACTION_END) {
  163. return false;
  164. }
  165. #else
  166. transport_rgblight_master();
  167. if (soft_serial_transaction(GET_SLAVE_MATRIX) != TRANSACTION_END) {
  168. return false;
  169. }
  170. #endif
  171. // TODO: if MATRIX_COLS > 8 change to unpack()
  172. for (int i = 0; i < ROWS_PER_HAND; ++i) {
  173. matrix[i] = serial_s2m_buffer.smatrix[i];
  174. }
  175. #ifdef BACKLIGHT_ENABLE
  176. // Write backlight level for slave to read
  177. serial_m2s_buffer.backlight_level = is_backlight_enabled() ? get_backlight_level() : 0;
  178. #endif
  179. #ifdef ENCODER_ENABLE
  180. encoder_update_raw((uint8_t *)serial_s2m_buffer.encoder_state);
  181. #endif
  182. return true;
  183. }
  184. void transport_slave(matrix_row_t matrix[]) {
  185. transport_rgblight_slave();
  186. // TODO: if MATRIX_COLS > 8 change to pack()
  187. for (int i = 0; i < ROWS_PER_HAND; ++i) {
  188. serial_s2m_buffer.smatrix[i] = matrix[i];
  189. }
  190. #ifdef BACKLIGHT_ENABLE
  191. backlight_set(serial_m2s_buffer.backlight_level);
  192. #endif
  193. #ifdef ENCODER_ENABLE
  194. encoder_state_raw((uint8_t *)serial_s2m_buffer.encoder_state);
  195. #endif
  196. }
  197. #endif