You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

234 lines
6.8 KiB

  1. #include "quantum.h"
  2. #include "serial.h"
  3. #include "printf.h"
  4. #include "ch.h"
  5. #include "hal.h"
  6. #ifndef USART_CR1_M0
  7. # define USART_CR1_M0 USART_CR1_M // some platforms (f1xx) dont have this so
  8. #endif
  9. #ifndef USE_GPIOV1
  10. // The default PAL alternate modes are used to signal that the pins are used for USART
  11. # ifndef SERIAL_USART_TX_PAL_MODE
  12. # define SERIAL_USART_TX_PAL_MODE 7
  13. # endif
  14. #endif
  15. #ifndef SERIAL_USART_DRIVER
  16. # define SERIAL_USART_DRIVER SD1
  17. #endif
  18. #ifndef SERIAL_USART_CR1
  19. # define SERIAL_USART_CR1 (USART_CR1_PCE | USART_CR1_PS | USART_CR1_M0) // parity enable, odd parity, 9 bit length
  20. #endif
  21. #ifndef SERIAL_USART_CR2
  22. # define SERIAL_USART_CR2 (USART_CR2_STOP_1) // 2 stop bits
  23. #endif
  24. #ifndef SERIAL_USART_CR3
  25. # define SERIAL_USART_CR3 0
  26. #endif
  27. #ifdef SOFT_SERIAL_PIN
  28. # define SERIAL_USART_TX_PIN SOFT_SERIAL_PIN
  29. #endif
  30. #ifndef SELECT_SOFT_SERIAL_SPEED
  31. # define SELECT_SOFT_SERIAL_SPEED 1
  32. #endif
  33. #ifdef SERIAL_USART_SPEED
  34. // Allow advanced users to directly set SERIAL_USART_SPEED
  35. #elif SELECT_SOFT_SERIAL_SPEED == 0
  36. # define SERIAL_USART_SPEED 460800
  37. #elif SELECT_SOFT_SERIAL_SPEED == 1
  38. # define SERIAL_USART_SPEED 230400
  39. #elif SELECT_SOFT_SERIAL_SPEED == 2
  40. # define SERIAL_USART_SPEED 115200
  41. #elif SELECT_SOFT_SERIAL_SPEED == 3
  42. # define SERIAL_USART_SPEED 57600
  43. #elif SELECT_SOFT_SERIAL_SPEED == 4
  44. # define SERIAL_USART_SPEED 38400
  45. #elif SELECT_SOFT_SERIAL_SPEED == 5
  46. # define SERIAL_USART_SPEED 19200
  47. #else
  48. # error invalid SELECT_SOFT_SERIAL_SPEED value
  49. #endif
  50. #define TIMEOUT 100
  51. #define HANDSHAKE_MAGIC 7
  52. static inline msg_t sdWriteHalfDuplex(SerialDriver* driver, uint8_t* data, uint8_t size) {
  53. msg_t ret = sdWrite(driver, data, size);
  54. // Half duplex requires us to read back the data we just wrote - just throw it away
  55. uint8_t dump[size];
  56. sdRead(driver, dump, size);
  57. return ret;
  58. }
  59. #undef sdWrite
  60. #define sdWrite sdWriteHalfDuplex
  61. static inline msg_t sdWriteTimeoutHalfDuplex(SerialDriver* driver, uint8_t* data, uint8_t size, uint32_t timeout) {
  62. msg_t ret = sdWriteTimeout(driver, data, size, timeout);
  63. // Half duplex requires us to read back the data we just wrote - just throw it away
  64. uint8_t dump[size];
  65. sdReadTimeout(driver, dump, size, timeout);
  66. return ret;
  67. }
  68. #undef sdWriteTimeout
  69. #define sdWriteTimeout sdWriteTimeoutHalfDuplex
  70. static inline void sdClear(SerialDriver* driver) {
  71. while (sdGetTimeout(driver, TIME_IMMEDIATE) != MSG_TIMEOUT) {
  72. // Do nothing with the data
  73. }
  74. }
  75. static SerialConfig sdcfg = {
  76. (SERIAL_USART_SPEED), // speed - mandatory
  77. (SERIAL_USART_CR1), // CR1
  78. (SERIAL_USART_CR2), // CR2
  79. (SERIAL_USART_CR3) // CR3
  80. };
  81. void handle_soft_serial_slave(void);
  82. /*
  83. * This thread runs on the slave and responds to transactions initiated
  84. * by the master
  85. */
  86. static THD_WORKING_AREA(waSlaveThread, 2048);
  87. static THD_FUNCTION(SlaveThread, arg) {
  88. (void)arg;
  89. chRegSetThreadName("slave_transport");
  90. while (true) {
  91. handle_soft_serial_slave();
  92. }
  93. }
  94. __attribute__((weak)) void usart_init(void) {
  95. #if defined(USE_GPIOV1)
  96. palSetLineMode(SERIAL_USART_TX_PIN, PAL_MODE_STM32_ALTERNATE_OPENDRAIN);
  97. #else
  98. palSetLineMode(SERIAL_USART_TX_PIN, PAL_MODE_ALTERNATE(SERIAL_USART_TX_PAL_MODE) | PAL_STM32_OTYPE_OPENDRAIN);
  99. #endif
  100. }
  101. void usart_master_init(void) {
  102. usart_init();
  103. sdcfg.cr3 |= USART_CR3_HDSEL;
  104. sdStart(&SERIAL_USART_DRIVER, &sdcfg);
  105. }
  106. void usart_slave_init(void) {
  107. usart_init();
  108. sdcfg.cr3 |= USART_CR3_HDSEL;
  109. sdStart(&SERIAL_USART_DRIVER, &sdcfg);
  110. // Start transport thread
  111. chThdCreateStatic(waSlaveThread, sizeof(waSlaveThread), HIGHPRIO, SlaveThread, NULL);
  112. }
  113. static SSTD_t* Transaction_table = NULL;
  114. static uint8_t Transaction_table_size = 0;
  115. void soft_serial_initiator_init(SSTD_t* sstd_table, int sstd_table_size) {
  116. Transaction_table = sstd_table;
  117. Transaction_table_size = (uint8_t)sstd_table_size;
  118. usart_master_init();
  119. }
  120. void soft_serial_target_init(SSTD_t* sstd_table, int sstd_table_size) {
  121. Transaction_table = sstd_table;
  122. Transaction_table_size = (uint8_t)sstd_table_size;
  123. usart_slave_init();
  124. }
  125. void handle_soft_serial_slave(void) {
  126. uint8_t sstd_index = sdGet(&SERIAL_USART_DRIVER); // first chunk is always transaction id
  127. SSTD_t* trans = &Transaction_table[sstd_index];
  128. // Always write back the sstd_index as part of a basic handshake
  129. sstd_index ^= HANDSHAKE_MAGIC;
  130. sdWrite(&SERIAL_USART_DRIVER, &sstd_index, sizeof(sstd_index));
  131. if (trans->initiator2target_buffer_size) {
  132. sdRead(&SERIAL_USART_DRIVER, trans->initiator2target_buffer, trans->initiator2target_buffer_size);
  133. }
  134. if (trans->target2initiator_buffer_size) {
  135. sdWrite(&SERIAL_USART_DRIVER, trans->target2initiator_buffer, trans->target2initiator_buffer_size);
  136. }
  137. if (trans->status) {
  138. *trans->status = TRANSACTION_ACCEPTED;
  139. }
  140. }
  141. /////////
  142. // start transaction by initiator
  143. //
  144. // int soft_serial_transaction(int sstd_index)
  145. //
  146. // Returns:
  147. // TRANSACTION_END
  148. // TRANSACTION_NO_RESPONSE
  149. // TRANSACTION_DATA_ERROR
  150. #ifndef SERIAL_USE_MULTI_TRANSACTION
  151. int soft_serial_transaction(void) {
  152. uint8_t sstd_index = 0;
  153. #else
  154. int soft_serial_transaction(int index) {
  155. uint8_t sstd_index = index;
  156. #endif
  157. if (sstd_index > Transaction_table_size) return TRANSACTION_TYPE_ERROR;
  158. SSTD_t* trans = &Transaction_table[sstd_index];
  159. msg_t res = 0;
  160. sdClear(&SERIAL_USART_DRIVER);
  161. // First chunk is always transaction id
  162. sdWriteTimeout(&SERIAL_USART_DRIVER, &sstd_index, sizeof(sstd_index), TIME_MS2I(TIMEOUT));
  163. uint8_t sstd_index_shake = 0xFF;
  164. // Which we always read back first so that we can error out correctly
  165. // - due to the half duplex limitations on return codes, we always have to read *something*
  166. // - without the read, write only transactions *always* succeed, even during the boot process where the slave is not ready
  167. res = sdReadTimeout(&SERIAL_USART_DRIVER, &sstd_index_shake, sizeof(sstd_index_shake), TIME_MS2I(TIMEOUT));
  168. if (res < 0 || (sstd_index_shake != (sstd_index ^ HANDSHAKE_MAGIC))) {
  169. dprintf("serial::usart_shake NO_RESPONSE\n");
  170. return TRANSACTION_NO_RESPONSE;
  171. }
  172. if (trans->initiator2target_buffer_size) {
  173. res = sdWriteTimeout(&SERIAL_USART_DRIVER, trans->initiator2target_buffer, trans->initiator2target_buffer_size, TIME_MS2I(TIMEOUT));
  174. if (res < 0) {
  175. dprintf("serial::usart_transmit NO_RESPONSE\n");
  176. return TRANSACTION_NO_RESPONSE;
  177. }
  178. }
  179. if (trans->target2initiator_buffer_size) {
  180. res = sdReadTimeout(&SERIAL_USART_DRIVER, trans->target2initiator_buffer, trans->target2initiator_buffer_size, TIME_MS2I(TIMEOUT));
  181. if (res < 0) {
  182. dprintf("serial::usart_receive NO_RESPONSE\n");
  183. return TRANSACTION_NO_RESPONSE;
  184. }
  185. }
  186. return TRANSACTION_END;
  187. }