split_util.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. /* Copyright 2021 QMK
  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 3 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 "split_util.h"
  17. #include "matrix.h"
  18. #include "keyboard.h"
  19. #include "config.h"
  20. #include "timer.h"
  21. #include "transport.h"
  22. #include "quantum.h"
  23. #include "wait.h"
  24. #include "usb_util.h"
  25. #ifdef EE_HANDS
  26. # include "eeconfig.h"
  27. #endif
  28. #if defined(RGBLIGHT_ENABLE) && defined(RGBLED_SPLIT)
  29. # include "rgblight.h"
  30. #endif
  31. #ifndef SPLIT_USB_TIMEOUT
  32. # define SPLIT_USB_TIMEOUT 2000
  33. #endif
  34. #ifndef SPLIT_USB_TIMEOUT_POLL
  35. # define SPLIT_USB_TIMEOUT_POLL 10
  36. #endif
  37. // Max number of consecutive failed communications (one per scan cycle) before the communication is seen as disconnected.
  38. // Set to 0 to disable the disconnection check altogether.
  39. #ifndef SPLIT_MAX_CONNECTION_ERRORS
  40. # define SPLIT_MAX_CONNECTION_ERRORS 10
  41. #endif // SPLIT_MAX_CONNECTION_ERRORS
  42. // How long (in milliseconds) to block all connection attempts after the communication has been flagged as disconnected.
  43. // One communication attempt will be allowed everytime this amount of time has passed since the last attempt. If that attempt succeeds, the communication is seen as working again.
  44. // Set to 0 to disable communication throttling while disconnected
  45. #ifndef SPLIT_CONNECTION_CHECK_TIMEOUT
  46. # define SPLIT_CONNECTION_CHECK_TIMEOUT 500
  47. #endif // SPLIT_CONNECTION_CHECK_TIMEOUT
  48. static uint8_t connection_errors = 0;
  49. volatile bool isLeftHand = true;
  50. #if defined(SPLIT_USB_DETECT)
  51. static bool usbIsActive(void) {
  52. for (uint8_t i = 0; i < (SPLIT_USB_TIMEOUT / SPLIT_USB_TIMEOUT_POLL); i++) {
  53. // This will return true if a USB connection has been established
  54. if (usb_connected_state()) {
  55. return true;
  56. }
  57. wait_ms(SPLIT_USB_TIMEOUT_POLL);
  58. }
  59. return false;
  60. }
  61. #else
  62. static inline bool usbIsActive(void) {
  63. return usb_vbus_state();
  64. }
  65. #endif
  66. #ifdef SPLIT_HAND_MATRIX_GRID
  67. void matrix_io_delay(void);
  68. static uint8_t peek_matrix_intersection(pin_t out_pin, pin_t in_pin) {
  69. setPinInputHigh(in_pin);
  70. setPinOutput(out_pin);
  71. writePinLow(out_pin);
  72. // It's almost unnecessary, but wait until it's down to low, just in case.
  73. wait_us(1);
  74. uint8_t pin_state = readPin(in_pin);
  75. // Set out_pin to a setting that is less susceptible to noise.
  76. setPinInputHigh(out_pin);
  77. matrix_io_delay(); // Wait for the pull-up to go HIGH.
  78. return pin_state;
  79. }
  80. #endif
  81. __attribute__((weak)) bool is_keyboard_left(void) {
  82. #if defined(SPLIT_HAND_PIN)
  83. // Test pin SPLIT_HAND_PIN for High/Low, if low it's right hand
  84. setPinInput(SPLIT_HAND_PIN);
  85. # ifdef SPLIT_HAND_PIN_LOW_IS_LEFT
  86. return !readPin(SPLIT_HAND_PIN);
  87. # else
  88. return readPin(SPLIT_HAND_PIN);
  89. # endif
  90. #elif defined(SPLIT_HAND_MATRIX_GRID)
  91. # ifdef SPLIT_HAND_MATRIX_GRID_LOW_IS_RIGHT
  92. return peek_matrix_intersection(SPLIT_HAND_MATRIX_GRID);
  93. # else
  94. return !peek_matrix_intersection(SPLIT_HAND_MATRIX_GRID);
  95. # endif
  96. #elif defined(EE_HANDS)
  97. return eeconfig_read_handedness();
  98. #elif defined(MASTER_RIGHT)
  99. return !is_keyboard_master();
  100. #endif
  101. return is_keyboard_master();
  102. }
  103. __attribute__((weak)) bool is_keyboard_master(void) {
  104. static enum { UNKNOWN, MASTER, SLAVE } usbstate = UNKNOWN;
  105. // only check once, as this is called often
  106. if (usbstate == UNKNOWN) {
  107. usbstate = usbIsActive() ? MASTER : SLAVE;
  108. // Avoid NO_USB_STARTUP_CHECK - Disable USB as the previous checks seem to enable it somehow
  109. if (usbstate == SLAVE) {
  110. usb_disconnect();
  111. }
  112. }
  113. return (usbstate == MASTER);
  114. }
  115. // this code runs before the keyboard is fully initialized
  116. void split_pre_init(void) {
  117. isLeftHand = is_keyboard_left();
  118. #if defined(RGBLIGHT_ENABLE) && defined(RGBLED_SPLIT)
  119. uint8_t num_rgb_leds_split[2] = RGBLED_SPLIT;
  120. if (isLeftHand) {
  121. rgblight_set_clipping_range(0, num_rgb_leds_split[0]);
  122. } else {
  123. rgblight_set_clipping_range(num_rgb_leds_split[0], num_rgb_leds_split[1]);
  124. }
  125. #endif
  126. if (is_keyboard_master()) {
  127. #if defined(USE_I2C) && defined(SSD1306OLED)
  128. matrix_master_OLED_init();
  129. #endif
  130. transport_master_init();
  131. }
  132. }
  133. // this code runs after the keyboard is fully initialized
  134. // - avoids race condition during matrix_init_quantum where slave can start
  135. // receiving before the init process has completed
  136. void split_post_init(void) {
  137. if (!is_keyboard_master()) {
  138. transport_slave_init();
  139. }
  140. }
  141. bool is_transport_connected(void) {
  142. return connection_errors < SPLIT_MAX_CONNECTION_ERRORS;
  143. }
  144. bool transport_master_if_connected(matrix_row_t master_matrix[], matrix_row_t slave_matrix[]) {
  145. #if SPLIT_MAX_CONNECTION_ERRORS > 0 && SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  146. // Throttle transaction attempts if target doesn't seem to be connected
  147. // Without this, a solo half becomes unusable due to constant read timeouts
  148. static uint16_t connection_check_timer = 0;
  149. const bool is_disconnected = !is_transport_connected();
  150. if (is_disconnected && timer_elapsed(connection_check_timer) < SPLIT_CONNECTION_CHECK_TIMEOUT) {
  151. return false;
  152. }
  153. #endif // SPLIT_MAX_CONNECTION_ERRORS > 0 && SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  154. __attribute__((unused)) bool okay = transport_master(master_matrix, slave_matrix);
  155. #if SPLIT_MAX_CONNECTION_ERRORS > 0
  156. if (!okay) {
  157. if (connection_errors < UINT8_MAX) {
  158. connection_errors++;
  159. }
  160. # if SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  161. bool connected = is_transport_connected();
  162. if (!connected) {
  163. connection_check_timer = timer_read();
  164. dprintln("Target disconnected, throttling connection attempts");
  165. }
  166. return connected;
  167. } else if (is_disconnected) {
  168. dprintln("Target connected");
  169. # endif // SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  170. }
  171. connection_errors = 0;
  172. #endif // SPLIT_MAX_CONNECTION_ERRORS > 0
  173. return true;
  174. }