drashna_transport.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603
  1. /* Copyright 2020 Christopher Courtney, aka Drashna Jael're (@drashna) <drashna@live.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 <string.h>
  17. #include <stddef.h>
  18. #include "matrix.h"
  19. #include QMK_KEYBOARD_H
  20. #define ROWS_PER_HAND (MATRIX_ROWS / 2)
  21. #define SYNC_TIMER_OFFSET 2
  22. #ifdef RGBLIGHT_ENABLE
  23. # include "rgblight.h"
  24. #endif
  25. #ifdef BACKLIGHT_ENABLE
  26. # include "backlight.h"
  27. #endif
  28. #ifdef ENCODER_ENABLE
  29. # include "encoder.h"
  30. static pin_t encoders_pad[] = ENCODERS_PAD_A;
  31. # define NUMBER_OF_ENCODERS (sizeof(encoders_pad) / sizeof(pin_t))
  32. #endif
  33. #ifdef POINTING_DEVICE_ENABLE
  34. static uint16_t device_cpi = 0;
  35. static int8_t split_mouse_x = 0, split_mouse_y = 0;
  36. #endif
  37. #ifdef OLED_DRIVER_ENABLE
  38. # include "oled_driver.h"
  39. #endif
  40. #if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  41. # include "led_matrix.h"
  42. #endif
  43. #if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  44. # include "rgb_matrix.h"
  45. #endif
  46. #if defined(USE_I2C)
  47. # include "i2c_master.h"
  48. # include "i2c_slave.h"
  49. typedef struct _I2C_slave_buffer_t {
  50. # ifndef DISABLE_SYNC_TIMER
  51. uint32_t sync_timer;
  52. # endif
  53. # ifdef SPLIT_TRANSPORT_MIRROR
  54. matrix_row_t mmatrix[ROWS_PER_HAND];
  55. # endif
  56. matrix_row_t smatrix[ROWS_PER_HAND];
  57. # ifdef SPLIT_MODS_ENABLE
  58. uint8_t real_mods;
  59. uint8_t weak_mods;
  60. # ifndef NO_ACTION_ONESHOT
  61. uint8_t oneshot_mods;
  62. # endif
  63. # endif
  64. # ifdef BACKLIGHT_ENABLE
  65. uint8_t backlight_level;
  66. # endif
  67. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  68. rgblight_syncinfo_t rgblight_sync;
  69. # endif
  70. # ifdef ENCODER_ENABLE
  71. uint8_t encoder_state[NUMBER_OF_ENCODERS];
  72. # endif
  73. # ifdef WPM_ENABLE
  74. uint8_t current_wpm;
  75. # endif
  76. # if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  77. led_eeconfig_t led_matrix;
  78. bool led_suspend_state;
  79. # endif
  80. # if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  81. rgb_config_t rgb_matrix;
  82. bool rgb_suspend_state;
  83. # endif
  84. int8_t mouse_x;
  85. int8_t mouse_y;
  86. uint16_t device_cpi;
  87. bool oled_on;
  88. layer_state_t t_layer_state;
  89. layer_state_t t_default_layer_state;
  90. } __attribute__((packed)) I2C_slave_buffer_t;
  91. static I2C_slave_buffer_t *const i2c_buffer = (I2C_slave_buffer_t *)i2c_slave_reg;
  92. # define I2C_BACKLIGHT_START offsetof(I2C_slave_buffer_t, backlight_level)
  93. # define I2C_RGB_START offsetof(I2C_slave_buffer_t, rgblight_sync)
  94. # define I2C_KEYMAP_MASTER_START offsetof(I2C_slave_buffer_t, mmatrix)
  95. # define I2C_KEYMAP_SLAVE_START offsetof(I2C_slave_buffer_t, smatrix)
  96. # define I2C_SYNC_TIME_START offsetof(I2C_slave_buffer_t, sync_timer)
  97. # define I2C_REAL_MODS_START offsetof(I2C_slave_buffer_t, real_mods)
  98. # define I2C_WEAK_MODS_START offsetof(I2C_slave_buffer_t, weak_mods)
  99. # define I2C_ONESHOT_MODS_START offsetof(I2C_slave_buffer_t, oneshot_mods)
  100. # define I2C_ENCODER_START offsetof(I2C_slave_buffer_t, encoder_state)
  101. # define I2C_WPM_START offsetof(I2C_slave_buffer_t, current_wpm)
  102. # define I2C_MOUSE_X_START offsetof(I2C_slave_buffer_t, mouse_x)
  103. # define I2C_MOUSE_Y_START offsetof(I2C_slave_buffer_t, mouse_y)
  104. # define I2C_MOUSE_DPI_START offsetof(I2C_slave_buffer_t, device_cpi)
  105. # define I2C_OLED_ON_START offsetof(I2C_slave_buffer_t, oled_on)
  106. # define I2C_LAYER_STATE_START offsetof(I2C_slave_buffer_t, t_layer_state)
  107. # define I2C_DEFAULT_LAYER_STATE_START offsetof(I2C_slave_buffer_t, t_default_layer_state)
  108. # define I2C_LED_MATRIX_START offsetof(I2C_slave_buffer_t, led_matrix)
  109. # define I2C_LED_SUSPEND_START offsetof(I2C_slave_buffer_t, led_suspend_state)
  110. # define I2C_RGB_MATRIX_START offsetof(I2C_slave_buffer_t, rgb_matrix)
  111. # define I2C_RGB_SUSPEND_START offsetof(I2C_slave_buffer_t, rgb_suspend_state)
  112. # define TIMEOUT 100
  113. # ifndef SLAVE_I2C_ADDRESS
  114. # define SLAVE_I2C_ADDRESS 0x32
  115. # endif
  116. // Get rows from other half over i2c
  117. bool transport_master(matrix_row_t master_matrix[], matrix_row_t slave_matrix[]) {
  118. i2c_readReg(SLAVE_I2C_ADDRESS, I2C_KEYMAP_SLAVE_START, (void *)slave_matrix, sizeof(i2c_buffer->smatrix), TIMEOUT);
  119. # ifdef SPLIT_TRANSPORT_MIRROR
  120. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_KEYMAP_MASTER_START, (void *)master_matrix, sizeof(i2c_buffer->mmatrix), TIMEOUT);
  121. # endif
  122. // write backlight info
  123. # ifdef BACKLIGHT_ENABLE
  124. uint8_t level = is_backlight_enabled() ? get_backlight_level() : 0;
  125. if (level != i2c_buffer->backlight_level) {
  126. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_BACKLIGHT_START, (void *)&level, sizeof(level), TIMEOUT) >= 0) {
  127. i2c_buffer->backlight_level = level;
  128. }
  129. }
  130. # endif
  131. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  132. if (rgblight_get_change_flags()) {
  133. rgblight_syncinfo_t rgblight_sync;
  134. rgblight_get_syncinfo(&rgblight_sync);
  135. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_RGB_START, (void *)&rgblight_sync, sizeof(rgblight_sync), TIMEOUT) >= 0) {
  136. rgblight_clear_change_flags();
  137. }
  138. }
  139. # endif
  140. # ifdef ENCODER_ENABLE
  141. i2c_readReg(SLAVE_I2C_ADDRESS, I2C_ENCODER_START, (void *)i2c_buffer->encoder_state, sizeof(i2c_buffer->encoder_state), TIMEOUT);
  142. encoder_update_raw(i2c_buffer->encoder_state);
  143. # endif
  144. # ifdef WPM_ENABLE
  145. uint8_t current_wpm = get_current_wpm();
  146. if (current_wpm != i2c_buffer->current_wpm) {
  147. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_WPM_START, (void *)&current_wpm, sizeof(current_wpm), TIMEOUT) >= 0) {
  148. i2c_buffer->current_wpm = current_wpm;
  149. }
  150. }
  151. # endif
  152. # ifdef POINTING_DEVICE_ENABLE
  153. if (is_keyboard_left()) {
  154. report_mouse_t temp_report = pointing_device_get_report();
  155. i2c_readReg(SLAVE_I2C_ADDRESS, I2C_MOUSE_X_START, (void *)&i2c_buffer->mouse_x, sizeof(i2c_buffer->mouse_x), TIMEOUT);
  156. temp_report.x = i2c_buffer->mouse_x;
  157. i2c_readReg(SLAVE_I2C_ADDRESS, I2C_MOUSE_Y_START, (void *)&i2c_buffer->mouse_y, sizeof(i2c_buffer->mouse_y), TIMEOUT);
  158. temp_report.y = i2c_buffer->mouse_y;
  159. pointing_device_set_report(temp_report);
  160. if (device_cpi != i2c_buffer->device_cpi) {
  161. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_MOUSE_DPI_START, (void *)&device_cpi, sizeof(device_cpi), TIMEOUT) >= 0) {
  162. i2c_buffer->device_cpi = device_cpi
  163. }
  164. }
  165. }
  166. # endif
  167. # ifdef SPLIT_MODS_ENABLE
  168. uint8_t real_mods = get_mods();
  169. if (real_mods != i2c_buffer->real_mods) {
  170. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_REAL_MODS_START, (void *)&real_mods, sizeof(real_mods), TIMEOUT) >= 0) {
  171. i2c_buffer->real_mods = real_mods;
  172. }
  173. }
  174. uint8_t weak_mods = get_weak_mods();
  175. if (weak_mods != i2c_buffer->weak_mods) {
  176. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_WEAK_MODS_START, (void *)&weak_mods, sizeof(weak_mods), TIMEOUT) >= 0) {
  177. i2c_buffer->weak_mods = weak_mods;
  178. }
  179. }
  180. # ifndef NO_ACTION_ONESHOT
  181. uint8_t oneshot_mods = get_oneshot_mods();
  182. if (oneshot_mods != i2c_buffer->oneshot_mods) {
  183. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_ONESHOT_MODS_START, (void *)&oneshot_mods, sizeof(oneshot_mods), TIMEOUT) >= 0) {
  184. i2c_buffer->oneshot_mods = oneshot_mods;
  185. }
  186. }
  187. # endif
  188. # endif
  189. if (layer_state != i2c_buffer->t_layer_state) {
  190. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_LAYER_STATE_START, (void *)&layer_state, sizeof(layer_state), TIMEOUT) >= 0) {
  191. i2c_buffer->t_layer_state = layer_state;
  192. }
  193. }
  194. if (default_layer_state != i2c_buffer->t_default_layer_state) {
  195. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_DEFAULT_LAYER_STATE_START, (void *)&default_layer_state, sizeof(default_layer_state), TIMEOUT) >= 0) {
  196. i2c_buffer->t_default_layer_state = default_layer_state;
  197. }
  198. }
  199. # ifdef OLED_DRIVER_ENABLE
  200. bool is_oled_on = is_oled_on();
  201. if (is_oled_on != i2c_buffer->oled_on) {
  202. if (i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_LAYER_STATE_START, (void *)&is_oled_on, sizeof(is_oled_on), TIMEOUT) >= 0) {
  203. i2c_buffer->oled_on = is_oled_on;
  204. }
  205. }
  206. # endif
  207. # if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  208. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_LED_MATRIX_START, (void *)led_matrix_eeconfig, sizeof(i2c_buffer->led_matrix), TIMEOUT);
  209. bool suspend_state = led_matrix_get_suspend_state();
  210. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_LED_SUSPEND_START, (void *)suspend_state, sizeof(i2c_buffer->led_suspend_state), TIMEOUT);
  211. # endif
  212. # if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  213. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_RGB_MATRIX_START, (void *)rgb_matrix_config, sizeof(i2c_buffer->rgb_matrix), TIMEOUT);
  214. bool suspend_state = rgb_matrix_get_suspend_state();
  215. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_RGB_SUSPEND_START, (void *)suspend_state, sizeof(i2c_buffer->rgb_suspend_state), TIMEOUT);
  216. # endif
  217. # ifndef DISABLE_SYNC_TIMER
  218. i2c_buffer->sync_timer = sync_timer_read32() + SYNC_TIMER_OFFSET;
  219. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_SYNC_TIME_START, (void *)&i2c_buffer->sync_timer, sizeof(i2c_buffer->sync_timer), TIMEOUT);
  220. # endif
  221. return true;
  222. }
  223. void transport_slave(matrix_row_t master_matrix[], matrix_row_t slave_matrix[]) {
  224. # ifndef DISABLE_SYNC_TIMER
  225. sync_timer_update(i2c_buffer->sync_timer);
  226. # endif
  227. // Copy matrix to I2C buffer
  228. memcpy((void *)i2c_buffer->smatrix, (void *)slave_matrix, sizeof(i2c_buffer->smatrix));
  229. # ifdef SPLIT_TRANSPORT_MIRROR
  230. memcpy((void *)master_matrix, (void *)i2c_buffer->mmatrix, sizeof(i2c_buffer->mmatrix));
  231. # endif
  232. // Read Backlight Info
  233. # ifdef BACKLIGHT_ENABLE
  234. backlight_set(i2c_buffer->backlight_level);
  235. # endif
  236. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  237. // Update the RGB with the new data
  238. if (i2c_buffer->rgblight_sync.status.change_flags != 0) {
  239. rgblight_update_sync(&i2c_buffer->rgblight_sync, false);
  240. i2c_buffer->rgblight_sync.status.change_flags = 0;
  241. }
  242. # endif
  243. # ifdef ENCODER_ENABLE
  244. encoder_state_raw(i2c_buffer->encoder_state);
  245. # endif
  246. # ifdef WPM_ENABLE
  247. set_current_wpm(i2c_buffer->current_wpm);
  248. # endif
  249. # ifdef POINTING_DEVICE_ENABLE
  250. if (!is_keyboard_left()) {
  251. static uint16_t cpi;
  252. if (cpi != i2c_buffer->device_cpi) {
  253. cpi = i2c_buffer->device_cpi;
  254. pmw_set_cpi(cpi);
  255. }
  256. i2c_buffer->mouse_x = split_mouse_x;
  257. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_MOUSE_X_START, (void *)&i2c_buffer->mouse_x, sizeof(i2c_buffer->mouse_x), TIMEOUT);
  258. i2c_buffer->mouse_y = split_mouse_y;
  259. i2c_writeReg(SLAVE_I2C_ADDRESS, I2C_MOUSE_Y_START, (void *)&i2c_buffer->mouse_y, sizeof(i2c_buffer->mouse_y), TIMEOUT);
  260. }
  261. # endif
  262. # ifdef SPLIT_MODS_ENABLE
  263. set_mods(i2c_buffer->real_mods);
  264. set_weak_mods(i2c_buffer->weak_mods);
  265. # ifndef NO_ACTION_ONESHOT
  266. set_oneshot_mods(i2c_buffer->oneshot_mods);
  267. # endif
  268. # endif
  269. if (layer_state != i2c_buffer->t_layer_state) {
  270. layer_state = i2c_buffer->t_layer_state;
  271. }
  272. if (default_layer_state != i2c_buffer->t_default_layer_state) {
  273. default_layer_state = i2c_buffer->t_default_layer_state;
  274. }
  275. # ifdef OLED_DRIVER_ENABLE
  276. if (i2c_buffer->oled_on) {
  277. oled_on();
  278. } else {
  279. oled_off();
  280. }
  281. # endif
  282. # if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  283. memcpy((void *)i2c_buffer->led_matrix, (void *)led_matrix_eeconfig, sizeof(i2c_buffer->led_matrix));
  284. led_matrix_set_suspend_state(i2c_buffer->led_suspend_state);
  285. # endif
  286. # if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  287. memcpy((void *)i2c_buffer->rgb_matrix, (void *)rgb_matrix_config, sizeof(i2c_buffer->rgb_matrix));
  288. rgb_matrix_set_suspend_state(i2c_buffer->rgb_suspend_state);
  289. # endif
  290. }
  291. void transport_master_init(void) { i2c_init(); }
  292. void transport_slave_init(void) { i2c_slave_init(SLAVE_I2C_ADDRESS); }
  293. #else // USE_SERIAL
  294. # include "serial.h"
  295. typedef struct _Serial_s2m_buffer_t {
  296. // TODO: if MATRIX_COLS > 8 change to uint8_t packed_matrix[] for pack/unpack
  297. matrix_row_t smatrix[ROWS_PER_HAND];
  298. # ifdef ENCODER_ENABLE
  299. uint8_t encoder_state[NUMBER_OF_ENCODERS];
  300. # endif
  301. int8_t mouse_x;
  302. int8_t mouse_y;
  303. } __attribute__((packed)) Serial_s2m_buffer_t;
  304. typedef struct _Serial_m2s_buffer_t {
  305. # ifdef SPLIT_MODS_ENABLE
  306. uint8_t real_mods;
  307. uint8_t weak_mods;
  308. # ifndef NO_ACTION_ONESHOT
  309. uint8_t oneshot_mods;
  310. # endif
  311. # endif
  312. # ifndef DISABLE_SYNC_TIMER
  313. uint32_t sync_timer;
  314. # endif
  315. # ifdef SPLIT_TRANSPORT_MIRROR
  316. matrix_row_t mmatrix[ROWS_PER_HAND];
  317. # endif
  318. # ifdef BACKLIGHT_ENABLE
  319. uint8_t backlight_level;
  320. # endif
  321. # ifdef WPM_ENABLE
  322. uint8_t current_wpm;
  323. # endif
  324. uint16_t device_cpi;
  325. bool oled_on;
  326. layer_state_t t_layer_state;
  327. layer_state_t t_default_layer_state;
  328. # if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  329. led_eeconfig_t led_matrix;
  330. bool led_suspend_state;
  331. # endif
  332. # if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  333. rgb_config_t rgb_matrix;
  334. bool rgb_suspend_state;
  335. # endif
  336. } __attribute__((packed)) Serial_m2s_buffer_t;
  337. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  338. // When MCUs on both sides drive their respective RGB LED chains,
  339. // it is necessary to synchronize, so it is necessary to communicate RGB
  340. // information. In that case, define RGBLIGHT_SPLIT with info on the number
  341. // of LEDs on each half.
  342. //
  343. // Otherwise, if the master side MCU drives both sides RGB LED chains,
  344. // there is no need to communicate.
  345. typedef struct _Serial_rgblight_t {
  346. rgblight_syncinfo_t rgblight_sync;
  347. } Serial_rgblight_t;
  348. volatile Serial_rgblight_t serial_rgblight = {};
  349. uint8_t volatile status_rgblight = 0;
  350. # endif
  351. volatile Serial_s2m_buffer_t serial_s2m_buffer = {};
  352. volatile Serial_m2s_buffer_t serial_m2s_buffer = {};
  353. uint8_t volatile status0 = 0;
  354. enum serial_transaction_id {
  355. GET_SLAVE_MATRIX = 0,
  356. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  357. PUT_RGBLIGHT,
  358. # endif
  359. };
  360. SSTD_t transactions[] = {
  361. [GET_SLAVE_MATRIX] =
  362. {
  363. (uint8_t *)&status0,
  364. sizeof(serial_m2s_buffer),
  365. (uint8_t *)&serial_m2s_buffer,
  366. sizeof(serial_s2m_buffer),
  367. (uint8_t *)&serial_s2m_buffer,
  368. },
  369. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  370. [PUT_RGBLIGHT] =
  371. {
  372. (uint8_t *)&status_rgblight, sizeof(serial_rgblight), (uint8_t *)&serial_rgblight, 0, NULL // no slave to master transfer
  373. },
  374. # endif
  375. };
  376. void transport_master_init(void) { soft_serial_initiator_init(transactions, TID_LIMIT(transactions)); }
  377. void transport_slave_init(void) { soft_serial_target_init(transactions, TID_LIMIT(transactions)); }
  378. # if defined(RGBLIGHT_ENABLE) && defined(RGBLIGHT_SPLIT)
  379. // rgblight synchronization information communication.
  380. void transport_rgblight_master(void) {
  381. if (rgblight_get_change_flags()) {
  382. rgblight_get_syncinfo((rgblight_syncinfo_t *)&serial_rgblight.rgblight_sync);
  383. if (soft_serial_transaction(PUT_RGBLIGHT) == TRANSACTION_END) {
  384. rgblight_clear_change_flags();
  385. }
  386. }
  387. }
  388. void transport_rgblight_slave(void) {
  389. if (status_rgblight == TRANSACTION_ACCEPTED) {
  390. rgblight_update_sync((rgblight_syncinfo_t *)&serial_rgblight.rgblight_sync, false);
  391. status_rgblight = TRANSACTION_END;
  392. }
  393. }
  394. # else
  395. # define transport_rgblight_master()
  396. # define transport_rgblight_slave()
  397. # endif
  398. bool transport_master(matrix_row_t master_matrix[], matrix_row_t slave_matrix[]) {
  399. # ifndef SERIAL_USE_MULTI_TRANSACTION
  400. if (soft_serial_transaction() != TRANSACTION_END) {
  401. return false;
  402. }
  403. # else
  404. transport_rgblight_master();
  405. if (soft_serial_transaction(GET_SLAVE_MATRIX) != TRANSACTION_END) {
  406. return false;
  407. }
  408. # endif
  409. // TODO: if MATRIX_COLS > 8 change to unpack()
  410. for (int i = 0; i < ROWS_PER_HAND; ++i) {
  411. slave_matrix[i] = serial_s2m_buffer.smatrix[i];
  412. # ifdef SPLIT_TRANSPORT_MIRROR
  413. serial_m2s_buffer.mmatrix[i] = master_matrix[i];
  414. # endif
  415. }
  416. # ifdef BACKLIGHT_ENABLE
  417. // Write backlight level for slave to read
  418. serial_m2s_buffer.backlight_level = is_backlight_enabled() ? get_backlight_level() : 0;
  419. # endif
  420. # ifdef ENCODER_ENABLE
  421. encoder_update_raw((uint8_t *)serial_s2m_buffer.encoder_state);
  422. # endif
  423. # ifdef WPM_ENABLE
  424. // Write wpm to slave
  425. serial_m2s_buffer.current_wpm = get_current_wpm();
  426. # endif
  427. # ifdef SPLIT_MODS_ENABLE
  428. serial_m2s_buffer.real_mods = get_mods();
  429. serial_m2s_buffer.weak_mods = get_weak_mods();
  430. # ifndef NO_ACTION_ONESHOT
  431. serial_m2s_buffer.oneshot_mods = get_oneshot_mods();
  432. # endif
  433. # endif
  434. # ifdef POINTING_DEVICE_ENABLE
  435. if (is_keyboard_left()) {
  436. report_mouse_t temp_report = pointing_device_get_report();
  437. temp_report.x = serial_s2m_buffer.mouse_x;
  438. temp_report.y = serial_s2m_buffer.mouse_y;
  439. pointing_device_set_report(temp_report);
  440. serial_m2s_buffer.device_cpi = device_cpi;
  441. }
  442. # endif
  443. serial_m2s_buffer.t_layer_state = layer_state;
  444. serial_m2s_buffer.t_default_layer_state = default_layer_state;
  445. # ifdef OLED_DRIVER_ENABLE
  446. serial_m2s_buffer.oled_on = is_oled_on();
  447. # endif
  448. # if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  449. serial_m2s_buffer.led_matrix = led_matrix_eeconfig;
  450. serial_m2s_buffer.led_suspend_state = led_matrix_get_suspend_state();
  451. # endif
  452. # if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  453. serial_m2s_buffer.rgb_matrix = rgb_matrix_config;
  454. serial_m2s_buffer.rgb_suspend_state = rgb_matrix_get_suspend_state();
  455. # endif
  456. # ifndef DISABLE_SYNC_TIMER
  457. serial_m2s_buffer.sync_timer = sync_timer_read32() + SYNC_TIMER_OFFSET;
  458. # endif
  459. return true;
  460. }
  461. void transport_slave(matrix_row_t master_matrix[], matrix_row_t slave_matrix[]) {
  462. transport_rgblight_slave();
  463. # ifndef DISABLE_SYNC_TIMER
  464. sync_timer_update(serial_m2s_buffer.sync_timer);
  465. # endif
  466. // TODO: if MATRIX_COLS > 8 change to pack()
  467. for (int i = 0; i < ROWS_PER_HAND; ++i) {
  468. serial_s2m_buffer.smatrix[i] = slave_matrix[i];
  469. # ifdef SPLIT_TRANSPORT_MIRROR
  470. master_matrix[i] = serial_m2s_buffer.mmatrix[i];
  471. # endif
  472. }
  473. # ifdef BACKLIGHT_ENABLE
  474. backlight_set(serial_m2s_buffer.backlight_level);
  475. # endif
  476. # ifdef ENCODER_ENABLE
  477. encoder_state_raw((uint8_t *)serial_s2m_buffer.encoder_state);
  478. # endif
  479. # ifdef WPM_ENABLE
  480. set_current_wpm(serial_m2s_buffer.current_wpm);
  481. # endif
  482. # ifdef SPLIT_MODS_ENABLE
  483. set_mods(serial_m2s_buffer.real_mods);
  484. set_weak_mods(serial_m2s_buffer.weak_mods);
  485. # ifndef NO_ACTION_ONESHOT
  486. set_oneshot_mods(serial_m2s_buffer.oneshot_mods);
  487. # endif
  488. # endif
  489. # ifdef POINTING_DEVICE_ENABLE
  490. if (!is_keyboard_left()) {
  491. static uint16_t cpi;
  492. if (cpi != serial_m2s_buffer.device_cpi) {
  493. cpi = serial_m2s_buffer.device_cpi;
  494. pmw_set_cpi(cpi);
  495. }
  496. serial_s2m_buffer.mouse_x = split_mouse_x;
  497. serial_s2m_buffer.mouse_y = split_mouse_y;
  498. }
  499. # endif
  500. if (layer_state != serial_m2s_buffer.t_layer_state) {
  501. layer_state = serial_m2s_buffer.t_layer_state;
  502. }
  503. if (default_layer_state != serial_m2s_buffer.t_default_layer_state) {
  504. default_layer_state = serial_m2s_buffer.t_default_layer_state;
  505. }
  506. # ifdef OLED_DRIVER_ENABLE
  507. if (serial_m2s_buffer.oled_on) {
  508. oled_on();
  509. } else {
  510. oled_off();
  511. }
  512. # endif
  513. # if defined(LED_MATRIX_ENABLE) && defined(LED_MATRIX_SPLIT)
  514. led_matrix_eeconfig = serial_m2s_buffer.led_matrix;
  515. led_matrix_set_suspend_state(serial_m2s_buffer.led_suspend_state);
  516. # endif
  517. # if defined(RGB_MATRIX_ENABLE) && defined(RGB_MATRIX_SPLIT)
  518. rgb_matrix_config = serial_m2s_buffer.rgb_matrix;
  519. rgb_matrix_set_suspend_state(serial_m2s_buffer.rgb_suspend_state);
  520. # endif
  521. }
  522. #endif