gpio_scene_usb_uart_config.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #include "../usb_uart_bridge.h"
  2. #include "../gpio_app_i.h"
  3. #include "furi_hal.h"
  4. typedef enum {
  5. UsbUartLineIndexVcp,
  6. UsbUartLineIndexBaudrate,
  7. UsbUartLineIndexUart,
  8. UsbUartLineIndexFlow,
  9. } LineIndex;
  10. static UsbUartConfig* cfg_set;
  11. static const char* vcp_ch[] = {"0 (CLI)", "1"};
  12. static const char* uart_ch[] = {"13,14", "15,16"};
  13. static const char* flow_pins[] = {"None", "2,3", "6,7", "16,15"};
  14. static const char* baudrate_mode[] = {"Host"};
  15. static const uint32_t baudrate_list[] = {
  16. 2400,
  17. 9600,
  18. 19200,
  19. 38400,
  20. 57600,
  21. 115200,
  22. 230400,
  23. 460800,
  24. 921600,
  25. };
  26. bool gpio_scene_usb_uart_cfg_on_event(void* context, SceneManagerEvent event) {
  27. UNUSED(context);
  28. UNUSED(event);
  29. return false;
  30. }
  31. void line_ensure_flow_invariant(GpioApp* app) {
  32. // GPIO pins PC0, PC1 (16,15) are unavailable for RTS/DTR when LPUART is
  33. // selected. This function enforces that invariant by resetting flow_pins
  34. // to None if it is configured to 16,15 when LPUART is selected.
  35. uint8_t available_flow_pins = cfg_set->uart_ch == FuriHalUartIdLPUART1 ? 3 : 4;
  36. VariableItem* item = app->var_item_flow;
  37. variable_item_set_values_count(item, available_flow_pins);
  38. if(cfg_set->flow_pins >= available_flow_pins) {
  39. cfg_set->flow_pins = 0;
  40. usb_uart_set_config(app->usb_uart_bridge, cfg_set);
  41. variable_item_set_current_value_index(item, cfg_set->flow_pins);
  42. variable_item_set_current_value_text(item, flow_pins[cfg_set->flow_pins]);
  43. }
  44. }
  45. static void line_vcp_cb(VariableItem* item) {
  46. GpioApp* app = variable_item_get_context(item);
  47. uint8_t index = variable_item_get_current_value_index(item);
  48. variable_item_set_current_value_text(item, vcp_ch[index]);
  49. cfg_set->vcp_ch = index;
  50. usb_uart_set_config(app->usb_uart_bridge, cfg_set);
  51. }
  52. static void line_port_cb(VariableItem* item) {
  53. GpioApp* app = variable_item_get_context(item);
  54. uint8_t index = variable_item_get_current_value_index(item);
  55. variable_item_set_current_value_text(item, uart_ch[index]);
  56. if(index == 0)
  57. cfg_set->uart_ch = FuriHalUartIdUSART1;
  58. else if(index == 1)
  59. cfg_set->uart_ch = FuriHalUartIdLPUART1;
  60. usb_uart_set_config(app->usb_uart_bridge, cfg_set);
  61. line_ensure_flow_invariant(app);
  62. }
  63. static void line_flow_cb(VariableItem* item) {
  64. GpioApp* app = variable_item_get_context(item);
  65. uint8_t index = variable_item_get_current_value_index(item);
  66. variable_item_set_current_value_text(item, flow_pins[index]);
  67. cfg_set->flow_pins = index;
  68. usb_uart_set_config(app->usb_uart_bridge, cfg_set);
  69. }
  70. static void line_baudrate_cb(VariableItem* item) {
  71. GpioApp* app = variable_item_get_context(item);
  72. uint8_t index = variable_item_get_current_value_index(item);
  73. char br_text[8];
  74. if(index > 0) {
  75. snprintf(br_text, 7, "%lu", baudrate_list[index - 1]);
  76. variable_item_set_current_value_text(item, br_text);
  77. cfg_set->baudrate = baudrate_list[index - 1];
  78. } else {
  79. variable_item_set_current_value_text(item, baudrate_mode[index]);
  80. cfg_set->baudrate = 0;
  81. }
  82. cfg_set->baudrate_mode = index;
  83. usb_uart_set_config(app->usb_uart_bridge, cfg_set);
  84. }
  85. void gpio_scene_usb_uart_cfg_on_enter(void* context) {
  86. GpioApp* app = context;
  87. VariableItemList* var_item_list = app->var_item_list;
  88. cfg_set = malloc(sizeof(UsbUartConfig));
  89. usb_uart_get_config(app->usb_uart_bridge, cfg_set);
  90. VariableItem* item;
  91. char br_text[8];
  92. item = variable_item_list_add(var_item_list, "USB Channel", 2, line_vcp_cb, app);
  93. variable_item_set_current_value_index(item, cfg_set->vcp_ch);
  94. variable_item_set_current_value_text(item, vcp_ch[cfg_set->vcp_ch]);
  95. item = variable_item_list_add(
  96. var_item_list,
  97. "Baudrate",
  98. sizeof(baudrate_list) / sizeof(baudrate_list[0]) + 1,
  99. line_baudrate_cb,
  100. app);
  101. variable_item_set_current_value_index(item, cfg_set->baudrate_mode);
  102. if(cfg_set->baudrate_mode > 0) {
  103. snprintf(br_text, 7, "%lu", baudrate_list[cfg_set->baudrate_mode - 1]);
  104. variable_item_set_current_value_text(item, br_text);
  105. } else {
  106. variable_item_set_current_value_text(item, baudrate_mode[cfg_set->baudrate_mode]);
  107. }
  108. item = variable_item_list_add(var_item_list, "UART Pins", 2, line_port_cb, app);
  109. variable_item_set_current_value_index(item, cfg_set->uart_ch);
  110. variable_item_set_current_value_text(item, uart_ch[cfg_set->uart_ch]);
  111. item = variable_item_list_add(
  112. var_item_list, "RTS/DTR Pins", COUNT_OF(flow_pins), line_flow_cb, app);
  113. variable_item_set_current_value_index(item, cfg_set->flow_pins);
  114. variable_item_set_current_value_text(item, flow_pins[cfg_set->flow_pins]);
  115. app->var_item_flow = item;
  116. line_ensure_flow_invariant(app);
  117. variable_item_list_set_selected_item(
  118. var_item_list, scene_manager_get_scene_state(app->scene_manager, GpioAppViewUsbUartCfg));
  119. view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUartCfg);
  120. }
  121. void gpio_scene_usb_uart_cfg_on_exit(void* context) {
  122. GpioApp* app = context;
  123. scene_manager_set_scene_state(
  124. app->scene_manager,
  125. GpioAppViewUsbUartCfg,
  126. variable_item_list_get_selected_item_index(app->var_item_list));
  127. variable_item_list_reset(app->var_item_list);
  128. free(cfg_set);
  129. }