split_util.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  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_assert((SPLIT_USB_TIMEOUT / SPLIT_USB_TIMEOUT_POLL) <= UINT16_MAX, "Please lower SPLIT_USB_TIMEOUT and/or increase SPLIT_USB_TIMEOUT_POLL.");
  52. static bool usbIsActive(void) {
  53. for (uint16_t i = 0; i < (SPLIT_USB_TIMEOUT / SPLIT_USB_TIMEOUT_POLL); i++) {
  54. // This will return true if a USB connection has been established
  55. if (usb_connected_state()) {
  56. return true;
  57. }
  58. wait_ms(SPLIT_USB_TIMEOUT_POLL);
  59. }
  60. return false;
  61. }
  62. #else
  63. static inline bool usbIsActive(void) {
  64. return usb_vbus_state();
  65. }
  66. #endif
  67. #ifdef SPLIT_HAND_MATRIX_GRID
  68. void matrix_io_delay(void);
  69. static uint8_t peek_matrix_intersection(pin_t out_pin, pin_t in_pin) {
  70. setPinInputHigh(in_pin);
  71. setPinOutput(out_pin);
  72. writePinLow(out_pin);
  73. // It's almost unnecessary, but wait until it's down to low, just in case.
  74. wait_us(1);
  75. uint8_t pin_state = readPin(in_pin);
  76. // Set out_pin to a setting that is less susceptible to noise.
  77. setPinInputHigh(out_pin);
  78. matrix_io_delay(); // Wait for the pull-up to go HIGH.
  79. return pin_state;
  80. }
  81. #endif
  82. __attribute__((weak)) bool is_keyboard_left(void) {
  83. #if defined(SPLIT_HAND_PIN)
  84. // Test pin SPLIT_HAND_PIN for High/Low, if low it's right hand
  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. #if defined(SPLIT_HAND_PIN)
  118. setPinInput(SPLIT_HAND_PIN);
  119. wait_us(100);
  120. #elif defined(EE_HANDS)
  121. if (!eeconfig_is_enabled()) {
  122. eeconfig_init();
  123. }
  124. #endif
  125. isLeftHand = is_keyboard_left();
  126. #if defined(RGBLIGHT_ENABLE) && defined(RGBLED_SPLIT)
  127. uint8_t num_rgb_leds_split[2] = RGBLED_SPLIT;
  128. if (isLeftHand) {
  129. rgblight_set_clipping_range(0, num_rgb_leds_split[0]);
  130. } else {
  131. rgblight_set_clipping_range(num_rgb_leds_split[0], num_rgb_leds_split[1]);
  132. }
  133. #endif
  134. if (is_keyboard_master()) {
  135. #if defined(USE_I2C) && defined(SSD1306OLED)
  136. matrix_master_OLED_init();
  137. #endif
  138. transport_master_init();
  139. }
  140. }
  141. // this code runs after the keyboard is fully initialized
  142. // - avoids race condition during matrix_init_quantum where slave can start
  143. // receiving before the init process has completed
  144. void split_post_init(void) {
  145. if (!is_keyboard_master()) {
  146. transport_slave_init();
  147. }
  148. }
  149. bool is_transport_connected(void) {
  150. return connection_errors < SPLIT_MAX_CONNECTION_ERRORS;
  151. }
  152. bool transport_master_if_connected(matrix_row_t master_matrix[], matrix_row_t slave_matrix[]) {
  153. #if SPLIT_MAX_CONNECTION_ERRORS > 0 && SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  154. // Throttle transaction attempts if target doesn't seem to be connected
  155. // Without this, a solo half becomes unusable due to constant read timeouts
  156. static uint16_t connection_check_timer = 0;
  157. const bool is_disconnected = !is_transport_connected();
  158. if (is_disconnected && timer_elapsed(connection_check_timer) < SPLIT_CONNECTION_CHECK_TIMEOUT) {
  159. return false;
  160. }
  161. #endif // SPLIT_MAX_CONNECTION_ERRORS > 0 && SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  162. __attribute__((unused)) bool okay = transport_master(master_matrix, slave_matrix);
  163. #if SPLIT_MAX_CONNECTION_ERRORS > 0
  164. if (!okay) {
  165. if (connection_errors < UINT8_MAX) {
  166. connection_errors++;
  167. }
  168. # if SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  169. bool connected = is_transport_connected();
  170. if (!connected) {
  171. connection_check_timer = timer_read();
  172. dprintln("Target disconnected, throttling connection attempts");
  173. }
  174. return connected;
  175. } else if (is_disconnected) {
  176. dprintln("Target connected");
  177. # endif // SPLIT_CONNECTION_CHECK_TIMEOUT > 0
  178. }
  179. connection_errors = 0;
  180. #endif // SPLIT_MAX_CONNECTION_ERRORS > 0
  181. return true;
  182. }