8 #define ROWS_PER_HAND (MATRIX_ROWS / 2)
10 #ifdef RGBLIGHT_ENABLE
11 # include "rgblight.h"
14 #ifdef BACKLIGHT_ENABLE
15 # include "backlight.h"
16 extern backlight_config_t backlight_config;
23 #if defined(USE_I2C) || defined(EH)
25 # include "i2c_master.h"
26 # include "i2c_slave.h"
28 typedef struct _I2C_slave_buffer_t {
29 matrix_row_t smatrix[ROWS_PER_HAND];
30 uint8_t backlight_level;
31 #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
32 rgblight_syncinfo_t rgblight_sync;
35 uint8_t encoder_state[NUMBER_OF_ENCODERS];
39 static I2C_slave_buffer_t * const i2c_buffer = (I2C_slave_buffer_t *)i2c_slave_reg;
41 # define I2C_BACKLIGHT_START offsetof(I2C_slave_buffer_t, backlight_level)
42 # define I2C_RGB_START offsetof(I2C_slave_buffer_t, rgblight_sync)
43 # define I2C_KEYMAP_START offsetof(I2C_slave_buffer_t, smatrix)
44 # define I2C_ENCODER_START offsetof(I2C_slave_buffer_t, encoder_state)
48 # ifndef SLAVE_I2C_ADDRESS
49 # define SLAVE_I2C_ADDRESS 0x32
52 // Get rows from other half over i2c
53 bool transport_master(matrix_row_t matrix[]) {
54 i2c_readReg(SLAVE_I2C_ADDRESS, I2C_KEYMAP_START, (void *)matrix, sizeof(i2c_buffer->smatrix), TIMEOUT);
56 // write backlight info
57 # ifdef BACKLIGHT_ENABLE
58 uint8_t level = get_backlight_level();
59 if (level != i2c_buffer->backlight_level) {
60 if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_BACKLIGHT_START, (void *)&level, sizeof(level), TIMEOUT) >= 0) {
61 i2c_buffer->backlight_level = level;
66 # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
67 if (rgblight_get_change_flags()) {
68 rgblight_syncinfo_t rgblight_sync;
69 rgblight_get_syncinfo(&rgblight_sync);
70 if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_RGB_START,
71 (void *)&rgblight_sync, sizeof(rgblight_sync), TIMEOUT) >= 0) {
72 rgblight_clear_change_flags();
77 # ifdef ENCODER_ENABLE
78 i2c_readReg(SLAVE_I2C_ADDRESS, I2C_ENCODER_START, (void *)i2c_buffer->encoder_state, sizeof(I2C_slave_buffer_t.encoder_state), TIMEOUT);
79 encoder_update_raw(i2c_buffer->encoder_state);
85 void transport_slave(matrix_row_t matrix[]) {
86 // Copy matrix to I2C buffer
87 memcpy((void*)i2c_buffer->smatrix, (void *)matrix, sizeof(i2c_buffer->smatrix));
89 // Read Backlight Info
90 # ifdef BACKLIGHT_ENABLE
91 backlight_set(i2c_buffer->backlight_level);
94 # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
95 // Update the RGB with the new data
96 if (i2c_buffer->rgblight_sync.status.change_flags != 0) {
97 rgblight_update_sync(&i2c_buffer->rgblight_sync, false);
98 i2c_buffer->rgblight_sync.status.change_flags = 0;
102 # ifdef ENCODER_ENABLE
103 encoder_state_raw(i2c_buffer->encoder_state);
107 void transport_master_init(void) { i2c_init(); }
109 void transport_slave_init(void) { i2c_slave_init(SLAVE_I2C_ADDRESS); }
115 typedef struct _Serial_s2m_buffer_t {
116 // TODO: if MATRIX_COLS > 8 change to uint8_t packed_matrix[] for pack/unpack
117 matrix_row_t smatrix[ROWS_PER_HAND];
119 # ifdef ENCODER_ENABLE
120 uint8_t encoder_state[NUMBER_OF_ENCODERS];
123 } Serial_s2m_buffer_t;
125 typedef struct _Serial_m2s_buffer_t {
126 # ifdef BACKLIGHT_ENABLE
127 uint8_t backlight_level;
129 } Serial_m2s_buffer_t;
131 #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
132 // When MCUs on both sides drive their respective RGB LED chains,
133 // it is necessary to synchronize, so it is necessary to communicate RGB
134 // information. In that case, define RGBLIGHT_SPLIT with info on the number
135 // of LEDs on each half.
137 // Otherwise, if the master side MCU drives both sides RGB LED chains,
138 // there is no need to communicate.
140 typedef struct _Serial_rgblight_t {
141 rgblight_syncinfo_t rgblight_sync;
144 volatile Serial_rgblight_t serial_rgblight = {};
145 uint8_t volatile status_rgblight = 0;
148 volatile Serial_s2m_buffer_t serial_s2m_buffer = {};
149 volatile Serial_m2s_buffer_t serial_m2s_buffer = {};
150 uint8_t volatile status0 = 0;
152 enum serial_transaction_id {
153 GET_SLAVE_MATRIX = 0,
154 #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
159 SSTD_t transactions[] = {
160 [GET_SLAVE_MATRIX] = {
162 sizeof(serial_m2s_buffer),
163 (uint8_t *)&serial_m2s_buffer,
164 sizeof(serial_s2m_buffer),
165 (uint8_t *)&serial_s2m_buffer,
167 #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
169 (uint8_t *)&status_rgblight,
170 sizeof(serial_rgblight),
171 (uint8_t *)&serial_rgblight,
172 0, NULL // no slave to master transfer
177 void transport_master_init(void) { soft_serial_initiator_init(transactions, TID_LIMIT(transactions)); }
179 void transport_slave_init(void) { soft_serial_target_init(transactions, TID_LIMIT(transactions)); }
181 #if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
183 // rgblight synchronization information communication.
185 void transport_rgblight_master(void) {
186 if (rgblight_get_change_flags()) {
187 rgblight_get_syncinfo((rgblight_syncinfo_t *)&serial_rgblight.rgblight_sync);
188 if (soft_serial_transaction(PUT_RGBLIGHT) == TRANSACTION_END) {
189 rgblight_clear_change_flags();
194 void transport_rgblight_slave(void) {
195 if (status_rgblight == TRANSACTION_ACCEPTED) {
196 rgblight_update_sync((rgblight_syncinfo_t *)&serial_rgblight.rgblight_sync,
198 status_rgblight = TRANSACTION_END;
203 #define transport_rgblight_master()
204 #define transport_rgblight_slave()
207 bool transport_master(matrix_row_t matrix[]) {
208 #ifndef SERIAL_USE_MULTI_TRANSACTION
209 if (soft_serial_transaction() != TRANSACTION_END) {
213 transport_rgblight_master();
214 if (soft_serial_transaction(GET_SLAVE_MATRIX) != TRANSACTION_END) {
219 // TODO: if MATRIX_COLS > 8 change to unpack()
220 for (int i = 0; i < ROWS_PER_HAND; ++i) {
221 matrix[i] = serial_s2m_buffer.smatrix[i];
224 # ifdef BACKLIGHT_ENABLE
225 // Write backlight level for slave to read
226 serial_m2s_buffer.backlight_level = backlight_config.enable ? backlight_config.level : 0;
229 # ifdef ENCODER_ENABLE
230 encoder_update_raw((uint8_t *)serial_s2m_buffer.encoder_state);
236 void transport_slave(matrix_row_t matrix[]) {
237 transport_rgblight_slave();
238 // TODO: if MATRIX_COLS > 8 change to pack()
239 for (int i = 0; i < ROWS_PER_HAND; ++i) {
240 serial_s2m_buffer.smatrix[i] = matrix[i];
242 # ifdef BACKLIGHT_ENABLE
243 backlight_set(serial_m2s_buffer.backlight_level);
246 # ifdef ENCODER_ENABLE
247 encoder_state_raw((uint8_t *)serial_s2m_buffer.encoder_state);