Merge branch 'dev' into release-candidate
This commit is contained in:
		
						commit
						e1ff6da6cd
					
				| @ -79,12 +79,18 @@ BadUsbApp* bad_usb_app_alloc(char* arg) { | |||||||
| 
 | 
 | ||||||
|     view_dispatcher_attach_to_gui(app->view_dispatcher, app->gui, ViewDispatcherTypeFullscreen); |     view_dispatcher_attach_to_gui(app->view_dispatcher, app->gui, ViewDispatcherTypeFullscreen); | ||||||
| 
 | 
 | ||||||
|     if(*app->file_name != '\0') { |     if(furi_hal_usb_is_locked()) { | ||||||
|         scene_manager_next_scene(app->scene_manager, BadUsbSceneWork); |         app->error = BadUsbAppErrorCloseRpc; | ||||||
|     } else if(bad_usb_check_assets()) { |  | ||||||
|         scene_manager_next_scene(app->scene_manager, BadUsbSceneFileSelect); |  | ||||||
|     } else { |  | ||||||
|         scene_manager_next_scene(app->scene_manager, BadUsbSceneError); |         scene_manager_next_scene(app->scene_manager, BadUsbSceneError); | ||||||
|  |     } else { | ||||||
|  |         if(*app->file_name != '\0') { | ||||||
|  |             scene_manager_next_scene(app->scene_manager, BadUsbSceneWork); | ||||||
|  |         } else if(bad_usb_check_assets()) { | ||||||
|  |             scene_manager_next_scene(app->scene_manager, BadUsbSceneFileSelect); | ||||||
|  |         } else { | ||||||
|  |             app->error = BadUsbAppErrorNoFiles; | ||||||
|  |             scene_manager_next_scene(app->scene_manager, BadUsbSceneError); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return app; |     return app; | ||||||
|  | |||||||
| @ -18,6 +18,11 @@ | |||||||
| #define BAD_USB_APP_EXTENSION ".txt" | #define BAD_USB_APP_EXTENSION ".txt" | ||||||
| #define BAD_USB_FILE_NAME_LEN 40 | #define BAD_USB_FILE_NAME_LEN 40 | ||||||
| 
 | 
 | ||||||
|  | typedef enum { | ||||||
|  |     BadUsbAppErrorNoFiles, | ||||||
|  |     BadUsbAppErrorCloseRpc, | ||||||
|  | } BadUsbAppError; | ||||||
|  | 
 | ||||||
| struct BadUsbApp { | struct BadUsbApp { | ||||||
|     Gui* gui; |     Gui* gui; | ||||||
|     ViewDispatcher* view_dispatcher; |     ViewDispatcher* view_dispatcher; | ||||||
| @ -26,6 +31,7 @@ struct BadUsbApp { | |||||||
|     DialogsApp* dialogs; |     DialogsApp* dialogs; | ||||||
|     Widget* widget; |     Widget* widget; | ||||||
| 
 | 
 | ||||||
|  |     BadUsbAppError error; | ||||||
|     char file_name[BAD_USB_FILE_NAME_LEN + 1]; |     char file_name[BAD_USB_FILE_NAME_LEN + 1]; | ||||||
|     BadUsb* bad_usb_view; |     BadUsb* bad_usb_view; | ||||||
|     BadUsbScript* bad_usb_script; |     BadUsbScript* bad_usb_script; | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| #include "../bad_usb_app_i.h" | #include "../bad_usb_app_i.h" | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     SubghzCustomEventErrorBack, |     BadUsbCustomEventErrorBack, | ||||||
| } BadUsbCustomEvent; | } BadUsbCustomEvent; | ||||||
| 
 | 
 | ||||||
| static void | static void | ||||||
| @ -10,23 +10,33 @@ static void | |||||||
|     BadUsbApp* app = context; |     BadUsbApp* app = context; | ||||||
| 
 | 
 | ||||||
|     if((result == GuiButtonTypeLeft) && (type == InputTypeShort)) { |     if((result == GuiButtonTypeLeft) && (type == InputTypeShort)) { | ||||||
|         view_dispatcher_send_custom_event(app->view_dispatcher, SubghzCustomEventErrorBack); |         view_dispatcher_send_custom_event(app->view_dispatcher, BadUsbCustomEventErrorBack); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void bad_usb_scene_error_on_enter(void* context) { | void bad_usb_scene_error_on_enter(void* context) { | ||||||
|     BadUsbApp* app = context; |     BadUsbApp* app = context; | ||||||
| 
 | 
 | ||||||
|     widget_add_icon_element(app->widget, 0, 0, &I_SDQuestion_35x43); |     if(app->error == BadUsbAppErrorNoFiles) { | ||||||
| 
 |         widget_add_icon_element(app->widget, 0, 0, &I_SDQuestion_35x43); | ||||||
|     widget_add_string_multiline_element( |         widget_add_string_multiline_element( | ||||||
|         app->widget, |             app->widget, | ||||||
|         81, |             81, | ||||||
|         4, |             4, | ||||||
|         AlignCenter, |             AlignCenter, | ||||||
|         AlignTop, |             AlignTop, | ||||||
|         FontSecondary, |             FontSecondary, | ||||||
|         "No SD card or\napp data found.\nThis app will not\nwork without\nrequired files."); |             "No SD card or\napp data found.\nThis app will not\nwork without\nrequired files."); | ||||||
|  |     } else if(app->error == BadUsbAppErrorCloseRpc) { | ||||||
|  |         widget_add_string_multiline_element( | ||||||
|  |             app->widget, | ||||||
|  |             63, | ||||||
|  |             10, | ||||||
|  |             AlignCenter, | ||||||
|  |             AlignTop, | ||||||
|  |             FontSecondary, | ||||||
|  |             "Disconnect from\ncompanion app\nto use this function"); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     widget_add_button_element( |     widget_add_button_element( | ||||||
|         app->widget, GuiButtonTypeLeft, "Back", bad_usb_scene_error_event_callback, app); |         app->widget, GuiButtonTypeLeft, "Back", bad_usb_scene_error_event_callback, app); | ||||||
| @ -39,7 +49,7 @@ bool bad_usb_scene_error_on_event(void* context, SceneManagerEvent event) { | |||||||
|     bool consumed = false; |     bool consumed = false; | ||||||
| 
 | 
 | ||||||
|     if(event.type == SceneManagerEventTypeCustom) { |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|         if(event.event == SubghzCustomEventErrorBack) { |         if(event.event == BadUsbCustomEventErrorBack) { | ||||||
|             view_dispatcher_stop(app->view_dispatcher); |             view_dispatcher_stop(app->view_dispatcher); | ||||||
|             consumed = true; |             consumed = true; | ||||||
|         } |         } | ||||||
|  | |||||||
| @ -42,7 +42,8 @@ int32_t usb_mouse_app(void* p) { | |||||||
|     ViewPort* view_port = view_port_alloc(); |     ViewPort* view_port = view_port_alloc(); | ||||||
| 
 | 
 | ||||||
|     FuriHalUsbInterface* usb_mode_prev = furi_hal_usb_get_config(); |     FuriHalUsbInterface* usb_mode_prev = furi_hal_usb_get_config(); | ||||||
|     furi_hal_usb_set_config(&usb_hid, NULL); |     furi_hal_usb_unlock(); | ||||||
|  |     furi_check(furi_hal_usb_set_config(&usb_hid, NULL) == true); | ||||||
| 
 | 
 | ||||||
|     view_port_draw_callback_set(view_port, usb_mouse_render_callback, NULL); |     view_port_draw_callback_set(view_port, usb_mouse_render_callback, NULL); | ||||||
|     view_port_input_callback_set(view_port, usb_mouse_input_callback, event_queue); |     view_port_input_callback_set(view_port, usb_mouse_input_callback, event_queue); | ||||||
|  | |||||||
| @ -51,6 +51,10 @@ GpioApp* gpio_app_alloc() { | |||||||
|     view_dispatcher_add_view( |     view_dispatcher_add_view( | ||||||
|         app->view_dispatcher, GpioAppViewGpioTest, gpio_test_get_view(app->gpio_test)); |         app->view_dispatcher, GpioAppViewGpioTest, gpio_test_get_view(app->gpio_test)); | ||||||
| 
 | 
 | ||||||
|  |     app->widget = widget_alloc(); | ||||||
|  |     view_dispatcher_add_view( | ||||||
|  |         app->view_dispatcher, GpioAppViewUsbUartCloseRpc, widget_get_view(app->widget)); | ||||||
|  | 
 | ||||||
|     app->gpio_usb_uart = gpio_usb_uart_alloc(); |     app->gpio_usb_uart = gpio_usb_uart_alloc(); | ||||||
|     view_dispatcher_add_view( |     view_dispatcher_add_view( | ||||||
|         app->view_dispatcher, GpioAppViewUsbUart, gpio_usb_uart_get_view(app->gpio_usb_uart)); |         app->view_dispatcher, GpioAppViewUsbUart, gpio_usb_uart_get_view(app->gpio_usb_uart)); | ||||||
| @ -73,7 +77,9 @@ void gpio_app_free(GpioApp* app) { | |||||||
|     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewGpioTest); |     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewGpioTest); | ||||||
|     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUart); |     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUart); | ||||||
|     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUartCfg); |     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUartCfg); | ||||||
|  |     view_dispatcher_remove_view(app->view_dispatcher, GpioAppViewUsbUartCloseRpc); | ||||||
|     variable_item_list_free(app->var_item_list); |     variable_item_list_free(app->var_item_list); | ||||||
|  |     widget_free(app->widget); | ||||||
|     gpio_test_free(app->gpio_test); |     gpio_test_free(app->gpio_test); | ||||||
|     gpio_usb_uart_free(app->gpio_usb_uart); |     gpio_usb_uart_free(app->gpio_usb_uart); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -12,6 +12,7 @@ | |||||||
| #include <gui/modules/submenu.h> | #include <gui/modules/submenu.h> | ||||||
| #include <notification/notification_messages.h> | #include <notification/notification_messages.h> | ||||||
| #include <gui/modules/variable_item_list.h> | #include <gui/modules/variable_item_list.h> | ||||||
|  | #include <gui/modules/widget.h> | ||||||
| #include "views/gpio_test.h" | #include "views/gpio_test.h" | ||||||
| #include "views/gpio_usb_uart.h" | #include "views/gpio_usb_uart.h" | ||||||
| 
 | 
 | ||||||
| @ -20,6 +21,7 @@ struct GpioApp { | |||||||
|     NotificationApp* notifications; |     NotificationApp* notifications; | ||||||
|     ViewDispatcher* view_dispatcher; |     ViewDispatcher* view_dispatcher; | ||||||
|     SceneManager* scene_manager; |     SceneManager* scene_manager; | ||||||
|  |     Widget* widget; | ||||||
| 
 | 
 | ||||||
|     VariableItemList* var_item_list; |     VariableItemList* var_item_list; | ||||||
|     GpioTest* gpio_test; |     GpioTest* gpio_test; | ||||||
| @ -32,4 +34,5 @@ typedef enum { | |||||||
|     GpioAppViewGpioTest, |     GpioAppViewGpioTest, | ||||||
|     GpioAppViewUsbUart, |     GpioAppViewUsbUart, | ||||||
|     GpioAppViewUsbUartCfg, |     GpioAppViewUsbUartCfg, | ||||||
|  |     GpioAppViewUsbUartCloseRpc, | ||||||
| } GpioAppView; | } GpioAppView; | ||||||
|  | |||||||
| @ -6,5 +6,7 @@ typedef enum { | |||||||
|     GpioStartEventManualConrol, |     GpioStartEventManualConrol, | ||||||
|     GpioStartEventUsbUart, |     GpioStartEventUsbUart, | ||||||
| 
 | 
 | ||||||
|  |     GpioCustomEventErrorBack, | ||||||
|  | 
 | ||||||
|     GpioUsbUartEventConfig, |     GpioUsbUartEventConfig, | ||||||
| } GpioCustomEvent; | } GpioCustomEvent; | ||||||
|  | |||||||
| @ -2,3 +2,4 @@ ADD_SCENE(gpio, start, Start) | |||||||
| ADD_SCENE(gpio, test, Test) | ADD_SCENE(gpio, test, Test) | ||||||
| ADD_SCENE(gpio, usb_uart, UsbUart) | ADD_SCENE(gpio, usb_uart, UsbUart) | ||||||
| ADD_SCENE(gpio, usb_uart_cfg, UsbUartCfg) | ADD_SCENE(gpio, usb_uart_cfg, UsbUartCfg) | ||||||
|  | ADD_SCENE(gpio, usb_uart_close_rpc, UsbUartCloseRpc) | ||||||
|  | |||||||
| @ -1,5 +1,6 @@ | |||||||
| #include "../gpio_app_i.h" | #include "../gpio_app_i.h" | ||||||
| #include "furi_hal_power.h" | #include "furi_hal_power.h" | ||||||
|  | #include "furi_hal_usb.h" | ||||||
| 
 | 
 | ||||||
| enum GpioItem { | enum GpioItem { | ||||||
|     GpioItemUsbUart, |     GpioItemUsbUart, | ||||||
| @ -86,7 +87,11 @@ bool gpio_scene_start_on_event(void* context, SceneManagerEvent event) { | |||||||
|             scene_manager_next_scene(app->scene_manager, GpioSceneTest); |             scene_manager_next_scene(app->scene_manager, GpioSceneTest); | ||||||
|         } else if(event.event == GpioStartEventUsbUart) { |         } else if(event.event == GpioStartEventUsbUart) { | ||||||
|             scene_manager_set_scene_state(app->scene_manager, GpioSceneStart, GpioItemUsbUart); |             scene_manager_set_scene_state(app->scene_manager, GpioSceneStart, GpioItemUsbUart); | ||||||
|             scene_manager_next_scene(app->scene_manager, GpioSceneUsbUart); |             if(!furi_hal_usb_is_locked()) { | ||||||
|  |                 scene_manager_next_scene(app->scene_manager, GpioSceneUsbUart); | ||||||
|  |             } else { | ||||||
|  |                 scene_manager_next_scene(app->scene_manager, GpioSceneUsbUartCloseRpc); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|         consumed = true; |         consumed = true; | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -31,7 +31,7 @@ void gpio_scene_usb_uart_on_enter(void* context) { | |||||||
|     usb_uart_get_state(app->usb_uart_bridge, &scene_usb_uart->state); |     usb_uart_get_state(app->usb_uart_bridge, &scene_usb_uart->state); | ||||||
| 
 | 
 | ||||||
|     gpio_usb_uart_set_callback(app->gpio_usb_uart, gpio_scene_usb_uart_callback, app); |     gpio_usb_uart_set_callback(app->gpio_usb_uart, gpio_scene_usb_uart_callback, app); | ||||||
|     scene_manager_set_scene_state(app->scene_manager, GpioAppViewUsbUart, 0); |     scene_manager_set_scene_state(app->scene_manager, GpioSceneUsbUart, 0); | ||||||
|     view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUart); |     view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUart); | ||||||
|     notification_message(app->notifications, &sequence_display_lock); |     notification_message(app->notifications, &sequence_display_lock); | ||||||
| } | } | ||||||
| @ -39,8 +39,8 @@ void gpio_scene_usb_uart_on_enter(void* context) { | |||||||
| bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) { | bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) { | ||||||
|     GpioApp* app = context; |     GpioApp* app = context; | ||||||
|     if(event.type == SceneManagerEventTypeCustom) { |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|         scene_manager_set_scene_state(app->scene_manager, GpioAppViewUsbUart, 1); |         scene_manager_set_scene_state(app->scene_manager, GpioSceneUsbUart, 1); | ||||||
|         scene_manager_next_scene(app->scene_manager, GpioAppViewUsbUartCfg); |         scene_manager_next_scene(app->scene_manager, GpioSceneUsbUartCfg); | ||||||
|         return true; |         return true; | ||||||
|     } else if(event.type == SceneManagerEventTypeTick) { |     } else if(event.type == SceneManagerEventTypeTick) { | ||||||
|         uint32_t tx_cnt_last = scene_usb_uart->state.tx_cnt; |         uint32_t tx_cnt_last = scene_usb_uart->state.tx_cnt; | ||||||
| @ -58,7 +58,7 @@ bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) { | |||||||
| 
 | 
 | ||||||
| void gpio_scene_usb_uart_on_exit(void* context) { | void gpio_scene_usb_uart_on_exit(void* context) { | ||||||
|     GpioApp* app = context; |     GpioApp* app = context; | ||||||
|     uint32_t prev_state = scene_manager_get_scene_state(app->scene_manager, GpioAppViewUsbUart); |     uint32_t prev_state = scene_manager_get_scene_state(app->scene_manager, GpioSceneUsbUart); | ||||||
|     if(prev_state == 0) { |     if(prev_state == 0) { | ||||||
|         usb_uart_disable(app->usb_uart_bridge); |         usb_uart_disable(app->usb_uart_bridge); | ||||||
|         free(scene_usb_uart); |         free(scene_usb_uart); | ||||||
|  | |||||||
							
								
								
									
										53
									
								
								applications/gpio/scenes/gpio_scene_usb_uart_close_rpc.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								applications/gpio/scenes/gpio_scene_usb_uart_close_rpc.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | #include "../gpio_app_i.h" | ||||||
|  | #include "../gpio_custom_event.h" | ||||||
|  | 
 | ||||||
|  | static void gpio_scene_usb_uart_close_rpc_event_callback( | ||||||
|  |     GuiButtonType result, | ||||||
|  |     InputType type, | ||||||
|  |     void* context) { | ||||||
|  |     furi_assert(context); | ||||||
|  |     GpioApp* app = context; | ||||||
|  | 
 | ||||||
|  |     if((result == GuiButtonTypeLeft) && (type == InputTypeShort)) { | ||||||
|  |         view_dispatcher_send_custom_event(app->view_dispatcher, GpioCustomEventErrorBack); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void gpio_scene_usb_uart_close_rpc_on_enter(void* context) { | ||||||
|  |     GpioApp* app = context; | ||||||
|  | 
 | ||||||
|  |     widget_add_string_multiline_element( | ||||||
|  |         app->widget, | ||||||
|  |         63, | ||||||
|  |         10, | ||||||
|  |         AlignCenter, | ||||||
|  |         AlignTop, | ||||||
|  |         FontSecondary, | ||||||
|  |         "Disconnect from\ncompanion app\nto use this function"); | ||||||
|  | 
 | ||||||
|  |     widget_add_button_element( | ||||||
|  |         app->widget, GuiButtonTypeLeft, "Back", gpio_scene_usb_uart_close_rpc_event_callback, app); | ||||||
|  | 
 | ||||||
|  |     view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUartCloseRpc); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool gpio_scene_usb_uart_close_rpc_on_event(void* context, SceneManagerEvent event) { | ||||||
|  |     GpioApp* app = context; | ||||||
|  |     bool consumed = false; | ||||||
|  | 
 | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == GpioCustomEventErrorBack) { | ||||||
|  |             if(!scene_manager_previous_scene(app->scene_manager)) { | ||||||
|  |                 scene_manager_stop(app->scene_manager); | ||||||
|  |                 view_dispatcher_stop(app->view_dispatcher); | ||||||
|  |             } | ||||||
|  |             consumed = true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return consumed; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void gpio_scene_usb_uart_close_rpc_on_exit(void* context) { | ||||||
|  |     GpioApp* app = context; | ||||||
|  |     widget_reset(app->widget); | ||||||
|  | } | ||||||
| @ -83,11 +83,12 @@ static void usb_uart_on_irq_cb(UartIrqEvent ev, uint8_t data, void* context) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void usb_uart_vcp_init(UsbUartBridge* usb_uart, uint8_t vcp_ch) { | static void usb_uart_vcp_init(UsbUartBridge* usb_uart, uint8_t vcp_ch) { | ||||||
|  |     furi_hal_usb_unlock(); | ||||||
|     if(vcp_ch == 0) { |     if(vcp_ch == 0) { | ||||||
|         furi_hal_usb_set_config(&usb_cdc_single, NULL); |         furi_check(furi_hal_usb_set_config(&usb_cdc_single, NULL) == true); | ||||||
|         furi_hal_vcp_disable(); |         furi_hal_vcp_disable(); | ||||||
|     } else { |     } else { | ||||||
|         furi_hal_usb_set_config(&usb_cdc_dual, NULL); |         furi_check(furi_hal_usb_set_config(&usb_cdc_dual, NULL) == true); | ||||||
|     } |     } | ||||||
|     furi_hal_cdc_set_callbacks(vcp_ch, (CdcCallbacks*)&cdc_cb, usb_uart); |     furi_hal_cdc_set_callbacks(vcp_ch, (CdcCallbacks*)&cdc_cb, usb_uart); | ||||||
| } | } | ||||||
| @ -247,6 +248,7 @@ static int32_t usb_uart_worker(void* context) { | |||||||
| 
 | 
 | ||||||
|     usb_uart_vcp_deinit(usb_uart, usb_uart->cfg.vcp_ch); |     usb_uart_vcp_deinit(usb_uart, usb_uart->cfg.vcp_ch); | ||||||
|     usb_uart_serial_deinit(usb_uart, usb_uart->cfg.uart_ch); |     usb_uart_serial_deinit(usb_uart, usb_uart->cfg.uart_ch); | ||||||
|  |     furi_hal_usb_unlock(); | ||||||
|     furi_hal_usb_set_config(usb_mode_prev, NULL); |     furi_hal_usb_set_config(usb_mode_prev, NULL); | ||||||
|     if(usb_uart->cfg.flow_pins != 0) { |     if(usb_uart->cfg.flow_pins != 0) { | ||||||
|         hal_gpio_init_simple(flow_pins[usb_uart->cfg.flow_pins - 1][0], GpioModeAnalog); |         hal_gpio_init_simple(flow_pins[usb_uart->cfg.flow_pins - 1][0], GpioModeAnalog); | ||||||
|  | |||||||
| @ -1,6 +1,7 @@ | |||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
| #include <stdint.h> | #include <stdint.h> | ||||||
|  | #include <stdbool.h> | ||||||
| 
 | 
 | ||||||
| typedef struct UsbUartBridge UsbUartBridge; | typedef struct UsbUartBridge UsbUartBridge; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -2,10 +2,12 @@ | |||||||
| #include <furi.h> | #include <furi.h> | ||||||
| #include <rpc/rpc.h> | #include <rpc/rpc.h> | ||||||
| #include <furi_hal.h> | #include <furi_hal.h> | ||||||
|  | #include <semphr.h> | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     Cli* cli; |     Cli* cli; | ||||||
|     bool session_close_request; |     bool session_close_request; | ||||||
|  |     SemaphoreHandle_t terminate_semaphore; | ||||||
| } CliRpc; | } CliRpc; | ||||||
| 
 | 
 | ||||||
| #define CLI_READ_BUFFER_SIZE 64 | #define CLI_READ_BUFFER_SIZE 64 | ||||||
| @ -26,19 +28,30 @@ static void rpc_session_close_callback(void* context) { | |||||||
|     cli_rpc->session_close_request = true; |     cli_rpc->session_close_request = true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void rpc_session_terminated_callback(void* context) { | ||||||
|  |     furi_check(context); | ||||||
|  |     CliRpc* cli_rpc = context; | ||||||
|  | 
 | ||||||
|  |     xSemaphoreGive(cli_rpc->terminate_semaphore); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void rpc_cli_command_start_session(Cli* cli, string_t args, void* context) { | void rpc_cli_command_start_session(Cli* cli, string_t args, void* context) { | ||||||
|     Rpc* rpc = context; |     Rpc* rpc = context; | ||||||
| 
 | 
 | ||||||
|  |     furi_hal_usb_lock(); | ||||||
|     RpcSession* rpc_session = rpc_session_open(rpc); |     RpcSession* rpc_session = rpc_session_open(rpc); | ||||||
|     if(rpc_session == NULL) { |     if(rpc_session == NULL) { | ||||||
|         printf("Session start error\r\n"); |         printf("Session start error\r\n"); | ||||||
|  |         furi_hal_usb_unlock(); | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     CliRpc cli_rpc = {.cli = cli, .session_close_request = false}; |     CliRpc cli_rpc = {.cli = cli, .session_close_request = false}; | ||||||
|  |     cli_rpc.terminate_semaphore = xSemaphoreCreateBinary(); | ||||||
|     rpc_session_set_context(rpc_session, &cli_rpc); |     rpc_session_set_context(rpc_session, &cli_rpc); | ||||||
|     rpc_session_set_send_bytes_callback(rpc_session, rpc_send_bytes_callback); |     rpc_session_set_send_bytes_callback(rpc_session, rpc_send_bytes_callback); | ||||||
|     rpc_session_set_close_callback(rpc_session, rpc_session_close_callback); |     rpc_session_set_close_callback(rpc_session, rpc_session_close_callback); | ||||||
|  |     rpc_session_set_terminated_callback(rpc_session, rpc_session_terminated_callback); | ||||||
| 
 | 
 | ||||||
|     uint8_t* buffer = malloc(CLI_READ_BUFFER_SIZE); |     uint8_t* buffer = malloc(CLI_READ_BUFFER_SIZE); | ||||||
|     size_t size_received = 0; |     size_t size_received = 0; | ||||||
| @ -57,5 +70,11 @@ void rpc_cli_command_start_session(Cli* cli, string_t args, void* context) { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     rpc_session_close(rpc_session); |     rpc_session_close(rpc_session); | ||||||
|  | 
 | ||||||
|  |     furi_check(xSemaphoreTake(cli_rpc.terminate_semaphore, portMAX_DELAY)); | ||||||
|  | 
 | ||||||
|  |     vSemaphoreDelete(cli_rpc.terminate_semaphore); | ||||||
|  | 
 | ||||||
|     free(buffer); |     free(buffer); | ||||||
|  |     furi_hal_usb_unlock(); | ||||||
| } | } | ||||||
|  | |||||||
| @ -12,16 +12,26 @@ static void u2f_scene_error_event_callback(GuiButtonType result, InputType type, | |||||||
| void u2f_scene_error_on_enter(void* context) { | void u2f_scene_error_on_enter(void* context) { | ||||||
|     U2fApp* app = context; |     U2fApp* app = context; | ||||||
| 
 | 
 | ||||||
|     widget_add_icon_element(app->widget, 0, 0, &I_SDQuestion_35x43); |     if(app->error == U2fAppErrorNoFiles) { | ||||||
| 
 |         widget_add_icon_element(app->widget, 0, 0, &I_SDQuestion_35x43); | ||||||
|     widget_add_string_multiline_element( |         widget_add_string_multiline_element( | ||||||
|         app->widget, |             app->widget, | ||||||
|         81, |             81, | ||||||
|         4, |             4, | ||||||
|         AlignCenter, |             AlignCenter, | ||||||
|         AlignTop, |             AlignTop, | ||||||
|         FontSecondary, |             FontSecondary, | ||||||
|         "No SD card or\napp data found.\nThis app will not\nwork without\nrequired files."); |             "No SD card or\napp data found.\nThis app will not\nwork without\nrequired files."); | ||||||
|  |     } else if(app->error == U2fAppErrorCloseRpc) { | ||||||
|  |         widget_add_string_multiline_element( | ||||||
|  |             app->widget, | ||||||
|  |             63, | ||||||
|  |             10, | ||||||
|  |             AlignCenter, | ||||||
|  |             AlignTop, | ||||||
|  |             FontSecondary, | ||||||
|  |             "Disconnect from\ncompanion app\nto use this function"); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     widget_add_button_element( |     widget_add_button_element( | ||||||
|         app->widget, GuiButtonTypeLeft, "Back", u2f_scene_error_event_callback, app); |         app->widget, GuiButtonTypeLeft, "Back", u2f_scene_error_event_callback, app); | ||||||
|  | |||||||
| @ -48,10 +48,16 @@ U2fApp* u2f_app_alloc() { | |||||||
|     view_dispatcher_add_view( |     view_dispatcher_add_view( | ||||||
|         app->view_dispatcher, U2fAppViewMain, u2f_view_get_view(app->u2f_view)); |         app->view_dispatcher, U2fAppViewMain, u2f_view_get_view(app->u2f_view)); | ||||||
| 
 | 
 | ||||||
|     if(u2f_data_check(true)) { |     if(furi_hal_usb_is_locked()) { | ||||||
|         scene_manager_next_scene(app->scene_manager, U2fSceneMain); |         app->error = U2fAppErrorCloseRpc; | ||||||
|     } else { |  | ||||||
|         scene_manager_next_scene(app->scene_manager, U2fSceneError); |         scene_manager_next_scene(app->scene_manager, U2fSceneError); | ||||||
|  |     } else { | ||||||
|  |         if(u2f_data_check(true)) { | ||||||
|  |             scene_manager_next_scene(app->scene_manager, U2fSceneMain); | ||||||
|  |         } else { | ||||||
|  |             app->error = U2fAppErrorNoFiles; | ||||||
|  |             scene_manager_next_scene(app->scene_manager, U2fSceneError); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return app; |     return app; | ||||||
|  | |||||||
| @ -15,6 +15,11 @@ | |||||||
| #include "u2f_hid.h" | #include "u2f_hid.h" | ||||||
| #include "u2f.h" | #include "u2f.h" | ||||||
| 
 | 
 | ||||||
|  | typedef enum { | ||||||
|  |     U2fAppErrorNoFiles, | ||||||
|  |     U2fAppErrorCloseRpc, | ||||||
|  | } U2fAppError; | ||||||
|  | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     U2fCustomEventNone, |     U2fCustomEventNone, | ||||||
| 
 | 
 | ||||||
| @ -52,4 +57,5 @@ struct U2fApp { | |||||||
|     U2fData* u2f_instance; |     U2fData* u2f_instance; | ||||||
|     GpioCustomEvent event_cur; |     GpioCustomEvent event_cur; | ||||||
|     bool u2f_ready; |     bool u2f_ready; | ||||||
|  |     U2fAppError error; | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -191,7 +191,7 @@ static int32_t u2f_hid_worker(void* context) { | |||||||
|     FURI_LOG_D(WORKER_TAG, "Init"); |     FURI_LOG_D(WORKER_TAG, "Init"); | ||||||
| 
 | 
 | ||||||
|     FuriHalUsbInterface* usb_mode_prev = furi_hal_usb_get_config(); |     FuriHalUsbInterface* usb_mode_prev = furi_hal_usb_get_config(); | ||||||
|     furi_hal_usb_set_config(&usb_hid_u2f, NULL); |     furi_check(furi_hal_usb_set_config(&usb_hid_u2f, NULL) == true); | ||||||
| 
 | 
 | ||||||
|     u2f_hid->lock_timer = osTimerNew(u2f_hid_lock_timeout_callback, osTimerOnce, u2f_hid, NULL); |     u2f_hid->lock_timer = osTimerNew(u2f_hid_lock_timeout_callback, osTimerOnce, u2f_hid, NULL); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -16,6 +16,7 @@ typedef struct { | |||||||
|     osTimerId_t tmr; |     osTimerId_t tmr; | ||||||
|     bool enabled; |     bool enabled; | ||||||
|     bool connected; |     bool connected; | ||||||
|  |     bool mode_lock; | ||||||
|     FuriHalUsbInterface* if_cur; |     FuriHalUsbInterface* if_cur; | ||||||
|     FuriHalUsbInterface* if_next; |     FuriHalUsbInterface* if_next; | ||||||
|     void* if_ctx; |     void* if_ctx; | ||||||
| @ -89,27 +90,47 @@ void furi_hal_usb_init(void) { | |||||||
|     FURI_LOG_I(TAG, "Init OK"); |     FURI_LOG_I(TAG, "Init OK"); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void furi_hal_usb_set_config(FuriHalUsbInterface* new_if, void* ctx) { | bool furi_hal_usb_set_config(FuriHalUsbInterface* new_if, void* ctx) { | ||||||
|  |     if(usb.mode_lock) { | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     usb.if_next = new_if; |     usb.if_next = new_if; | ||||||
|     usb.if_ctx = ctx; |     usb.if_ctx = ctx; | ||||||
|     if(usb.thread == NULL) { |     if(usb.thread == NULL) { | ||||||
|         // Service thread hasn't started yet, so just save interface mode
 |         // Service thread hasn't started yet, so just save interface mode
 | ||||||
|         return; |         return true; | ||||||
|     } |     } | ||||||
|     furi_assert(usb.thread); |     furi_assert(usb.thread); | ||||||
|     osThreadFlagsSet(furi_thread_get_thread_id(usb.thread), EventModeChange); |     osThreadFlagsSet(furi_thread_get_thread_id(usb.thread), EventModeChange); | ||||||
|  |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| FuriHalUsbInterface* furi_hal_usb_get_config() { | FuriHalUsbInterface* furi_hal_usb_get_config() { | ||||||
|     return usb.if_cur; |     return usb.if_cur; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void furi_hal_usb_lock() { | ||||||
|  |     FURI_LOG_I(TAG, "Mode lock"); | ||||||
|  |     usb.mode_lock = true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void furi_hal_usb_unlock() { | ||||||
|  |     FURI_LOG_I(TAG, "Mode unlock"); | ||||||
|  |     usb.mode_lock = false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool furi_hal_usb_is_locked() { | ||||||
|  |     return usb.mode_lock; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void furi_hal_usb_disable() { | void furi_hal_usb_disable() { | ||||||
|     furi_assert(usb.thread); |     furi_assert(usb.thread); | ||||||
|     osThreadFlagsSet(furi_thread_get_thread_id(usb.thread), EventDisable); |     osThreadFlagsSet(furi_thread_get_thread_id(usb.thread), EventDisable); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void furi_hal_usb_enable() { | void furi_hal_usb_enable() { | ||||||
|  |     furi_assert(usb.thread); | ||||||
|     osThreadFlagsSet(furi_thread_get_thread_id(usb.thread), EventEnable); |     osThreadFlagsSet(furi_thread_get_thread_id(usb.thread), EventEnable); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -42,8 +42,9 @@ void furi_hal_usb_init(); | |||||||
|  * |  * | ||||||
|  * @param      mode new USB device mode |  * @param      mode new USB device mode | ||||||
|  * @param      ctx context passed to device mode init function |  * @param      ctx context passed to device mode init function | ||||||
|  |  * @return     true - mode switch started, false - mode switch is locked | ||||||
|  */ |  */ | ||||||
| void furi_hal_usb_set_config(FuriHalUsbInterface* new_if, void* ctx); | bool furi_hal_usb_set_config(FuriHalUsbInterface* new_if, void* ctx); | ||||||
| 
 | 
 | ||||||
| /** Get USB device configuration
 | /** Get USB device configuration
 | ||||||
|  * |  * | ||||||
| @ -51,6 +52,20 @@ void furi_hal_usb_set_config(FuriHalUsbInterface* new_if, void* ctx); | |||||||
|  */ |  */ | ||||||
| FuriHalUsbInterface* furi_hal_usb_get_config(); | FuriHalUsbInterface* furi_hal_usb_get_config(); | ||||||
| 
 | 
 | ||||||
|  | /** Lock USB device mode switch
 | ||||||
|  |  */ | ||||||
|  | void furi_hal_usb_lock(); | ||||||
|  | 
 | ||||||
|  | /** Unlock USB device mode switch
 | ||||||
|  |  */ | ||||||
|  | void furi_hal_usb_unlock(); | ||||||
|  | 
 | ||||||
|  | /** Check if USB device mode switch locked
 | ||||||
|  |  *  | ||||||
|  |  * @return    lock state | ||||||
|  |  */ | ||||||
|  | bool furi_hal_usb_is_locked(); | ||||||
|  | 
 | ||||||
| /** Disable USB device
 | /** Disable USB device
 | ||||||
|  */ |  */ | ||||||
| void furi_hal_usb_disable(); | void furi_hal_usb_disable(); | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Aleksandr Kutuzov
						Aleksandr Kutuzov