[FL-1815, FL-1851, FL-1856] SubGhz: preparation for certification, add deleting stored signals and rename file in SubGHz app (#714)
* [FL-1811] FuriHal: move core2 startup to hal init stage, prevent working with flash controller till core2 startup finish. #704 * SubGhz: fix GO0 low on last hop transmission, decreased DutyCycle in tests * SubGhz: test_static fix max 5 sec in transmission mode, DutyCycle <23% * [FL-1815] SubGhz: prohibiting transmission if it is not within the permitted range for the given region * SubGhz: fix F7 furi-hal-subghz * SubGhz: fix logic working tests * SubGhz: fix princeton encoder for test * SubGhz: add log princeton encoder * [FL-1856] Subghz: fix output a double error if the file cannot be opened * [FL-1851] SubGhz: add deleting Stored Signals in SubGHz App * SubGhz: add rename file SubGhz app * SubGhz: update stats message in princeton * SubGhz: correct spelling * SubGhz: fix FM config, add hardware signal processing less than 16 μs, add added filter for processing short signals * SubGhz: add Scher-Khan MAGICAR Dinamic protocol * SubGhz: sync fury targets Co-authored-by: あく <alleteam@gmail.com>
This commit is contained in:
		
							parent
							
								
									416e1bda35
								
							
						
					
					
						commit
						a8981d317a
					
				| @ -7,6 +7,10 @@ ADD_SCENE(subghz, save_success, SaveSuccess) | |||||||
| ADD_SCENE(subghz, saved, Saved) | ADD_SCENE(subghz, saved, Saved) | ||||||
| ADD_SCENE(subghz, transmitter, Transmitter) | ADD_SCENE(subghz, transmitter, Transmitter) | ||||||
| ADD_SCENE(subghz, show_error, ShowError) | ADD_SCENE(subghz, show_error, ShowError) | ||||||
|  | ADD_SCENE(subghz, show_only_rx, ShowOnlyRx) | ||||||
|  | ADD_SCENE(subghz, saved_menu, SavedMenu) | ||||||
|  | ADD_SCENE(subghz, delete, Delete) | ||||||
|  | ADD_SCENE(subghz, delete_success, DeleteSuccess) | ||||||
| ADD_SCENE(subghz, test, Test) | ADD_SCENE(subghz, test, Test) | ||||||
| ADD_SCENE(subghz, test_static, TestStatic) | ADD_SCENE(subghz, test_static, TestStatic) | ||||||
| ADD_SCENE(subghz, test_carrier, TestCarrier) | ADD_SCENE(subghz, test_carrier, TestCarrier) | ||||||
|  | |||||||
							
								
								
									
										71
									
								
								applications/subghz/scenes/subghz_scene_delete.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								applications/subghz/scenes/subghz_scene_delete.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | |||||||
|  | #include "../subghz_i.h" | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SubGhzSceneDeleteInfoCustomEventDelete, | ||||||
|  | } SubGhzSceneDeleteInfoCustomEvent; | ||||||
|  | 
 | ||||||
|  | void subghz_scene_delete_callback(GuiButtonType result, InputType type, void* context) { | ||||||
|  |     furi_assert(context); | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     if((result == GuiButtonTypeRight) && (type == InputTypeShort)) { | ||||||
|  |         view_dispatcher_send_custom_event( | ||||||
|  |             subghz->view_dispatcher, SubGhzSceneDeleteInfoCustomEventDelete); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_scene_delete_on_enter(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     char buffer_str[16]; | ||||||
|  |     snprintf( | ||||||
|  |         buffer_str, | ||||||
|  |         sizeof(buffer_str), | ||||||
|  |         "%03ld.%02ld", | ||||||
|  |         subghz->txrx->frequency / 1000000 % 1000, | ||||||
|  |         subghz->txrx->frequency / 10000 % 100); | ||||||
|  |     widget_add_string_element( | ||||||
|  |         subghz->widget, 78, 0, AlignLeft, AlignTop, FontSecondary, buffer_str); | ||||||
|  |     if(subghz->txrx->preset == FuriHalSubGhzPresetOok650Async || | ||||||
|  |        subghz->txrx->preset == FuriHalSubGhzPresetOok270Async) { | ||||||
|  |         snprintf(buffer_str, sizeof(buffer_str), "AM"); | ||||||
|  |     } else if(subghz->txrx->preset == FuriHalSubGhzPreset2FSKAsync) { | ||||||
|  |         snprintf(buffer_str, sizeof(buffer_str), "FM"); | ||||||
|  |     } else { | ||||||
|  |         furi_crash(NULL); | ||||||
|  |     } | ||||||
|  |     widget_add_string_element( | ||||||
|  |         subghz->widget, 113, 0, AlignLeft, AlignTop, FontSecondary, buffer_str); | ||||||
|  |     string_t text; | ||||||
|  |     string_init(text); | ||||||
|  |     subghz->txrx->protocol_result->to_string(subghz->txrx->protocol_result, text); | ||||||
|  |     widget_add_string_multiline_element( | ||||||
|  |         subghz->widget, 0, 0, AlignLeft, AlignTop, FontSecondary, string_get_cstr(text)); | ||||||
|  |     string_clear(text); | ||||||
|  | 
 | ||||||
|  |     widget_add_button_element( | ||||||
|  |         subghz->widget, GuiButtonTypeRight, "Delete", subghz_scene_delete_callback, subghz); | ||||||
|  | 
 | ||||||
|  |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewWidget); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool subghz_scene_delete_on_event(void* context, SceneManagerEvent event) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SubGhzSceneDeleteInfoCustomEventDelete) { | ||||||
|  |             memcpy(subghz->file_name_tmp, subghz->file_name, strlen(subghz->file_name)); | ||||||
|  |             if(subghz_delete_file(subghz)) { | ||||||
|  |                 scene_manager_next_scene(subghz->scene_manager, SubGhzSceneDeleteSuccess); | ||||||
|  |             } else { | ||||||
|  |                 scene_manager_search_and_switch_to_previous_scene( | ||||||
|  |                     subghz->scene_manager, SubGhzSceneStart); | ||||||
|  |             } | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_scene_delete_on_exit(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     widget_clear(subghz->widget); | ||||||
|  | } | ||||||
							
								
								
									
										48
									
								
								applications/subghz/scenes/subghz_scene_delete_success.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								applications/subghz/scenes/subghz_scene_delete_success.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | |||||||
|  | #include "../subghz_i.h" | ||||||
|  | 
 | ||||||
|  | #define SCENE_DELETE_SUCCESS_CUSTOM_EVENT (0UL) | ||||||
|  | 
 | ||||||
|  | void subghz_scene_delete_success_popup_callback(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     view_dispatcher_send_custom_event(subghz->view_dispatcher, SCENE_DELETE_SUCCESS_CUSTOM_EVENT); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_scene_delete_success_on_enter(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     // Setup view
 | ||||||
|  |     Popup* popup = subghz->popup; | ||||||
|  |     popup_set_icon(popup, 0, 2, &I_DolphinMafia_115x62); | ||||||
|  |     popup_set_header(popup, "Deleted", 83, 19, AlignLeft, AlignBottom); | ||||||
|  |     popup_set_timeout(popup, 1500); | ||||||
|  |     popup_set_context(popup, subghz); | ||||||
|  |     popup_set_callback(popup, subghz_scene_delete_success_popup_callback); | ||||||
|  |     popup_enable_timeout(popup); | ||||||
|  |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewPopup); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool subghz_scene_delete_success_on_event(void* context, SceneManagerEvent event) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SCENE_DELETE_SUCCESS_CUSTOM_EVENT) { | ||||||
|  |             return scene_manager_search_and_switch_to_previous_scene( | ||||||
|  |                 subghz->scene_manager, SubGhzSceneStart); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_scene_delete_success_on_exit(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     // Clear view
 | ||||||
|  |     Popup* popup = subghz->popup; | ||||||
|  |     popup_set_header(popup, NULL, 0, 0, AlignCenter, AlignBottom); | ||||||
|  |     popup_set_text(popup, NULL, 0, 0, AlignCenter, AlignTop); | ||||||
|  |     popup_set_icon(popup, 0, 0, NULL); | ||||||
|  |     popup_set_callback(popup, NULL); | ||||||
|  |     popup_set_context(popup, NULL); | ||||||
|  |     popup_set_timeout(popup, 0); | ||||||
|  |     popup_disable_timeout(popup); | ||||||
|  | } | ||||||
| @ -101,7 +101,6 @@ bool subghz_scene_receiver_info_on_event(void* context, SceneManagerEvent event) | |||||||
|     if(event.type == SceneManagerEventTypeCustom) { |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|         if(event.event == SubGhzSceneReceiverInfoCustomEventTxStart) { |         if(event.event == SubGhzSceneReceiverInfoCustomEventTxStart) { | ||||||
|             //CC1101 Stop RX -> Start TX
 |             //CC1101 Stop RX -> Start TX
 | ||||||
|             subghz->state_notifications = NOTIFICATION_TX_STATE; |  | ||||||
|             if(subghz->txrx->hopper_state != SubGhzHopperStateOFF) { |             if(subghz->txrx->hopper_state != SubGhzHopperStateOFF) { | ||||||
|                 subghz->txrx->hopper_state = SubGhzHopperStatePause; |                 subghz->txrx->hopper_state = SubGhzHopperStatePause; | ||||||
|             } |             } | ||||||
| @ -112,7 +111,11 @@ bool subghz_scene_receiver_info_on_event(void* context, SceneManagerEvent event) | |||||||
|                 return false; |                 return false; | ||||||
|             } |             } | ||||||
|             if(subghz->txrx->txrx_state == SubGhzTxRxStateIdle) { |             if(subghz->txrx->txrx_state == SubGhzTxRxStateIdle) { | ||||||
|                 subghz_tx_start(subghz); |                 if(!subghz_tx_start(subghz)) { | ||||||
|  |                     scene_manager_next_scene(subghz->scene_manager, SubGhzSceneShowOnlyRx); | ||||||
|  |                 } else { | ||||||
|  |                     subghz->state_notifications = NOTIFICATION_TX_STATE; | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             return true; |             return true; | ||||||
|         } else if(event.event == SubGhzSceneReceiverInfoCustomEventTxStop) { |         } else if(event.event == SubGhzSceneReceiverInfoCustomEventTxStop) { | ||||||
| @ -145,6 +148,7 @@ bool subghz_scene_receiver_info_on_event(void* context, SceneManagerEvent event) | |||||||
|             } |             } | ||||||
|             if(subghz->txrx->protocol_result && subghz->txrx->protocol_result->to_save_string && |             if(subghz->txrx->protocol_result && subghz->txrx->protocol_result->to_save_string && | ||||||
|                strcmp(subghz->txrx->protocol_result->name, "KeeLoq")) { |                strcmp(subghz->txrx->protocol_result->name, "KeeLoq")) { | ||||||
|  |                 subghz_file_name_clear(subghz); | ||||||
|                 scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveName); |                 scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveName); | ||||||
|             } |             } | ||||||
|             return true; |             return true; | ||||||
|  | |||||||
| @ -16,15 +16,19 @@ void subghz_scene_save_name_on_enter(void* context) { | |||||||
|     TextInput* text_input = subghz->text_input; |     TextInput* text_input = subghz->text_input; | ||||||
|     bool dev_name_empty = false; |     bool dev_name_empty = false; | ||||||
| 
 | 
 | ||||||
|     set_random_name(subghz->text_store, sizeof(subghz->text_store)); |     if(!strcmp(subghz->file_name, "")) { | ||||||
|  |         set_random_name(subghz->file_name, sizeof(subghz->file_name)); | ||||||
|         dev_name_empty = true; |         dev_name_empty = true; | ||||||
|  |     } else { | ||||||
|  |         memcpy(subghz->file_name_tmp, subghz->file_name, strlen(subghz->file_name)); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     text_input_set_header_text(text_input, "Name signal"); |     text_input_set_header_text(text_input, "Name signal"); | ||||||
|     text_input_set_result_callback( |     text_input_set_result_callback( | ||||||
|         text_input, |         text_input, | ||||||
|         subghz_scene_save_name_text_input_callback, |         subghz_scene_save_name_text_input_callback, | ||||||
|         subghz, |         subghz, | ||||||
|         subghz->text_store, |         subghz->file_name, | ||||||
|         22, //Max len name
 |         22, //Max len name
 | ||||||
|         dev_name_empty); |         dev_name_empty); | ||||||
|     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewTextInput); |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewTextInput); | ||||||
| @ -35,8 +39,12 @@ bool subghz_scene_save_name_on_event(void* context, SceneManagerEvent event) { | |||||||
| 
 | 
 | ||||||
|     if(event.type == SceneManagerEventTypeCustom) { |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|         if(event.event == SCENE_SAVE_NAME_CUSTOM_EVENT) { |         if(event.event == SCENE_SAVE_NAME_CUSTOM_EVENT) { | ||||||
|             if(strcmp(subghz->text_store, "") && |             if(strcmp(subghz->file_name, "") && | ||||||
|                subghz_save_protocol_to_file(subghz, subghz->text_store)) { |                subghz_save_protocol_to_file(subghz, subghz->file_name)) { | ||||||
|  |                 if(strcmp(subghz->file_name_tmp, "")) { | ||||||
|  |                     subghz_delete_file(subghz); | ||||||
|  |                 } | ||||||
|  |                 subghz_file_name_clear(subghz); | ||||||
|                 scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveSuccess); |                 scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveSuccess); | ||||||
|                 return true; |                 return true; | ||||||
|             } else { |             } else { | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ void subghz_scene_saved_on_enter(void* context) { | |||||||
|     SubGhz* subghz = context; |     SubGhz* subghz = context; | ||||||
| 
 | 
 | ||||||
|     if(subghz_load_protocol_from_file(subghz)) { |     if(subghz_load_protocol_from_file(subghz)) { | ||||||
|         scene_manager_next_scene(subghz->scene_manager, SubGhzSceneTransmitter); |         scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSavedMenu); | ||||||
|     } else { |     } else { | ||||||
|         scene_manager_search_and_switch_to_previous_scene(subghz->scene_manager, SubGhzSceneStart); |         scene_manager_search_and_switch_to_previous_scene(subghz->scene_manager, SubGhzSceneStart); | ||||||
|     } |     } | ||||||
|  | |||||||
							
								
								
									
										69
									
								
								applications/subghz/scenes/subghz_scene_saved_menu.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								applications/subghz/scenes/subghz_scene_saved_menu.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | |||||||
|  | #include "../subghz_i.h" | ||||||
|  | 
 | ||||||
|  | enum SubmenuIndex { | ||||||
|  |     SubmenuIndexEmulate, | ||||||
|  |     SubmenuIndexEdit, | ||||||
|  |     SubmenuIndexDelete, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | void subghz_scene_saved_menu_submenu_callback(void* context, uint32_t index) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     view_dispatcher_send_custom_event(subghz->view_dispatcher, index); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_scene_saved_menu_on_enter(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     submenu_add_item( | ||||||
|  |         subghz->submenu, | ||||||
|  |         "Emulate", | ||||||
|  |         SubmenuIndexEmulate, | ||||||
|  |         subghz_scene_saved_menu_submenu_callback, | ||||||
|  |         subghz); | ||||||
|  |     submenu_add_item( | ||||||
|  |         subghz->submenu, | ||||||
|  |         "Edit name", | ||||||
|  |         SubmenuIndexEdit, | ||||||
|  |         subghz_scene_saved_menu_submenu_callback, | ||||||
|  |         subghz); | ||||||
|  |     submenu_add_item( | ||||||
|  |         subghz->submenu, | ||||||
|  |         "Delete", | ||||||
|  |         SubmenuIndexDelete, | ||||||
|  |         subghz_scene_saved_menu_submenu_callback, | ||||||
|  |         subghz); | ||||||
|  | 
 | ||||||
|  |     submenu_set_selected_item( | ||||||
|  |         subghz->submenu, | ||||||
|  |         scene_manager_get_scene_state(subghz->scene_manager, SubGhzSceneSavedMenu)); | ||||||
|  | 
 | ||||||
|  |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewMenu); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool subghz_scene_saved_menu_on_event(void* context, SceneManagerEvent event) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SubmenuIndexEmulate) { | ||||||
|  |             scene_manager_set_scene_state( | ||||||
|  |                 subghz->scene_manager, SubGhzSceneSavedMenu, SubmenuIndexEmulate); | ||||||
|  |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneTransmitter); | ||||||
|  |             return true; | ||||||
|  |         } else if(event.event == SubmenuIndexDelete) { | ||||||
|  |             scene_manager_set_scene_state( | ||||||
|  |                 subghz->scene_manager, SubGhzSceneSavedMenu, SubmenuIndexDelete); | ||||||
|  |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneDelete); | ||||||
|  |             return true; | ||||||
|  |         } else if(event.event == SubmenuIndexEdit) { | ||||||
|  |             scene_manager_set_scene_state( | ||||||
|  |                 subghz->scene_manager, SubGhzSceneSavedMenu, SubmenuIndexEdit); | ||||||
|  |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveName); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_scene_saved_menu_on_exit(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     submenu_clean(subghz->submenu); | ||||||
|  | } | ||||||
| @ -174,6 +174,7 @@ bool subghz_scene_set_type_on_event(void* context, SceneManagerEvent event) { | |||||||
|         if(generated_protocol) { |         if(generated_protocol) { | ||||||
|             subghz->txrx->frequency = subghz_frequencies[subghz_frequencies_433_92]; |             subghz->txrx->frequency = subghz_frequencies[subghz_frequencies_433_92]; | ||||||
|             subghz->txrx->preset = FuriHalSubGhzPresetOok650Async; |             subghz->txrx->preset = FuriHalSubGhzPresetOok650Async; | ||||||
|  |             subghz_file_name_clear(subghz); | ||||||
|             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveName); |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneSaveName); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|  | |||||||
							
								
								
									
										53
									
								
								applications/subghz/scenes/subghz_scene_show_only_rx.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								applications/subghz/scenes/subghz_scene_show_only_rx.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | #include "../subghz_i.h" | ||||||
|  | 
 | ||||||
|  | #define SCENE_NO_MAN_CUSTOM_EVENT (11UL) | ||||||
|  | 
 | ||||||
|  | void subghz_scene_show_only_rx_popup_callback(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     view_dispatcher_send_custom_event(subghz->view_dispatcher, SCENE_NO_MAN_CUSTOM_EVENT); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const void subghz_scene_show_only_rx_on_enter(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     // Setup view
 | ||||||
|  |     Popup* popup = subghz->popup; | ||||||
|  |     popup_set_icon(popup, 67, 12, &I_DolphinFirstStart7_61x51); | ||||||
|  |     popup_set_text( | ||||||
|  |         popup, | ||||||
|  |         "This frequency can\nonly be used for RX\nin your region", | ||||||
|  |         38, | ||||||
|  |         40, | ||||||
|  |         AlignCenter, | ||||||
|  |         AlignBottom); | ||||||
|  |     popup_set_timeout(popup, 1500); | ||||||
|  |     popup_set_context(popup, subghz); | ||||||
|  |     popup_set_callback(popup, subghz_scene_show_only_rx_popup_callback); | ||||||
|  |     popup_enable_timeout(popup); | ||||||
|  |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewPopup); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const bool subghz_scene_show_only_rx_on_event(void* context, SceneManagerEvent event) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SCENE_NO_MAN_CUSTOM_EVENT) { | ||||||
|  |             scene_manager_previous_scene(subghz->scene_manager); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const void subghz_scene_show_only_rx_on_exit(void* context) { | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  | 
 | ||||||
|  |     // Clear view
 | ||||||
|  |     Popup* popup = subghz->popup; | ||||||
|  |     popup_set_header(popup, NULL, 0, 0, AlignCenter, AlignBottom); | ||||||
|  |     popup_set_text(popup, NULL, 0, 0, AlignCenter, AlignTop); | ||||||
|  |     popup_set_icon(popup, 0, 0, NULL); | ||||||
|  |     popup_set_callback(popup, NULL); | ||||||
|  |     popup_set_context(popup, NULL); | ||||||
|  |     popup_set_timeout(popup, 0); | ||||||
|  |     popup_disable_timeout(popup); | ||||||
|  | } | ||||||
| @ -1,12 +1,27 @@ | |||||||
| #include "../subghz_i.h" | #include "../subghz_i.h" | ||||||
|  | #include "../views/subghz_test_carrier.h" | ||||||
|  | 
 | ||||||
|  | void subghz_scene_test_carrier_callback(SubghzTestCarrierEvent event, void* context) { | ||||||
|  |     furi_assert(context); | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     view_dispatcher_send_custom_event(subghz->view_dispatcher, event); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| void subghz_scene_test_carrier_on_enter(void* context) { | void subghz_scene_test_carrier_on_enter(void* context) { | ||||||
|     SubGhz* subghz = context; |     SubGhz* subghz = context; | ||||||
|  |     subghz_test_carrier_set_callback( | ||||||
|  |         subghz->subghz_test_carrier, subghz_scene_test_carrier_callback, subghz); | ||||||
|     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewTestCarrier); |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewTestCarrier); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool subghz_scene_test_carrier_on_event(void* context, SceneManagerEvent event) { | bool subghz_scene_test_carrier_on_event(void* context, SceneManagerEvent event) { | ||||||
|     // SubGhz* subghz = context;
 |     SubGhz* subghz = context; | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SubghzTestCarrierEventOnlyRx) { | ||||||
|  |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneShowOnlyRx); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,12 +1,27 @@ | |||||||
| #include "../subghz_i.h" | #include "../subghz_i.h" | ||||||
|  | #include "../views/subghz_test_packet.h" | ||||||
|  | 
 | ||||||
|  | void subghz_scene_test_packet_callback(SubghzTestPacketEvent event, void* context) { | ||||||
|  |     furi_assert(context); | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     view_dispatcher_send_custom_event(subghz->view_dispatcher, event); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| void subghz_scene_test_packet_on_enter(void* context) { | void subghz_scene_test_packet_on_enter(void* context) { | ||||||
|     SubGhz* subghz = context; |     SubGhz* subghz = context; | ||||||
|  |     subghz_test_packet_set_callback( | ||||||
|  |         subghz->subghz_test_packet, subghz_scene_test_packet_callback, subghz); | ||||||
|     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewTestPacket); |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewTestPacket); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool subghz_scene_test_packet_on_event(void* context, SceneManagerEvent event) { | bool subghz_scene_test_packet_on_event(void* context, SceneManagerEvent event) { | ||||||
|     // SubGhz* subghz = context;
 |     SubGhz* subghz = context; | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SubghzTestPacketEventOnlyRx) { | ||||||
|  |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneShowOnlyRx); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,12 +1,27 @@ | |||||||
| #include "../subghz_i.h" | #include "../subghz_i.h" | ||||||
|  | #include "../views/subghz_test_static.h" | ||||||
|  | 
 | ||||||
|  | void subghz_scene_test_static_callback(SubghzTestStaticEvent event, void* context) { | ||||||
|  |     furi_assert(context); | ||||||
|  |     SubGhz* subghz = context; | ||||||
|  |     view_dispatcher_send_custom_event(subghz->view_dispatcher, event); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| void subghz_scene_test_static_on_enter(void* context) { | void subghz_scene_test_static_on_enter(void* context) { | ||||||
|     SubGhz* subghz = context; |     SubGhz* subghz = context; | ||||||
|  |     subghz_test_static_set_callback( | ||||||
|  |         subghz->subghz_test_static, subghz_scene_test_static_callback, subghz); | ||||||
|     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewStatic); |     view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewStatic); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool subghz_scene_test_static_on_event(void* context, SceneManagerEvent event) { | bool subghz_scene_test_static_on_event(void* context, SceneManagerEvent event) { | ||||||
|     // SubGhz* subghz = context;
 |     SubGhz* subghz = context; | ||||||
|  |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|  |         if(event.event == SubghzTestStaticEventOnlyRx) { | ||||||
|  |             scene_manager_next_scene(subghz->scene_manager, SubGhzSceneShowOnlyRx); | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -68,15 +68,19 @@ bool subghz_scene_transmitter_on_event(void* context, SceneManagerEvent event) { | |||||||
|     SubGhz* subghz = context; |     SubGhz* subghz = context; | ||||||
|     if(event.type == SceneManagerEventTypeCustom) { |     if(event.type == SceneManagerEventTypeCustom) { | ||||||
|         if(event.event == SubghzTransmitterEventSendStart) { |         if(event.event == SubghzTransmitterEventSendStart) { | ||||||
|             subghz->state_notifications = NOTIFICATION_TX_STATE; |             subghz->state_notifications = NOTIFICATION_IDLE_STATE; | ||||||
|             if(subghz->txrx->txrx_state == SubGhzTxRxStateRx) { |             if(subghz->txrx->txrx_state == SubGhzTxRxStateRx) { | ||||||
|                 subghz_rx_end(subghz); |                 subghz_rx_end(subghz); | ||||||
|             } |             } | ||||||
|             if((subghz->txrx->txrx_state == SubGhzTxRxStateIdle) || |             if((subghz->txrx->txrx_state == SubGhzTxRxStateIdle) || | ||||||
|                (subghz->txrx->txrx_state == SubGhzTxRxStateSleep)) { |                (subghz->txrx->txrx_state == SubGhzTxRxStateSleep)) { | ||||||
|                 subghz_tx_start(subghz); |                 if(!subghz_tx_start(subghz)) { | ||||||
|  |                     scene_manager_next_scene(subghz->scene_manager, SubGhzSceneShowOnlyRx); | ||||||
|  |                 } else { | ||||||
|  |                     subghz->state_notifications = NOTIFICATION_TX_STATE; | ||||||
|                     subghz_scene_transmitter_update_data_show(subghz); |                     subghz_scene_transmitter_update_data_show(subghz); | ||||||
|                 } |                 } | ||||||
|  |             } | ||||||
|             return true; |             return true; | ||||||
|         } else if(event.event == SubghzTransmitterEventSendStop) { |         } else if(event.event == SubghzTransmitterEventSendStop) { | ||||||
|             subghz->state_notifications = NOTIFICATION_IDLE_STATE; |             subghz->state_notifications = NOTIFICATION_IDLE_STATE; | ||||||
|  | |||||||
| @ -48,13 +48,15 @@ void subghz_cli_command_tx_carrier(Cli* cli, string_t args, void* context) { | |||||||
|     hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); |     hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); | ||||||
|     hal_gpio_write(&gpio_cc1101_g0, true); |     hal_gpio_write(&gpio_cc1101_g0, true); | ||||||
| 
 | 
 | ||||||
|     furi_hal_subghz_tx(); |     if(furi_hal_subghz_tx()) { | ||||||
| 
 |  | ||||||
|         printf("Transmitting at frequency %lu Hz\r\n", frequency); |         printf("Transmitting at frequency %lu Hz\r\n", frequency); | ||||||
|         printf("Press CTRL+C to stop\r\n"); |         printf("Press CTRL+C to stop\r\n"); | ||||||
|         while(!cli_cmd_interrupt_received(cli)) { |         while(!cli_cmd_interrupt_received(cli)) { | ||||||
|             osDelay(250); |             osDelay(250); | ||||||
|         } |         } | ||||||
|  |     } else { | ||||||
|  |         printf("This frequency can only be used for RX in your region\r\n"); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     furi_hal_subghz_set_path(FuriHalSubGhzPathIsolate); |     furi_hal_subghz_set_path(FuriHalSubGhzPathIsolate); | ||||||
|     furi_hal_subghz_sleep(); |     furi_hal_subghz_sleep(); | ||||||
|  | |||||||
| @ -40,19 +40,19 @@ uint32_t subghz_rx(SubGhz* subghz, uint32_t frequency) { | |||||||
|     return value; |     return value; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| uint32_t subghz_tx(SubGhz* subghz, uint32_t frequency) { | static bool subghz_tx(SubGhz* subghz, uint32_t frequency) { | ||||||
|     furi_assert(subghz); |     furi_assert(subghz); | ||||||
|     if(!furi_hal_subghz_is_frequency_valid(frequency)) { |     if(!furi_hal_subghz_is_frequency_valid(frequency)) { | ||||||
|         furi_crash(NULL); |         furi_crash(NULL); | ||||||
|     } |     } | ||||||
|     furi_assert(subghz->txrx->txrx_state != SubGhzTxRxStateSleep); |     furi_assert(subghz->txrx->txrx_state != SubGhzTxRxStateSleep); | ||||||
|     furi_hal_subghz_idle(); |     furi_hal_subghz_idle(); | ||||||
|     uint32_t value = furi_hal_subghz_set_frequency_and_path(frequency); |     furi_hal_subghz_set_frequency_and_path(frequency); | ||||||
|     hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); |     hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); | ||||||
|     hal_gpio_write(&gpio_cc1101_g0, true); |     hal_gpio_write(&gpio_cc1101_g0, true); | ||||||
|     furi_hal_subghz_tx(); |     bool ret = furi_hal_subghz_tx(); | ||||||
|     subghz->txrx->txrx_state = SubGhzTxRxStateTx; |     subghz->txrx->txrx_state = SubGhzTxRxStateTx; | ||||||
|     return value; |     return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void subghz_idle(SubGhz* subghz) { | void subghz_idle(SubGhz* subghz) { | ||||||
| @ -90,9 +90,10 @@ static void subghz_frequency_preset_to_str(SubGhz* subghz, string_t output) { | |||||||
|         (int)subghz->txrx->preset); |         (int)subghz->txrx->preset); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void subghz_tx_start(SubGhz* subghz) { | bool subghz_tx_start(SubGhz* subghz) { | ||||||
|     furi_assert(subghz); |     furi_assert(subghz); | ||||||
| 
 | 
 | ||||||
|  |     bool ret = false; | ||||||
|     subghz->txrx->encoder = subghz_protocol_encoder_common_alloc(); |     subghz->txrx->encoder = subghz_protocol_encoder_common_alloc(); | ||||||
|     subghz->txrx->encoder->repeat = 200; //max repeat with the button held down
 |     subghz->txrx->encoder->repeat = 200; //max repeat with the button held down
 | ||||||
|     //get upload
 |     //get upload
 | ||||||
| @ -105,16 +106,23 @@ void subghz_tx_start(SubGhz* subghz) { | |||||||
|                 subghz_begin(subghz, FuriHalSubGhzPresetOok270Async); |                 subghz_begin(subghz, FuriHalSubGhzPresetOok270Async); | ||||||
|             } |             } | ||||||
|             if(subghz->txrx->frequency) { |             if(subghz->txrx->frequency) { | ||||||
|                 subghz_tx(subghz, subghz->txrx->frequency); |                 ret = subghz_tx(subghz, subghz->txrx->frequency); | ||||||
|             } else { |             } else { | ||||||
|                 subghz_tx(subghz, 433920000); |                 ret = subghz_tx(subghz, 433920000); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |             if(ret) { | ||||||
|                 //Start TX
 |                 //Start TX
 | ||||||
|                 furi_hal_subghz_start_async_tx( |                 furi_hal_subghz_start_async_tx( | ||||||
|                     subghz_protocol_encoder_common_yield, subghz->txrx->encoder); |                     subghz_protocol_encoder_common_yield, subghz->txrx->encoder); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } | ||||||
|  |     if(!ret) { | ||||||
|  |         subghz_protocol_encoder_common_free(subghz->txrx->encoder); | ||||||
|  |         subghz_idle(subghz); | ||||||
|  |     } | ||||||
|  |     return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void subghz_tx_stop(SubGhz* subghz) { | void subghz_tx_stop(SubGhz* subghz) { | ||||||
| @ -125,8 +133,9 @@ void subghz_tx_stop(SubGhz* subghz) { | |||||||
|     subghz_protocol_encoder_common_free(subghz->txrx->encoder); |     subghz_protocol_encoder_common_free(subghz->txrx->encoder); | ||||||
|     subghz_idle(subghz); |     subghz_idle(subghz); | ||||||
|     //if protocol dynamic then we save the last upload
 |     //if protocol dynamic then we save the last upload
 | ||||||
|     if(subghz->txrx->protocol_result->type_protocol == SubGhzProtocolCommonTypeDynamic) { |     if((subghz->txrx->protocol_result->type_protocol == SubGhzProtocolCommonTypeDynamic) && | ||||||
|         subghz_save_protocol_to_file(subghz, subghz->text_store); |        (strcmp(subghz->file_name, ""))) { | ||||||
|  |         subghz_save_protocol_to_file(subghz, subghz->file_name); | ||||||
|     } |     } | ||||||
|     notification_message(subghz->notifications, &sequence_reset_red); |     notification_message(subghz->notifications, &sequence_reset_red); | ||||||
| } | } | ||||||
| @ -268,8 +277,8 @@ bool subghz_load_protocol_from_file(SubGhz* subghz) { | |||||||
|         file_worker, |         file_worker, | ||||||
|         SUBGHZ_APP_PATH_FOLDER, |         SUBGHZ_APP_PATH_FOLDER, | ||||||
|         SUBGHZ_APP_EXTENSION, |         SUBGHZ_APP_EXTENSION, | ||||||
|         subghz->text_store, |         subghz->file_name, | ||||||
|         sizeof(subghz->text_store), |         sizeof(subghz->file_name), | ||||||
|         NULL); |         NULL); | ||||||
| 
 | 
 | ||||||
|     if(res) { |     if(res) { | ||||||
| @ -278,7 +287,7 @@ bool subghz_load_protocol_from_file(SubGhz* subghz) { | |||||||
|             protocol_file_name, |             protocol_file_name, | ||||||
|             "%s/%s%s", |             "%s/%s%s", | ||||||
|             SUBGHZ_APP_PATH_FOLDER, |             SUBGHZ_APP_PATH_FOLDER, | ||||||
|             subghz->text_store, |             subghz->file_name, | ||||||
|             SUBGHZ_APP_EXTENSION); |             SUBGHZ_APP_EXTENSION); | ||||||
|     } else { |     } else { | ||||||
|         string_clear(temp_str); |         string_clear(temp_str); | ||||||
| @ -292,7 +301,7 @@ bool subghz_load_protocol_from_file(SubGhz* subghz) { | |||||||
|     do { |     do { | ||||||
|         if(!file_worker_open( |         if(!file_worker_open( | ||||||
|                file_worker, string_get_cstr(protocol_file_name), FSAM_READ, FSOM_OPEN_EXISTING)) { |                file_worker, string_get_cstr(protocol_file_name), FSAM_READ, FSOM_OPEN_EXISTING)) { | ||||||
|             break; |             return res; | ||||||
|         } |         } | ||||||
|         // Read and parse frequency from 1st line
 |         // Read and parse frequency from 1st line
 | ||||||
|         if(!file_worker_read_until(file_worker, temp_str, '\n')) { |         if(!file_worker_read_until(file_worker, temp_str, '\n')) { | ||||||
| @ -345,6 +354,40 @@ bool subghz_load_protocol_from_file(SubGhz* subghz) { | |||||||
|     return res; |     return res; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | bool subghz_delete_file(SubGhz* subghz) { | ||||||
|  |     furi_assert(subghz); | ||||||
|  | 
 | ||||||
|  |     bool result = true; | ||||||
|  |     FileWorker* file_worker = file_worker_alloc(false); | ||||||
|  |     string_t file_path; | ||||||
|  | 
 | ||||||
|  |     do { | ||||||
|  |         // Get key file path
 | ||||||
|  |         string_init_printf( | ||||||
|  |             file_path, | ||||||
|  |             "%s/%s%s", | ||||||
|  |             SUBGHZ_APP_PATH_FOLDER, | ||||||
|  |             subghz->file_name_tmp, | ||||||
|  |             SUBGHZ_APP_EXTENSION); | ||||||
|  |         // Delete original file
 | ||||||
|  |         if(!file_worker_remove(file_worker, string_get_cstr(file_path))) { | ||||||
|  |             result = false; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } while(0); | ||||||
|  | 
 | ||||||
|  |     string_clear(file_path); | ||||||
|  |     file_worker_close(file_worker); | ||||||
|  |     file_worker_free(file_worker); | ||||||
|  |     return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_file_name_clear(SubGhz* subghz) { | ||||||
|  |     furi_assert(subghz); | ||||||
|  |     memset(subghz->file_name, 0, sizeof(subghz->file_name)); | ||||||
|  |     memset(subghz->file_name_tmp, 0, sizeof(subghz->file_name_tmp)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| uint32_t subghz_random_serial(void) { | uint32_t subghz_random_serial(void) { | ||||||
|     static bool rand_generator_inited = false; |     static bool rand_generator_inited = false; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -29,7 +29,7 @@ | |||||||
| 
 | 
 | ||||||
| #include <gui/modules/variable-item-list.h> | #include <gui/modules/variable-item-list.h> | ||||||
| 
 | 
 | ||||||
| #define SUBGHZ_TEXT_STORE_SIZE 128 | #define SUBGHZ_TEXT_STORE_SIZE 40 | ||||||
| 
 | 
 | ||||||
| #define NOTIFICATION_STARTING_STATE 0u | #define NOTIFICATION_STARTING_STATE 0u | ||||||
| #define NOTIFICATION_IDLE_STATE 1u | #define NOTIFICATION_IDLE_STATE 1u | ||||||
| @ -90,7 +90,8 @@ struct SubGhz { | |||||||
|     Popup* popup; |     Popup* popup; | ||||||
|     TextInput* text_input; |     TextInput* text_input; | ||||||
|     Widget* widget; |     Widget* widget; | ||||||
|     char text_store[SUBGHZ_TEXT_STORE_SIZE + 1]; |     char file_name[SUBGHZ_TEXT_STORE_SIZE + 1]; | ||||||
|  |     char file_name_tmp[SUBGHZ_TEXT_STORE_SIZE + 1]; | ||||||
|     uint8_t state_notifications; |     uint8_t state_notifications; | ||||||
| 
 | 
 | ||||||
|     SubghzReceiver* subghz_receiver; |     SubghzReceiver* subghz_receiver; | ||||||
| @ -121,10 +122,12 @@ void subghz_begin(SubGhz* subghz, FuriHalSubGhzPreset preset); | |||||||
| uint32_t subghz_rx(SubGhz* subghz, uint32_t frequency); | uint32_t subghz_rx(SubGhz* subghz, uint32_t frequency); | ||||||
| void subghz_rx_end(SubGhz* subghz); | void subghz_rx_end(SubGhz* subghz); | ||||||
| void subghz_sleep(SubGhz* subghz); | void subghz_sleep(SubGhz* subghz); | ||||||
| void subghz_tx_start(SubGhz* subghz); | bool subghz_tx_start(SubGhz* subghz); | ||||||
| void subghz_tx_stop(SubGhz* subghz); | void subghz_tx_stop(SubGhz* subghz); | ||||||
| bool subghz_key_load(SubGhz* subghz, const char* file_path); | bool subghz_key_load(SubGhz* subghz, const char* file_path); | ||||||
| bool subghz_save_protocol_to_file(SubGhz* subghz, const char* dev_name); | bool subghz_save_protocol_to_file(SubGhz* subghz, const char* dev_name); | ||||||
| bool subghz_load_protocol_from_file(SubGhz* subghz); | bool subghz_load_protocol_from_file(SubGhz* subghz); | ||||||
|  | bool subghz_delete_file(SubGhz* subghz); | ||||||
|  | void subghz_file_name_clear(SubGhz* subghz); | ||||||
| uint32_t subghz_random_serial(void); | uint32_t subghz_random_serial(void); | ||||||
| void subghz_hopper_update(SubGhz* subghz); | void subghz_hopper_update(SubGhz* subghz); | ||||||
|  | |||||||
| @ -9,6 +9,8 @@ | |||||||
| struct SubghzTestCarrier { | struct SubghzTestCarrier { | ||||||
|     View* view; |     View* view; | ||||||
|     osTimerId timer; |     osTimerId timer; | ||||||
|  |     SubghzTestCarrierCallback callback; | ||||||
|  |     void* context; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
| @ -24,6 +26,16 @@ typedef struct { | |||||||
|     SubghzTestCarrierModelStatus status; |     SubghzTestCarrierModelStatus status; | ||||||
| } SubghzTestCarrierModel; | } SubghzTestCarrierModel; | ||||||
| 
 | 
 | ||||||
|  | void subghz_test_carrier_set_callback( | ||||||
|  |     SubghzTestCarrier* subghz_test_carrier, | ||||||
|  |     SubghzTestCarrierCallback callback, | ||||||
|  |     void* context) { | ||||||
|  |     furi_assert(subghz_test_carrier); | ||||||
|  |     furi_assert(callback); | ||||||
|  |     subghz_test_carrier->callback = callback; | ||||||
|  |     subghz_test_carrier->context = context; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void subghz_test_carrier_draw(Canvas* canvas, SubghzTestCarrierModel* model) { | void subghz_test_carrier_draw(Canvas* canvas, SubghzTestCarrierModel* model) { | ||||||
|     char buffer[64]; |     char buffer[64]; | ||||||
| 
 | 
 | ||||||
| @ -105,7 +117,11 @@ bool subghz_test_carrier_input(InputEvent* event, void* context) { | |||||||
|             } else { |             } else { | ||||||
|                 hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); |                 hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); | ||||||
|                 hal_gpio_write(&gpio_cc1101_g0, true); |                 hal_gpio_write(&gpio_cc1101_g0, true); | ||||||
|                 furi_hal_subghz_tx(); |                 if(!furi_hal_subghz_tx()) { | ||||||
|  |                     hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullNo, GpioSpeedLow); | ||||||
|  |                     subghz_test_carrier->callback( | ||||||
|  |                         SubghzTestCarrierEventOnlyRx, subghz_test_carrier->context); | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             return true; |             return true; | ||||||
|  | |||||||
| @ -2,8 +2,19 @@ | |||||||
| 
 | 
 | ||||||
| #include <gui/view.h> | #include <gui/view.h> | ||||||
| 
 | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SubghzTestCarrierEventOnlyRx, | ||||||
|  | } SubghzTestCarrierEvent; | ||||||
|  | 
 | ||||||
| typedef struct SubghzTestCarrier SubghzTestCarrier; | typedef struct SubghzTestCarrier SubghzTestCarrier; | ||||||
| 
 | 
 | ||||||
|  | typedef void (*SubghzTestCarrierCallback)(SubghzTestCarrierEvent event, void* context); | ||||||
|  | 
 | ||||||
|  | void subghz_test_carrier_set_callback( | ||||||
|  |     SubghzTestCarrier* subghz_test_carrier, | ||||||
|  |     SubghzTestCarrierCallback callback, | ||||||
|  |     void* context); | ||||||
|  | 
 | ||||||
| SubghzTestCarrier* subghz_test_carrier_alloc(); | SubghzTestCarrier* subghz_test_carrier_alloc(); | ||||||
| 
 | 
 | ||||||
| void subghz_test_carrier_free(SubghzTestCarrier* subghz_test_carrier); | void subghz_test_carrier_free(SubghzTestCarrier* subghz_test_carrier); | ||||||
|  | |||||||
| @ -16,12 +16,14 @@ struct SubghzTestPacket { | |||||||
| 
 | 
 | ||||||
|     SubGhzDecoderPrinceton* decoder; |     SubGhzDecoderPrinceton* decoder; | ||||||
|     SubGhzEncoderPrinceton* encoder; |     SubGhzEncoderPrinceton* encoder; | ||||||
| 
 |  | ||||||
|     volatile size_t packet_rx; |     volatile size_t packet_rx; | ||||||
|  |     SubghzTestPacketCallback callback; | ||||||
|  |     void* context; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     SubghzTestPacketModelStatusRx, |     SubghzTestPacketModelStatusRx, | ||||||
|  |     SubghzTestPacketModelStatusOnlyRx, | ||||||
|     SubghzTestPacketModelStatusTx, |     SubghzTestPacketModelStatusTx, | ||||||
| } SubghzTestPacketModelStatus; | } SubghzTestPacketModelStatus; | ||||||
| 
 | 
 | ||||||
| @ -36,6 +38,16 @@ typedef struct { | |||||||
| 
 | 
 | ||||||
| volatile bool subghz_test_packet_overrun = false; | volatile bool subghz_test_packet_overrun = false; | ||||||
| 
 | 
 | ||||||
|  | void subghz_test_packet_set_callback( | ||||||
|  |     SubghzTestPacket* subghz_test_packet, | ||||||
|  |     SubghzTestPacketCallback callback, | ||||||
|  |     void* context) { | ||||||
|  |     furi_assert(subghz_test_packet); | ||||||
|  |     furi_assert(callback); | ||||||
|  |     subghz_test_packet->callback = callback; | ||||||
|  |     subghz_test_packet->context = context; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static void subghz_test_packet_rx_callback(bool level, uint32_t duration, void* context) { | static void subghz_test_packet_rx_callback(bool level, uint32_t duration, void* context) { | ||||||
|     furi_assert(context); |     furi_assert(context); | ||||||
|     SubghzTestPacket* instance = context; |     SubghzTestPacket* instance = context; | ||||||
| @ -57,7 +69,7 @@ static void subghz_test_packet_rssi_timer_callback(void* context) { | |||||||
|             if(model->status == SubghzTestPacketModelStatusRx) { |             if(model->status == SubghzTestPacketModelStatusRx) { | ||||||
|                 model->rssi = furi_hal_subghz_get_rssi(); |                 model->rssi = furi_hal_subghz_get_rssi(); | ||||||
|                 model->packets = instance->packet_rx; |                 model->packets = instance->packet_rx; | ||||||
|             } else { |             } else if(model->status == SubghzTestPacketModelStatusTx) { | ||||||
|                 model->packets = SUBGHZ_TEST_PACKET_COUNT - |                 model->packets = SUBGHZ_TEST_PACKET_COUNT - | ||||||
|                                  subghz_encoder_princeton_get_repeat_left(instance->encoder); |                                  subghz_encoder_princeton_get_repeat_left(instance->encoder); | ||||||
|             } |             } | ||||||
| @ -124,7 +136,7 @@ static bool subghz_test_packet_input(InputEvent* event, void* context) { | |||||||
|         instance->view, (SubghzTestPacketModel * model) { |         instance->view, (SubghzTestPacketModel * model) { | ||||||
|             if(model->status == SubghzTestPacketModelStatusRx) { |             if(model->status == SubghzTestPacketModelStatusRx) { | ||||||
|                 furi_hal_subghz_stop_async_rx(); |                 furi_hal_subghz_stop_async_rx(); | ||||||
|             } else { |             } else if(model->status == SubghzTestPacketModelStatusTx) { | ||||||
|                 furi_hal_subghz_stop_async_tx(); |                 furi_hal_subghz_stop_async_tx(); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
| @ -137,10 +149,10 @@ static bool subghz_test_packet_input(InputEvent* event, void* context) { | |||||||
|             } else if(event->key == InputKeyUp) { |             } else if(event->key == InputKeyUp) { | ||||||
|                 if(model->path < FuriHalSubGhzPath868) model->path++; |                 if(model->path < FuriHalSubGhzPath868) model->path++; | ||||||
|             } else if(event->key == InputKeyOk) { |             } else if(event->key == InputKeyOk) { | ||||||
|                 if(model->status == SubghzTestPacketModelStatusTx) { |                 if(model->status == SubghzTestPacketModelStatusRx) { | ||||||
|                     model->status = SubghzTestPacketModelStatusRx; |  | ||||||
|                 } else { |  | ||||||
|                     model->status = SubghzTestPacketModelStatusTx; |                     model->status = SubghzTestPacketModelStatusTx; | ||||||
|  |                 } else { | ||||||
|  |                     model->status = SubghzTestPacketModelStatusRx; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
| @ -151,8 +163,13 @@ static bool subghz_test_packet_input(InputEvent* event, void* context) { | |||||||
|             if(model->status == SubghzTestPacketModelStatusRx) { |             if(model->status == SubghzTestPacketModelStatusRx) { | ||||||
|                 furi_hal_subghz_start_async_rx(subghz_test_packet_rx_callback, instance); |                 furi_hal_subghz_start_async_rx(subghz_test_packet_rx_callback, instance); | ||||||
|             } else { |             } else { | ||||||
|                 subghz_encoder_princeton_set(instance->encoder, 0x00AABBCC, 1000); |                 subghz_encoder_princeton_set( | ||||||
|                 furi_hal_subghz_start_async_tx(subghz_encoder_princeton_yield, instance->encoder); |                     instance->encoder, 0x00AABBCC, SUBGHZ_TEST_PACKET_COUNT); | ||||||
|  |                 if(!furi_hal_subghz_start_async_tx( | ||||||
|  |                        subghz_encoder_princeton_yield, instance->encoder)) { | ||||||
|  |                     model->status = SubghzTestPacketModelStatusOnlyRx; | ||||||
|  |                     instance->callback(SubghzTestPacketEventOnlyRx, instance->context); | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             return true; |             return true; | ||||||
| @ -195,7 +212,7 @@ void subghz_test_packet_exit(void* context) { | |||||||
|         instance->view, (SubghzTestPacketModel * model) { |         instance->view, (SubghzTestPacketModel * model) { | ||||||
|             if(model->status == SubghzTestPacketModelStatusRx) { |             if(model->status == SubghzTestPacketModelStatusRx) { | ||||||
|                 furi_hal_subghz_stop_async_rx(); |                 furi_hal_subghz_stop_async_rx(); | ||||||
|             } else { |             } else if(model->status == SubghzTestPacketModelStatusTx) { | ||||||
|                 furi_hal_subghz_stop_async_tx(); |                 furi_hal_subghz_stop_async_tx(); | ||||||
|             } |             } | ||||||
|             return true; |             return true; | ||||||
|  | |||||||
| @ -2,8 +2,19 @@ | |||||||
| 
 | 
 | ||||||
| #include <gui/view.h> | #include <gui/view.h> | ||||||
| 
 | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SubghzTestPacketEventOnlyRx, | ||||||
|  | } SubghzTestPacketEvent; | ||||||
|  | 
 | ||||||
| typedef struct SubghzTestPacket SubghzTestPacket; | typedef struct SubghzTestPacket SubghzTestPacket; | ||||||
| 
 | 
 | ||||||
|  | typedef void (*SubghzTestPacketCallback)(SubghzTestPacketEvent event, void* context); | ||||||
|  | 
 | ||||||
|  | void subghz_test_packet_set_callback( | ||||||
|  |     SubghzTestPacket* subghz_test_packet, | ||||||
|  |     SubghzTestPacketCallback callback, | ||||||
|  |     void* context); | ||||||
|  | 
 | ||||||
| SubghzTestPacket* subghz_test_packet_alloc(); | SubghzTestPacket* subghz_test_packet_alloc(); | ||||||
| 
 | 
 | ||||||
| void subghz_test_packet_free(SubghzTestPacket* subghz_test_packet); | void subghz_test_packet_free(SubghzTestPacket* subghz_test_packet); | ||||||
|  | |||||||
| @ -8,6 +8,11 @@ | |||||||
| #include <notification/notification-messages.h> | #include <notification/notification-messages.h> | ||||||
| #include <lib/subghz/protocols/subghz_protocol_princeton.h> | #include <lib/subghz/protocols/subghz_protocol_princeton.h> | ||||||
| 
 | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SubghzTestStaticStatusIDLE, | ||||||
|  |     SubghzTestStaticStatusTX, | ||||||
|  | } SubghzTestStaticStatus; | ||||||
|  | 
 | ||||||
| static const uint32_t subghz_test_static_keys[] = { | static const uint32_t subghz_test_static_keys[] = { | ||||||
|     0x0074BADE, |     0x0074BADE, | ||||||
|     0x0074BADD, |     0x0074BADD, | ||||||
| @ -17,20 +22,28 @@ static const uint32_t subghz_test_static_keys[] = { | |||||||
| 
 | 
 | ||||||
| struct SubghzTestStatic { | struct SubghzTestStatic { | ||||||
|     View* view; |     View* view; | ||||||
|  |     SubghzTestStaticStatus satus_tx; | ||||||
|     SubGhzEncoderPrinceton* encoder; |     SubGhzEncoderPrinceton* encoder; | ||||||
|  |     SubghzTestStaticCallback callback; | ||||||
|  |     void* context; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| typedef enum { |  | ||||||
|     SubghzTestStaticStatusRx, |  | ||||||
|     SubghzTestStaticStatusTx, |  | ||||||
| } SubghzTestStaticStatus; |  | ||||||
| 
 |  | ||||||
| typedef struct { | typedef struct { | ||||||
|     uint8_t frequency; |     uint8_t frequency; | ||||||
|     uint32_t real_frequency; |     uint32_t real_frequency; | ||||||
|     uint8_t button; |     uint8_t button; | ||||||
| } SubghzTestStaticModel; | } SubghzTestStaticModel; | ||||||
| 
 | 
 | ||||||
|  | void subghz_test_static_set_callback( | ||||||
|  |     SubghzTestStatic* subghz_test_static, | ||||||
|  |     SubghzTestStaticCallback callback, | ||||||
|  |     void* context) { | ||||||
|  |     furi_assert(subghz_test_static); | ||||||
|  |     furi_assert(callback); | ||||||
|  |     subghz_test_static->callback = callback; | ||||||
|  |     subghz_test_static->context = context; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void subghz_test_static_draw(Canvas* canvas, SubghzTestStaticModel* model) { | void subghz_test_static_draw(Canvas* canvas, SubghzTestStaticModel* model) { | ||||||
|     char buffer[64]; |     char buffer[64]; | ||||||
| 
 | 
 | ||||||
| @ -79,23 +92,31 @@ bool subghz_test_static_input(InputEvent* event, void* context) { | |||||||
|             if(event->key == InputKeyOk) { |             if(event->key == InputKeyOk) { | ||||||
|                 NotificationApp* notification = furi_record_open("notification"); |                 NotificationApp* notification = furi_record_open("notification"); | ||||||
|                 if(event->type == InputTypePress) { |                 if(event->type == InputTypePress) { | ||||||
|  |                     furi_hal_subghz_idle(); | ||||||
|  |                     furi_hal_subghz_set_frequency_and_path(subghz_frequencies[model->frequency]); | ||||||
|  |                     if(!furi_hal_subghz_tx()) { | ||||||
|  |                         instance->callback(SubghzTestStaticEventOnlyRx, instance->context); | ||||||
|  |                     } else { | ||||||
|                         notification_message_block(notification, &sequence_set_red_255); |                         notification_message_block(notification, &sequence_set_red_255); | ||||||
| 
 | 
 | ||||||
|                         FURI_LOG_I("SubghzTestStatic", "TX Start"); |                         FURI_LOG_I("SubghzTestStatic", "TX Start"); | ||||||
|                     furi_hal_subghz_idle(); |  | ||||||
|                     furi_hal_subghz_set_frequency_and_path(subghz_frequencies[model->frequency]); |  | ||||||
| 
 | 
 | ||||||
|                         subghz_encoder_princeton_set( |                         subghz_encoder_princeton_set( | ||||||
|                             instance->encoder, subghz_test_static_keys[model->button], 10000); |                             instance->encoder, subghz_test_static_keys[model->button], 10000); | ||||||
| 
 | 
 | ||||||
|                         furi_hal_subghz_start_async_tx( |                         furi_hal_subghz_start_async_tx( | ||||||
|                             subghz_encoder_princeton_yield, instance->encoder); |                             subghz_encoder_princeton_yield, instance->encoder); | ||||||
|  |                         instance->satus_tx = SubghzTestStaticStatusTX; | ||||||
|  |                     } | ||||||
|                 } else if(event->type == InputTypeRelease) { |                 } else if(event->type == InputTypeRelease) { | ||||||
|  |                     if(instance->satus_tx == SubghzTestStaticStatusTX) { | ||||||
|                         FURI_LOG_I("SubghzTestStatic", "TX Stop"); |                         FURI_LOG_I("SubghzTestStatic", "TX Stop"); | ||||||
|  |                         subghz_encoder_princeton_print_log(instance->encoder); | ||||||
|                         furi_hal_subghz_stop_async_tx(); |                         furi_hal_subghz_stop_async_tx(); | ||||||
| 
 |  | ||||||
|                         notification_message(notification, &sequence_reset_red); |                         notification_message(notification, &sequence_reset_red); | ||||||
|                     } |                     } | ||||||
|  |                     instance->satus_tx = SubghzTestStaticStatusIDLE; | ||||||
|  |                 } | ||||||
|                 furi_record_close("notification"); |                 furi_record_close("notification"); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
| @ -114,12 +135,14 @@ void subghz_test_static_enter(void* context) { | |||||||
| 
 | 
 | ||||||
|     hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); |     hal_gpio_init(&gpio_cc1101_g0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); | ||||||
|     hal_gpio_write(&gpio_cc1101_g0, false); |     hal_gpio_write(&gpio_cc1101_g0, false); | ||||||
|  |     instance->satus_tx = SubghzTestStaticStatusIDLE; | ||||||
| 
 | 
 | ||||||
|     with_view_model( |     with_view_model( | ||||||
|         instance->view, (SubghzTestStaticModel * model) { |         instance->view, (SubghzTestStaticModel * model) { | ||||||
|             model->frequency = subghz_frequencies_433_92; |             model->frequency = subghz_frequencies_433_92; | ||||||
|             model->real_frequency = subghz_frequencies[model->frequency]; |             model->real_frequency = subghz_frequencies[model->frequency]; | ||||||
|             model->button = 0; |             model->button = 0; | ||||||
|  | 
 | ||||||
|             return true; |             return true; | ||||||
|         }); |         }); | ||||||
| } | } | ||||||
|  | |||||||
| @ -2,8 +2,19 @@ | |||||||
| 
 | 
 | ||||||
| #include <gui/view.h> | #include <gui/view.h> | ||||||
| 
 | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SubghzTestStaticEventOnlyRx, | ||||||
|  | } SubghzTestStaticEvent; | ||||||
|  | 
 | ||||||
| typedef struct SubghzTestStatic SubghzTestStatic; | typedef struct SubghzTestStatic SubghzTestStatic; | ||||||
| 
 | 
 | ||||||
|  | typedef void (*SubghzTestStaticCallback)(SubghzTestStaticEvent event, void* context); | ||||||
|  | 
 | ||||||
|  | void subghz_test_static_set_callback( | ||||||
|  |     SubghzTestStatic* subghz_test_static, | ||||||
|  |     SubghzTestStaticCallback callback, | ||||||
|  |     void* context); | ||||||
|  | 
 | ||||||
| SubghzTestStatic* subghz_test_static_alloc(); | SubghzTestStatic* subghz_test_static_alloc(); | ||||||
| 
 | 
 | ||||||
| void subghz_test_static_free(SubghzTestStatic* subghz_static); | void subghz_test_static_free(SubghzTestStatic* subghz_static); | ||||||
|  | |||||||
| @ -1,4 +1,5 @@ | |||||||
| #include "furi-hal-subghz.h" | #include "furi-hal-subghz.h" | ||||||
|  | #include "furi-hal-version.h" | ||||||
| 
 | 
 | ||||||
| #include <furi-hal-gpio.h> | #include <furi-hal-gpio.h> | ||||||
| #include <furi-hal-spi.h> | #include <furi-hal-spi.h> | ||||||
| @ -10,6 +11,7 @@ | |||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| 
 | 
 | ||||||
| static volatile SubGhzState furi_hal_subghz_state = SubGhzStateInit; | static volatile SubGhzState furi_hal_subghz_state = SubGhzStateInit; | ||||||
|  | static volatile SubGhzRegulation furi_hal_subghz_regulation = SubGhzRegulationTxRx; | ||||||
| 
 | 
 | ||||||
| static const uint8_t furi_hal_subghz_preset_ook_270khz_async_regs[][2] = { | static const uint8_t furi_hal_subghz_preset_ook_270khz_async_regs[][2] = { | ||||||
|     // https://e2e.ti.com/support/wireless-connectivity/sub-1-ghz-group/sub-1-ghz/f/sub-1-ghz-forum/382066/cc1101---don-t-know-the-correct-registers-configuration
 |     // https://e2e.ti.com/support/wireless-connectivity/sub-1-ghz-group/sub-1-ghz/f/sub-1-ghz-forum/382066/cc1101---don-t-know-the-correct-registers-configuration
 | ||||||
| @ -41,11 +43,11 @@ static const uint8_t furi_hal_subghz_preset_ook_270khz_async_regs[][2] = { | |||||||
|      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 |      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 | ||||||
| 
 | 
 | ||||||
|     /* Automatic Gain Control */ |     /* Automatic Gain Control */ | ||||||
|     {CC1101_AGCTRL0, |     {CC1101_AGCCTRL0, | ||||||
|      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 |      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 | ||||||
|     {CC1101_AGCTRL1, |     {CC1101_AGCCTRL1, | ||||||
|      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 |      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|     {CC1101_AGCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 |     {CC1101_AGCCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 | ||||||
| 
 | 
 | ||||||
|     /* Wake on radio and timeouts control */ |     /* Wake on radio and timeouts control */ | ||||||
|     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 |     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 | ||||||
| @ -99,11 +101,15 @@ static const uint8_t furi_hal_subghz_preset_ook_650khz_async_regs[][2] = { | |||||||
|      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 |      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 | ||||||
| 
 | 
 | ||||||
|     /* Automatic Gain Control */ |     /* Automatic Gain Control */ | ||||||
|     {CC1101_AGCTRL0, |     // {CC1101_AGCTRL0,0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 | ||||||
|      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 |     // {CC1101_AGCTRL1,0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|     {CC1101_AGCTRL1, |     // {CC1101_AGCCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 | ||||||
|      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 |     //MAGN_TARGET for RX filter BW =< 100 kHz is 0x3. For higher RX filter BW's MAGN_TARGET is 0x7.
 | ||||||
|     {CC1101_AGCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 |     {CC1101_AGCCTRL0, | ||||||
|  |      0x91}, // 10 - Medium hysteresis, medium asymmetric dead zone, medium gain ; 01 - 16 samples agc; 00 - Normal AGC, 01 - 8dB boundary
 | ||||||
|  |     {CC1101_AGCCTRL1, | ||||||
|  |      0x0}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|  |     {CC1101_AGCCTRL2, 0x07}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 111 - MAIN_TARGET 42 dB
 | ||||||
| 
 | 
 | ||||||
|     /* Wake on radio and timeouts control */ |     /* Wake on radio and timeouts control */ | ||||||
|     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 |     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 | ||||||
| @ -131,23 +137,21 @@ static const uint8_t furi_hal_subghz_preset_2fsk_async_regs[][2] = { | |||||||
|     /* GPIO GD0 */ |     /* GPIO GD0 */ | ||||||
|     {CC1101_IOCFG0, 0x0D}, // GD0 as async serial data output/input
 |     {CC1101_IOCFG0, 0x0D}, // GD0 as async serial data output/input
 | ||||||
| 
 | 
 | ||||||
|     /* FIFO and internals */ |  | ||||||
|     {CC1101_FIFOTHR, 0x47}, // The only important bit is ADC_RETENTION
 |  | ||||||
| 
 |  | ||||||
|     /* Packet engine */ |  | ||||||
|     {CC1101_PKTCTRL0, 0x32}, // Async, continious, no whitening
 |  | ||||||
| 
 |  | ||||||
|     /* Frequency Synthesizer Control */ |     /* Frequency Synthesizer Control */ | ||||||
|     {CC1101_FSCTRL1, 0x06}, // IF = (26*10^6) / (2^10) * 0x06 = 152343.75Hz
 |     {CC1101_FSCTRL1, 0x06}, // IF = (26*10^6) / (2^10) * 0x06 = 152343.75Hz
 | ||||||
| 
 | 
 | ||||||
|     // Modem Configuration
 |     /* Packet engine */ | ||||||
|     {CC1101_MDMCFG0, 0x00}, |     {CC1101_PKTCTRL0, 0x32}, // Async, continious, no whitening
 | ||||||
|     {CC1101_MDMCFG1, 0x02}, |     {CC1101_PKTCTRL1, 0x04}, | ||||||
|     {CC1101_MDMCFG2, 0x04}, // Format 2-FSK/FM, No preamble/sync, Disable (current optimized)
 |  | ||||||
|     {CC1101_MDMCFG3, 0x8B}, // Data rate is 19.5885 kBaud
 |  | ||||||
|     {CC1101_MDMCFG4, 0x69}, // Rx BW filter is 270.833333 kHz
 |  | ||||||
| 
 | 
 | ||||||
|     {CC1101_DEVIATN, 0x47}, //Deviation 47.607422 khz
 |     // // Modem Configuration
 | ||||||
|  |     {CC1101_MDMCFG0, 0x00}, | ||||||
|  |     {CC1101_MDMCFG1, 0x2}, | ||||||
|  |     {CC1101_MDMCFG2, 0x4}, // Format 2-FSK/FM, No preamble/sync, Disable (current optimized)
 | ||||||
|  |     {CC1101_MDMCFG3, 0x83}, // Data rate is 4.79794 kBaud
 | ||||||
|  |     {CC1101_MDMCFG4, 0x67}, //Rx BW filter is 270.833333 kHz
 | ||||||
|  |     //{ CC1101_DEVIATN, 0x14 }, //Deviation 4.760742 kHz
 | ||||||
|  |     {CC1101_DEVIATN, 0x04}, //Deviation 2.380371 kHz
 | ||||||
| 
 | 
 | ||||||
|     /* Main Radio Control State Machine */ |     /* Main Radio Control State Machine */ | ||||||
|     {CC1101_MCSM0, 0x18}, // Autocalibrate on idle-to-rx/tx, PO_TIMEOUT is 64 cycles(149-155us)
 |     {CC1101_MCSM0, 0x18}, // Autocalibrate on idle-to-rx/tx, PO_TIMEOUT is 64 cycles(149-155us)
 | ||||||
| @ -157,18 +161,18 @@ static const uint8_t furi_hal_subghz_preset_2fsk_async_regs[][2] = { | |||||||
|      0x16}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 |      0x16}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 | ||||||
| 
 | 
 | ||||||
|     /* Automatic Gain Control */ |     /* Automatic Gain Control */ | ||||||
|     {CC1101_AGCTRL0, |     {CC1101_AGCCTRL0, | ||||||
|      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 |      0x91}, //10 - Medium hysteresis, medium asymmetric dead zone, medium gain ; 01 - 16 samples agc; 00 - Normal AGC, 01 - 8dB boundary
 | ||||||
|     {CC1101_AGCTRL1, |     {CC1101_AGCCTRL1, | ||||||
|      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 |      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|     {CC1101_AGCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 |     {CC1101_AGCCTRL2, 0x07}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 111 - MAIN_TARGET 42 dB
 | ||||||
| 
 | 
 | ||||||
|     /* Wake on radio and timeouts control */ |     /* Wake on radio and timeouts control */ | ||||||
|     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 |     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 | ||||||
| 
 | 
 | ||||||
|     /* Frontend configuration */ |     /* Frontend configuration */ | ||||||
|     {CC1101_FREND0, 0x10}, // Adjusts current TX LO buffer
 |     {CC1101_FREND0, 0x10}, // Adjusts current TX LO buffer
 | ||||||
|     {CC1101_FREND1, 0xB6}, //
 |     {CC1101_FREND1, 0x56}, | ||||||
| 
 | 
 | ||||||
|     /* Frequency Synthesizer Calibration, valid for 433.92 */ |     /* Frequency Synthesizer Calibration, valid for 433.92 */ | ||||||
|     {CC1101_FSCAL3, 0xE9}, |     {CC1101_FSCAL3, 0xE9}, | ||||||
| @ -281,7 +285,7 @@ void furi_hal_subghz_load_preset(FuriHalSubGhzPreset preset) { | |||||||
|     } else if(preset == FuriHalSubGhzPreset2FSKAsync) { |     } else if(preset == FuriHalSubGhzPreset2FSKAsync) { | ||||||
|         furi_hal_subghz_load_registers(furi_hal_subghz_preset_2fsk_async_regs); |         furi_hal_subghz_load_registers(furi_hal_subghz_preset_2fsk_async_regs); | ||||||
|         furi_hal_subghz_load_patable(furi_hal_subghz_preset_2fsk_async_patable); |         furi_hal_subghz_load_patable(furi_hal_subghz_preset_2fsk_async_patable); | ||||||
|     }else { |     } else { | ||||||
|         furi_crash(NULL); |         furi_crash(NULL); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -358,10 +362,12 @@ void furi_hal_subghz_rx() { | |||||||
|     furi_hal_spi_device_return(device); |     furi_hal_spi_device_return(device); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void furi_hal_subghz_tx() { | bool furi_hal_subghz_tx() { | ||||||
|  |     if(furi_hal_subghz_regulation != SubGhzRegulationTxRx) return false; | ||||||
|     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); |     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); | ||||||
|     cc1101_switch_to_tx(device); |     cc1101_switch_to_tx(device); | ||||||
|     furi_hal_spi_device_return(device); |     furi_hal_spi_device_return(device); | ||||||
|  |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| float furi_hal_subghz_get_rssi() { | float furi_hal_subghz_get_rssi() { | ||||||
| @ -385,6 +391,7 @@ bool furi_hal_subghz_is_frequency_valid(uint32_t value) { | |||||||
|        !(value >= 778999847 && value <= 928000000)) { |        !(value >= 778999847 && value <= 928000000)) { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -405,6 +412,46 @@ uint32_t furi_hal_subghz_set_frequency_and_path(uint32_t value) { | |||||||
| uint32_t furi_hal_subghz_set_frequency(uint32_t value) { | uint32_t furi_hal_subghz_set_frequency(uint32_t value) { | ||||||
|     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); |     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); | ||||||
| 
 | 
 | ||||||
|  |     //checking regional settings
 | ||||||
|  |     bool txrx = false; | ||||||
|  |     switch(furi_hal_version_get_hw_region()) { | ||||||
|  |     case FuriHalVersionRegionEuRu: | ||||||
|  |         //433,05..434,79; 868,15..868,55
 | ||||||
|  |         if(!(value >= 433050000 && value <= 434790000) && | ||||||
|  |            !(value >= 868150000 && value <= 8680550000)) { | ||||||
|  |         } else { | ||||||
|  |             txrx = true; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case FuriHalVersionRegionUsCaAu: | ||||||
|  |         //304,10..315,25; 433,05..434,79; 915,00..928,00
 | ||||||
|  |         if(!(value >= 304100000 && value <= 315250000) && | ||||||
|  |            !(value >= 433050000 && value <= 434790000) && | ||||||
|  |            !(value >= 915000000 && value <= 928000000)) { | ||||||
|  |         } else { | ||||||
|  |             txrx = true; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case FuriHalVersionRegionJp: | ||||||
|  |         //312,00..315,25; 920,50..923,50
 | ||||||
|  |         if(!(value >= 312000000 && value <= 315250000) && | ||||||
|  |            !(value >= 920500000 && value <= 923500000)) { | ||||||
|  |         } else { | ||||||
|  |             txrx = true; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  | 
 | ||||||
|  |     default: | ||||||
|  |         txrx = true; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if(txrx) { | ||||||
|  |         furi_hal_subghz_regulation = SubGhzRegulationTxRx; | ||||||
|  |     } else { | ||||||
|  |         furi_hal_subghz_regulation = SubGhzRegulationOnlyRx; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     uint32_t real_frequency = cc1101_set_frequency(device, value); |     uint32_t real_frequency = cc1101_set_frequency(device, value); | ||||||
|     cc1101_calibrate(device); |     cc1101_calibrate(device); | ||||||
| 
 | 
 | ||||||
| @ -482,7 +529,7 @@ void furi_hal_subghz_start_async_rx(FuriHalSubGhzCaptureCallback callback, void* | |||||||
|     TIM_InitStruct.Prescaler = 64 - 1; |     TIM_InitStruct.Prescaler = 64 - 1; | ||||||
|     TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP; |     TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP; | ||||||
|     TIM_InitStruct.Autoreload = 0x7FFFFFFE; |     TIM_InitStruct.Autoreload = 0x7FFFFFFE; | ||||||
|     TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV1; |     TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV4; | ||||||
|     LL_TIM_Init(TIM2, &TIM_InitStruct); |     LL_TIM_Init(TIM2, &TIM_InitStruct); | ||||||
| 
 | 
 | ||||||
|     // Timer: advanced
 |     // Timer: advanced
 | ||||||
| @ -505,7 +552,7 @@ void furi_hal_subghz_start_async_rx(FuriHalSubGhzCaptureCallback callback, void* | |||||||
|     LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_DIRECTTI); |     LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_DIRECTTI); | ||||||
|     LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1); |     LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1); | ||||||
|     LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_RISING); |     LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_RISING); | ||||||
|     LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV1); |     LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV32_N8); | ||||||
| 
 | 
 | ||||||
|     // ISR setup
 |     // ISR setup
 | ||||||
|     furi_hal_interrupt_set_timer_isr(TIM2, furi_hal_subghz_capture_ISR); |     furi_hal_interrupt_set_timer_isr(TIM2, furi_hal_subghz_capture_ISR); | ||||||
| @ -610,6 +657,8 @@ static void furi_hal_subghz_async_tx_timer_isr() { | |||||||
|         if(LL_TIM_GetAutoReload(TIM2) == 0) { |         if(LL_TIM_GetAutoReload(TIM2) == 0) { | ||||||
|             if(furi_hal_subghz_state == SubGhzStateAsyncTx) { |             if(furi_hal_subghz_state == SubGhzStateAsyncTx) { | ||||||
|                 furi_hal_subghz_state = SubGhzStateAsyncTxLast; |                 furi_hal_subghz_state = SubGhzStateAsyncTxLast; | ||||||
|  |                 //forcibly pulls the pin to the ground so that there is no carrier
 | ||||||
|  |                 hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullDown, GpioSpeedLow); | ||||||
|             } else { |             } else { | ||||||
|                 furi_hal_subghz_state = SubGhzStateAsyncTxEnd; |                 furi_hal_subghz_state = SubGhzStateAsyncTxEnd; | ||||||
|                 LL_TIM_DisableCounter(TIM2); |                 LL_TIM_DisableCounter(TIM2); | ||||||
| @ -618,10 +667,13 @@ static void furi_hal_subghz_async_tx_timer_isr() { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context) { | bool furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context) { | ||||||
|     furi_assert(furi_hal_subghz_state == SubGhzStateIdle); |     furi_assert(furi_hal_subghz_state == SubGhzStateIdle); | ||||||
|     furi_assert(callback); |     furi_assert(callback); | ||||||
| 
 | 
 | ||||||
|  |     //If transmission is prohibited by regional settings
 | ||||||
|  |     if(furi_hal_subghz_regulation != SubGhzRegulationTxRx) return false; | ||||||
|  | 
 | ||||||
|     furi_hal_subghz_async_tx.callback = callback; |     furi_hal_subghz_async_tx.callback = callback; | ||||||
|     furi_hal_subghz_async_tx.callback_context = context; |     furi_hal_subghz_async_tx.callback_context = context; | ||||||
| 
 | 
 | ||||||
| @ -696,6 +748,7 @@ void furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* | |||||||
| 
 | 
 | ||||||
|     LL_TIM_SetCounter(TIM2, 0); |     LL_TIM_SetCounter(TIM2, 0); | ||||||
|     LL_TIM_EnableCounter(TIM2); |     LL_TIM_EnableCounter(TIM2); | ||||||
|  |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool furi_hal_subghz_is_async_tx_complete() { | bool furi_hal_subghz_is_async_tx_complete() { | ||||||
|  | |||||||
| @ -1,4 +1,5 @@ | |||||||
| #include "furi-hal-subghz.h" | #include "furi-hal-subghz.h" | ||||||
|  | #include "furi-hal-version.h" | ||||||
| 
 | 
 | ||||||
| #include <furi-hal-gpio.h> | #include <furi-hal-gpio.h> | ||||||
| #include <furi-hal-spi.h> | #include <furi-hal-spi.h> | ||||||
| @ -10,6 +11,7 @@ | |||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| 
 | 
 | ||||||
| static volatile SubGhzState furi_hal_subghz_state = SubGhzStateInit; | static volatile SubGhzState furi_hal_subghz_state = SubGhzStateInit; | ||||||
|  | static volatile SubGhzRegulation furi_hal_subghz_regulation = SubGhzRegulationTxRx; | ||||||
| 
 | 
 | ||||||
| static const uint8_t furi_hal_subghz_preset_ook_270khz_async_regs[][2] = { | static const uint8_t furi_hal_subghz_preset_ook_270khz_async_regs[][2] = { | ||||||
|     // https://e2e.ti.com/support/wireless-connectivity/sub-1-ghz-group/sub-1-ghz/f/sub-1-ghz-forum/382066/cc1101---don-t-know-the-correct-registers-configuration
 |     // https://e2e.ti.com/support/wireless-connectivity/sub-1-ghz-group/sub-1-ghz/f/sub-1-ghz-forum/382066/cc1101---don-t-know-the-correct-registers-configuration
 | ||||||
| @ -41,11 +43,11 @@ static const uint8_t furi_hal_subghz_preset_ook_270khz_async_regs[][2] = { | |||||||
|      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 |      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 | ||||||
| 
 | 
 | ||||||
|     /* Automatic Gain Control */ |     /* Automatic Gain Control */ | ||||||
|     {CC1101_AGCTRL0, |     {CC1101_AGCCTRL0, | ||||||
|      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 |      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 | ||||||
|     {CC1101_AGCTRL1, |     {CC1101_AGCCTRL1, | ||||||
|      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 |      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|     {CC1101_AGCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 |     {CC1101_AGCCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 | ||||||
| 
 | 
 | ||||||
|     /* Wake on radio and timeouts control */ |     /* Wake on radio and timeouts control */ | ||||||
|     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 |     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 | ||||||
| @ -99,11 +101,15 @@ static const uint8_t furi_hal_subghz_preset_ook_650khz_async_regs[][2] = { | |||||||
|      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 |      0x18}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 | ||||||
| 
 | 
 | ||||||
|     /* Automatic Gain Control */ |     /* Automatic Gain Control */ | ||||||
|     {CC1101_AGCTRL0, |     // {CC1101_AGCTRL0,0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 | ||||||
|      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 |     // {CC1101_AGCTRL1,0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|     {CC1101_AGCTRL1, |     // {CC1101_AGCCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 | ||||||
|      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 |     //MAGN_TARGET for RX filter BW =< 100 kHz is 0x3. For higher RX filter BW's MAGN_TARGET is 0x7.
 | ||||||
|     {CC1101_AGCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 |     {CC1101_AGCCTRL0, | ||||||
|  |      0x91}, // 10 - Medium hysteresis, medium asymmetric dead zone, medium gain ; 01 - 16 samples agc; 00 - Normal AGC, 01 - 8dB boundary
 | ||||||
|  |     {CC1101_AGCCTRL1, | ||||||
|  |      0x0}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|  |     {CC1101_AGCCTRL2, 0x07}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 111 - MAIN_TARGET 42 dB
 | ||||||
| 
 | 
 | ||||||
|     /* Wake on radio and timeouts control */ |     /* Wake on radio and timeouts control */ | ||||||
|     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 |     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 | ||||||
| @ -131,23 +137,21 @@ static const uint8_t furi_hal_subghz_preset_2fsk_async_regs[][2] = { | |||||||
|     /* GPIO GD0 */ |     /* GPIO GD0 */ | ||||||
|     {CC1101_IOCFG0, 0x0D}, // GD0 as async serial data output/input
 |     {CC1101_IOCFG0, 0x0D}, // GD0 as async serial data output/input
 | ||||||
| 
 | 
 | ||||||
|     /* FIFO and internals */ |  | ||||||
|     {CC1101_FIFOTHR, 0x47}, // The only important bit is ADC_RETENTION
 |  | ||||||
| 
 |  | ||||||
|     /* Packet engine */ |  | ||||||
|     {CC1101_PKTCTRL0, 0x32}, // Async, continious, no whitening
 |  | ||||||
| 
 |  | ||||||
|     /* Frequency Synthesizer Control */ |     /* Frequency Synthesizer Control */ | ||||||
|     {CC1101_FSCTRL1, 0x06}, // IF = (26*10^6) / (2^10) * 0x06 = 152343.75Hz
 |     {CC1101_FSCTRL1, 0x06}, // IF = (26*10^6) / (2^10) * 0x06 = 152343.75Hz
 | ||||||
| 
 | 
 | ||||||
|     // Modem Configuration
 |     /* Packet engine */ | ||||||
|     {CC1101_MDMCFG0, 0x00}, |     {CC1101_PKTCTRL0, 0x32}, // Async, continious, no whitening
 | ||||||
|     {CC1101_MDMCFG1, 0x02}, |     {CC1101_PKTCTRL1, 0x04}, | ||||||
|     {CC1101_MDMCFG2, 0x04}, // Format 2-FSK/FM, No preamble/sync, Disable (current optimized)
 |  | ||||||
|     {CC1101_MDMCFG3, 0x8B}, // Data rate is 19.5885 kBaud
 |  | ||||||
|     {CC1101_MDMCFG4, 0x69}, // Rx BW filter is 270.833333 kHz
 |  | ||||||
| 
 | 
 | ||||||
|     {CC1101_DEVIATN, 0x47}, //Deviation 47.607422 khz
 |     // // Modem Configuration
 | ||||||
|  |     {CC1101_MDMCFG0, 0x00}, | ||||||
|  |     {CC1101_MDMCFG1, 0x2}, | ||||||
|  |     {CC1101_MDMCFG2, 0x4}, // Format 2-FSK/FM, No preamble/sync, Disable (current optimized)
 | ||||||
|  |     {CC1101_MDMCFG3, 0x83}, // Data rate is 4.79794 kBaud
 | ||||||
|  |     {CC1101_MDMCFG4, 0x67}, //Rx BW filter is 270.833333 kHz
 | ||||||
|  |     //{ CC1101_DEVIATN, 0x14 }, //Deviation 4.760742 kHz
 | ||||||
|  |     {CC1101_DEVIATN, 0x04}, //Deviation 2.380371 kHz
 | ||||||
| 
 | 
 | ||||||
|     /* Main Radio Control State Machine */ |     /* Main Radio Control State Machine */ | ||||||
|     {CC1101_MCSM0, 0x18}, // Autocalibrate on idle-to-rx/tx, PO_TIMEOUT is 64 cycles(149-155us)
 |     {CC1101_MCSM0, 0x18}, // Autocalibrate on idle-to-rx/tx, PO_TIMEOUT is 64 cycles(149-155us)
 | ||||||
| @ -157,18 +161,18 @@ static const uint8_t furi_hal_subghz_preset_2fsk_async_regs[][2] = { | |||||||
|      0x16}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 |      0x16}, // no frequency offset compensation, POST_K same as PRE_K, PRE_K is 4K, GATE is off
 | ||||||
| 
 | 
 | ||||||
|     /* Automatic Gain Control */ |     /* Automatic Gain Control */ | ||||||
|     {CC1101_AGCTRL0, |     {CC1101_AGCCTRL0, | ||||||
|      0x40}, // 01 - Low hysteresis, small asymmetric dead zone, medium gain; 00 - 8 samples agc; 00 - Normal AGC, 00 - 4dB boundary
 |      0x91}, //10 - Medium hysteresis, medium asymmetric dead zone, medium gain ; 01 - 16 samples agc; 00 - Normal AGC, 01 - 8dB boundary
 | ||||||
|     {CC1101_AGCTRL1, |     {CC1101_AGCCTRL1, | ||||||
|      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 |      0x00}, // 0; 0 - LNA 2 gain is decreased to minimum before decreasing LNA gain; 00 - Relative carrier sense threshold disabled; 0000 - RSSI to MAIN_TARGET
 | ||||||
|     {CC1101_AGCTRL2, 0x03}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 011 - MAIN_TARGET 24 dB
 |     {CC1101_AGCCTRL2, 0x07}, // 00 - DVGA all; 000 - MAX LNA+LNA2; 111 - MAIN_TARGET 42 dB
 | ||||||
| 
 | 
 | ||||||
|     /* Wake on radio and timeouts control */ |     /* Wake on radio and timeouts control */ | ||||||
|     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 |     {CC1101_WORCTRL, 0xFB}, // WOR_RES is 2^15 periods (0.91 - 0.94 s) 16.5 - 17.2 hours
 | ||||||
| 
 | 
 | ||||||
|     /* Frontend configuration */ |     /* Frontend configuration */ | ||||||
|     {CC1101_FREND0, 0x10}, // Adjusts current TX LO buffer
 |     {CC1101_FREND0, 0x10}, // Adjusts current TX LO buffer
 | ||||||
|     {CC1101_FREND1, 0xB6}, //
 |     {CC1101_FREND1, 0x56}, | ||||||
| 
 | 
 | ||||||
|     /* Frequency Synthesizer Calibration, valid for 433.92 */ |     /* Frequency Synthesizer Calibration, valid for 433.92 */ | ||||||
|     {CC1101_FSCAL3, 0xE9}, |     {CC1101_FSCAL3, 0xE9}, | ||||||
| @ -201,7 +205,9 @@ static const uint8_t furi_hal_subghz_preset_2fsk_async_patable[8] = { | |||||||
|     0x00, |     0x00, | ||||||
|     0x00, |     0x00, | ||||||
|     0x00, |     0x00, | ||||||
|     0x00}; |     0x00 | ||||||
|  | 
 | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| void furi_hal_subghz_init() { | void furi_hal_subghz_init() { | ||||||
|     furi_assert(furi_hal_subghz_state == SubGhzStateInit); |     furi_assert(furi_hal_subghz_state == SubGhzStateInit); | ||||||
| @ -279,7 +285,7 @@ void furi_hal_subghz_load_preset(FuriHalSubGhzPreset preset) { | |||||||
|     } else if(preset == FuriHalSubGhzPreset2FSKAsync) { |     } else if(preset == FuriHalSubGhzPreset2FSKAsync) { | ||||||
|         furi_hal_subghz_load_registers(furi_hal_subghz_preset_2fsk_async_regs); |         furi_hal_subghz_load_registers(furi_hal_subghz_preset_2fsk_async_regs); | ||||||
|         furi_hal_subghz_load_patable(furi_hal_subghz_preset_2fsk_async_patable); |         furi_hal_subghz_load_patable(furi_hal_subghz_preset_2fsk_async_patable); | ||||||
|     }else { |     } else { | ||||||
|         furi_crash(NULL); |         furi_crash(NULL); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -356,10 +362,12 @@ void furi_hal_subghz_rx() { | |||||||
|     furi_hal_spi_device_return(device); |     furi_hal_spi_device_return(device); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void furi_hal_subghz_tx() { | bool furi_hal_subghz_tx() { | ||||||
|  |     if(furi_hal_subghz_regulation != SubGhzRegulationTxRx) return false; | ||||||
|     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); |     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); | ||||||
|     cc1101_switch_to_tx(device); |     cc1101_switch_to_tx(device); | ||||||
|     furi_hal_spi_device_return(device); |     furi_hal_spi_device_return(device); | ||||||
|  |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| float furi_hal_subghz_get_rssi() { | float furi_hal_subghz_get_rssi() { | ||||||
| @ -383,6 +391,7 @@ bool furi_hal_subghz_is_frequency_valid(uint32_t value) { | |||||||
|        !(value >= 778999847 && value <= 928000000)) { |        !(value >= 778999847 && value <= 928000000)) { | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -403,6 +412,46 @@ uint32_t furi_hal_subghz_set_frequency_and_path(uint32_t value) { | |||||||
| uint32_t furi_hal_subghz_set_frequency(uint32_t value) { | uint32_t furi_hal_subghz_set_frequency(uint32_t value) { | ||||||
|     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); |     const FuriHalSpiDevice* device = furi_hal_spi_device_get(FuriHalSpiDeviceIdSubGhz); | ||||||
| 
 | 
 | ||||||
|  |     //checking regional settings
 | ||||||
|  |     bool txrx = false; | ||||||
|  |     switch(furi_hal_version_get_hw_region()) { | ||||||
|  |     case FuriHalVersionRegionEuRu: | ||||||
|  |         //433,05..434,79; 868,15..868,55
 | ||||||
|  |         if(!(value >= 433050000 && value <= 434790000) && | ||||||
|  |            !(value >= 868150000 && value <= 8680550000)) { | ||||||
|  |         } else { | ||||||
|  |             txrx = true; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case FuriHalVersionRegionUsCaAu: | ||||||
|  |         //304,10..315,25; 433,05..434,79; 915,00..928,00
 | ||||||
|  |         if(!(value >= 304100000 && value <= 315250000) && | ||||||
|  |            !(value >= 433050000 && value <= 434790000) && | ||||||
|  |            !(value >= 915000000 && value <= 928000000)) { | ||||||
|  |         } else { | ||||||
|  |             txrx = true; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case FuriHalVersionRegionJp: | ||||||
|  |         //312,00..315,25; 920,50..923,50
 | ||||||
|  |         if(!(value >= 312000000 && value <= 315250000) && | ||||||
|  |            !(value >= 920500000 && value <= 923500000)) { | ||||||
|  |         } else { | ||||||
|  |             txrx = true; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  | 
 | ||||||
|  |     default: | ||||||
|  |         txrx = true; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if(txrx) { | ||||||
|  |         furi_hal_subghz_regulation = SubGhzRegulationTxRx; | ||||||
|  |     } else { | ||||||
|  |         furi_hal_subghz_regulation = SubGhzRegulationOnlyRx; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     uint32_t real_frequency = cc1101_set_frequency(device, value); |     uint32_t real_frequency = cc1101_set_frequency(device, value); | ||||||
|     cc1101_calibrate(device); |     cc1101_calibrate(device); | ||||||
| 
 | 
 | ||||||
| @ -480,7 +529,7 @@ void furi_hal_subghz_start_async_rx(FuriHalSubGhzCaptureCallback callback, void* | |||||||
|     TIM_InitStruct.Prescaler = 64 - 1; |     TIM_InitStruct.Prescaler = 64 - 1; | ||||||
|     TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP; |     TIM_InitStruct.CounterMode = LL_TIM_COUNTERMODE_UP; | ||||||
|     TIM_InitStruct.Autoreload = 0x7FFFFFFE; |     TIM_InitStruct.Autoreload = 0x7FFFFFFE; | ||||||
|     TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV1; |     TIM_InitStruct.ClockDivision = LL_TIM_CLOCKDIVISION_DIV4; | ||||||
|     LL_TIM_Init(TIM2, &TIM_InitStruct); |     LL_TIM_Init(TIM2, &TIM_InitStruct); | ||||||
| 
 | 
 | ||||||
|     // Timer: advanced
 |     // Timer: advanced
 | ||||||
| @ -503,7 +552,7 @@ void furi_hal_subghz_start_async_rx(FuriHalSubGhzCaptureCallback callback, void* | |||||||
|     LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_DIRECTTI); |     LL_TIM_IC_SetActiveInput(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ACTIVEINPUT_DIRECTTI); | ||||||
|     LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1); |     LL_TIM_IC_SetPrescaler(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_ICPSC_DIV1); | ||||||
|     LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_RISING); |     LL_TIM_IC_SetPolarity(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_POLARITY_RISING); | ||||||
|     LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV1); |     LL_TIM_IC_SetFilter(TIM2, LL_TIM_CHANNEL_CH2, LL_TIM_IC_FILTER_FDIV32_N8); | ||||||
| 
 | 
 | ||||||
|     // ISR setup
 |     // ISR setup
 | ||||||
|     furi_hal_interrupt_set_timer_isr(TIM2, furi_hal_subghz_capture_ISR); |     furi_hal_interrupt_set_timer_isr(TIM2, furi_hal_subghz_capture_ISR); | ||||||
| @ -608,6 +657,8 @@ static void furi_hal_subghz_async_tx_timer_isr() { | |||||||
|         if(LL_TIM_GetAutoReload(TIM2) == 0) { |         if(LL_TIM_GetAutoReload(TIM2) == 0) { | ||||||
|             if(furi_hal_subghz_state == SubGhzStateAsyncTx) { |             if(furi_hal_subghz_state == SubGhzStateAsyncTx) { | ||||||
|                 furi_hal_subghz_state = SubGhzStateAsyncTxLast; |                 furi_hal_subghz_state = SubGhzStateAsyncTxLast; | ||||||
|  |                 //forcibly pulls the pin to the ground so that there is no carrier
 | ||||||
|  |                 hal_gpio_init(&gpio_cc1101_g0, GpioModeInput, GpioPullDown, GpioSpeedLow); | ||||||
|             } else { |             } else { | ||||||
|                 furi_hal_subghz_state = SubGhzStateAsyncTxEnd; |                 furi_hal_subghz_state = SubGhzStateAsyncTxEnd; | ||||||
|                 LL_TIM_DisableCounter(TIM2); |                 LL_TIM_DisableCounter(TIM2); | ||||||
| @ -616,10 +667,13 @@ static void furi_hal_subghz_async_tx_timer_isr() { | |||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context) { | bool furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context) { | ||||||
|     furi_assert(furi_hal_subghz_state == SubGhzStateIdle); |     furi_assert(furi_hal_subghz_state == SubGhzStateIdle); | ||||||
|     furi_assert(callback); |     furi_assert(callback); | ||||||
| 
 | 
 | ||||||
|  |     //If transmission is prohibited by regional settings
 | ||||||
|  |     if(furi_hal_subghz_regulation != SubGhzRegulationTxRx) return false; | ||||||
|  | 
 | ||||||
|     furi_hal_subghz_async_tx.callback = callback; |     furi_hal_subghz_async_tx.callback = callback; | ||||||
|     furi_hal_subghz_async_tx.callback_context = context; |     furi_hal_subghz_async_tx.callback_context = context; | ||||||
| 
 | 
 | ||||||
| @ -694,6 +748,7 @@ void furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* | |||||||
| 
 | 
 | ||||||
|     LL_TIM_SetCounter(TIM2, 0); |     LL_TIM_SetCounter(TIM2, 0); | ||||||
|     LL_TIM_EnableCounter(TIM2); |     LL_TIM_EnableCounter(TIM2); | ||||||
|  |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool furi_hal_subghz_is_async_tx_complete() { | bool furi_hal_subghz_is_async_tx_complete() { | ||||||
|  | |||||||
| @ -35,8 +35,15 @@ typedef enum { | |||||||
|     SubGhzStateAsyncTx,             /** Async TX started, DMA and timer is on */ |     SubGhzStateAsyncTx,             /** Async TX started, DMA and timer is on */ | ||||||
|     SubGhzStateAsyncTxLast,         /** Async TX continue, DMA completed and timer got last value to go */ |     SubGhzStateAsyncTxLast,         /** Async TX continue, DMA completed and timer got last value to go */ | ||||||
|     SubGhzStateAsyncTxEnd,          /** Async TX complete, cleanup needed */ |     SubGhzStateAsyncTxEnd,          /** Async TX complete, cleanup needed */ | ||||||
|  | 
 | ||||||
| } SubGhzState; | } SubGhzState; | ||||||
| 
 | 
 | ||||||
|  | /** SubGhz regulation, receive transmission on the current frequency for the region */ | ||||||
|  | typedef enum { | ||||||
|  |     SubGhzRegulationOnlyRx, /**only Rx*/ | ||||||
|  |     SubGhzRegulationTxRx, /**TxRx*/ | ||||||
|  | } SubGhzRegulation; | ||||||
|  | 
 | ||||||
| /** Initialize and switch to power save mode
 | /** Initialize and switch to power save mode
 | ||||||
|  * Used by internal API-HAL initalization routine |  * Used by internal API-HAL initalization routine | ||||||
|  * Can be used to reinitialize device to safe state and send it to sleep |  * Can be used to reinitialize device to safe state and send it to sleep | ||||||
| @ -100,8 +107,10 @@ void furi_hal_subghz_idle(); | |||||||
| /** Switch to Recieve */ | /** Switch to Recieve */ | ||||||
| void furi_hal_subghz_rx(); | void furi_hal_subghz_rx(); | ||||||
| 
 | 
 | ||||||
| /** Switch to Transmit */ | /** Switch to Transmit
 | ||||||
| void furi_hal_subghz_tx(); | * @return true if the transfer is allowed by belonging to the region | ||||||
|  | */ | ||||||
|  | bool furi_hal_subghz_tx(); | ||||||
| 
 | 
 | ||||||
| /** Get RSSI value in dBm */ | /** Get RSSI value in dBm */ | ||||||
| float furi_hal_subghz_get_rssi(); | float furi_hal_subghz_get_rssi(); | ||||||
| @ -152,8 +161,9 @@ typedef LevelDuration (*FuriHalSubGhzAsyncTxCallback)(void* context); | |||||||
| 
 | 
 | ||||||
| /** Start async TX
 | /** Start async TX
 | ||||||
|  * Initializes GPIO, TIM2 and DMA1 for signal output |  * Initializes GPIO, TIM2 and DMA1 for signal output | ||||||
|  |  * @return true if the transfer is allowed by belonging to the region | ||||||
|  */ |  */ | ||||||
| void furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context); | bool furi_hal_subghz_start_async_tx(FuriHalSubGhzAsyncTxCallback callback, void* context); | ||||||
| 
 | 
 | ||||||
| /** Wait for async transmission to complete */ | /** Wait for async transmission to complete */ | ||||||
| bool furi_hal_subghz_is_async_tx_complete(); | bool furi_hal_subghz_is_async_tx_complete(); | ||||||
|  | |||||||
| @ -48,9 +48,9 @@ extern "C" { | |||||||
| #define CC1101_MCSM0                    0x18    /** Main Radio Control State Machine configuration */ | #define CC1101_MCSM0                    0x18    /** Main Radio Control State Machine configuration */ | ||||||
| #define CC1101_FOCCFG                   0x19    /** Frequency Offset Compensation configuration */ | #define CC1101_FOCCFG                   0x19    /** Frequency Offset Compensation configuration */ | ||||||
| #define CC1101_BSCFG                    0x1A    /** Bit Synchronization configuration */ | #define CC1101_BSCFG                    0x1A    /** Bit Synchronization configuration */ | ||||||
| #define CC1101_AGCTRL2                  0x1B    /** AGC control */ | #define CC1101_AGCCTRL2                 0x1B    /** AGC control */ | ||||||
| #define CC1101_AGCTRL1                  0x1C    /** AGC control */ | #define CC1101_AGCCTRL1                 0x1C    /** AGC control */ | ||||||
| #define CC1101_AGCTRL0                  0x1D    /** AGC control */ | #define CC1101_AGCCTRL0                 0x1D    /** AGC control */ | ||||||
| #define CC1101_WOREVT1                  0x1E    /** High byte Event 0 timeout */ | #define CC1101_WOREVT1                  0x1E    /** High byte Event 0 timeout */ | ||||||
| #define CC1101_WOREVT0                  0x1F    /** Low byte Event 0 timeout */ | #define CC1101_WOREVT0                  0x1F    /** Low byte Event 0 timeout */ | ||||||
| #define CC1101_WORCTRL                  0x20    /** Wake On Radio control */ | #define CC1101_WORCTRL                  0x20    /** Wake On Radio control */ | ||||||
|  | |||||||
| @ -8,12 +8,17 @@ | |||||||
| #define SUBGHZ_PT_SHORT 400 | #define SUBGHZ_PT_SHORT 400 | ||||||
| #define SUBGHZ_PT_LONG (SUBGHZ_PT_SHORT * 3) | #define SUBGHZ_PT_LONG (SUBGHZ_PT_SHORT * 3) | ||||||
| #define SUBGHZ_PT_GUARD (SUBGHZ_PT_SHORT * 30) | #define SUBGHZ_PT_GUARD (SUBGHZ_PT_SHORT * 30) | ||||||
|  | #define SUBGHZ_PT_COUNT_KEY 5 | ||||||
|  | #define SUBGHZ_PT_TIMEOUT 320 | ||||||
| 
 | 
 | ||||||
| struct SubGhzEncoderPrinceton { | struct SubGhzEncoderPrinceton { | ||||||
|     uint32_t key; |     uint32_t key; | ||||||
|     uint16_t te; |     uint16_t te; | ||||||
|     size_t repeat; |     size_t repeat; | ||||||
|     size_t front; |     size_t front; | ||||||
|  |     size_t count_key; | ||||||
|  |     uint32_t time_high; | ||||||
|  |     uint32_t time_low; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
| @ -45,8 +50,11 @@ void subghz_encoder_princeton_set(SubGhzEncoderPrinceton* instance, uint32_t key | |||||||
|     furi_assert(instance); |     furi_assert(instance); | ||||||
|     instance->te = SUBGHZ_PT_SHORT; |     instance->te = SUBGHZ_PT_SHORT; | ||||||
|     instance->key = key; |     instance->key = key; | ||||||
|     instance->repeat = repeat; |     instance->repeat = repeat + 1; | ||||||
|     instance->front = 48; |     instance->front = 48; | ||||||
|  |     instance->count_key = SUBGHZ_PT_COUNT_KEY + 7; | ||||||
|  |     instance->time_high = 0; | ||||||
|  |     instance->time_low = 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| size_t subghz_encoder_princeton_get_repeat_left(SubGhzEncoderPrinceton* instance) { | size_t subghz_encoder_princeton_get_repeat_left(SubGhzEncoderPrinceton* instance) { | ||||||
| @ -54,9 +62,25 @@ size_t subghz_encoder_princeton_get_repeat_left(SubGhzEncoderPrinceton* instance | |||||||
|     return instance->repeat; |     return instance->repeat; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void subghz_encoder_princeton_print_log(void* context) { | ||||||
|  |     SubGhzEncoderPrinceton* instance = context; | ||||||
|  |     float duty_cycle = | ||||||
|  |         ((float)instance->time_high / (instance->time_high + instance->time_low)) * 100; | ||||||
|  |     FURI_LOG_I( | ||||||
|  |         "EncoderPrinceton", | ||||||
|  |         "Radio ON=%dus, OFF=%dus, DutyCycle=%d,%d%%", | ||||||
|  |         instance->time_high, | ||||||
|  |         instance->time_low, | ||||||
|  |         (uint32_t)duty_cycle, | ||||||
|  |         (uint32_t)((duty_cycle - (uint32_t)duty_cycle) * 100)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| LevelDuration subghz_encoder_princeton_yield(void* context) { | LevelDuration subghz_encoder_princeton_yield(void* context) { | ||||||
|     SubGhzEncoderPrinceton* instance = context; |     SubGhzEncoderPrinceton* instance = context; | ||||||
|     if(instance->repeat == 0) return level_duration_reset(); |     if(instance->repeat == 0) { | ||||||
|  |         subghz_encoder_princeton_print_log(instance); | ||||||
|  |         return level_duration_reset(); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     size_t bit = instance->front / 2; |     size_t bit = instance->front / 2; | ||||||
|     bool level = !(instance->front % 2); |     bool level = !(instance->front % 2); | ||||||
| @ -68,11 +92,33 @@ LevelDuration subghz_encoder_princeton_yield(void* context) { | |||||||
|         bool value = (((uint8_t*)&instance->key)[2 - byte] >> (7 - bit_in_byte)) & 1; |         bool value = (((uint8_t*)&instance->key)[2 - byte] >> (7 - bit_in_byte)) & 1; | ||||||
|         if(value) { |         if(value) { | ||||||
|             ret = level_duration_make(level, level ? instance->te * 3 : instance->te); |             ret = level_duration_make(level, level ? instance->te * 3 : instance->te); | ||||||
|  |             if(level) | ||||||
|  |                 instance->time_high += instance->te * 3; | ||||||
|  |             else | ||||||
|  |                 instance->time_low += instance->te; | ||||||
|         } else { |         } else { | ||||||
|             ret = level_duration_make(level, level ? instance->te : instance->te * 3); |             ret = level_duration_make(level, level ? instance->te : instance->te * 3); | ||||||
|  |             if(level) | ||||||
|  |                 instance->time_high += instance->te; | ||||||
|  |             else | ||||||
|  |                 instance->time_low += instance->te * 3; | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|  |         if(--instance->count_key != 0) { | ||||||
|             ret = level_duration_make(level, level ? instance->te : instance->te * 30); |             ret = level_duration_make(level, level ? instance->te : instance->te * 30); | ||||||
|  |             if(level) | ||||||
|  |                 instance->time_high += instance->te; | ||||||
|  |             else | ||||||
|  |                 instance->time_low += instance->te * 30; | ||||||
|  |         } else { | ||||||
|  |             instance->count_key = SUBGHZ_PT_COUNT_KEY + 6; | ||||||
|  |             instance->front = 48; | ||||||
|  |             ret = level_duration_make(level, level ? instance->te : SUBGHZ_PT_TIMEOUT * 1000); | ||||||
|  |             if(level) | ||||||
|  |                 instance->time_high += instance->te; | ||||||
|  |             else | ||||||
|  |                 instance->time_low += SUBGHZ_PT_TIMEOUT * 1000; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     instance->front++; |     instance->front++; | ||||||
|  | |||||||
| @ -33,6 +33,11 @@ void subghz_encoder_princeton_set(SubGhzEncoderPrinceton* instance, uint32_t key | |||||||
|  */ |  */ | ||||||
| size_t subghz_encoder_princeton_get_repeat_left(SubGhzEncoderPrinceton* instance); | size_t subghz_encoder_princeton_get_repeat_left(SubGhzEncoderPrinceton* instance); | ||||||
| 
 | 
 | ||||||
|  | /** Print encoder log
 | ||||||
|  |  * @param instance - SubGhzEncoderPrinceton instance | ||||||
|  |  */ | ||||||
|  | void subghz_encoder_princeton_print_log(void* context); | ||||||
|  | 
 | ||||||
| /** Get level duration
 | /** Get level duration
 | ||||||
|  * @param instance - SubGhzEncoderPrinceton instance |  * @param instance - SubGhzEncoderPrinceton instance | ||||||
|  * @return level duration |  * @return level duration | ||||||
|  | |||||||
							
								
								
									
										246
									
								
								lib/subghz/protocols/subghz_protocol_scher_khan.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										246
									
								
								lib/subghz/protocols/subghz_protocol_scher_khan.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,246 @@ | |||||||
|  | #include "subghz_protocol_scher_khan.h" | ||||||
|  | 
 | ||||||
|  | //https://phreakerclub.com/72
 | ||||||
|  | //https://phreakerclub.com/forum/showthread.php?t=7&page=2
 | ||||||
|  | //https://phreakerclub.com/forum/showthread.php?t=274&highlight=magicar
 | ||||||
|  | //!!!  https://phreakerclub.com/forum/showthread.php?t=489&highlight=magicar&page=5
 | ||||||
|  | 
 | ||||||
|  | struct SubGhzProtocolScherKhan { | ||||||
|  |     SubGhzProtocolCommon common; | ||||||
|  |     const char* protocol_name; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     ScherKhanDecoderStepReset = 0, | ||||||
|  |     ScherKhanDecoderStepCheckPreambula, | ||||||
|  |     ScherKhanDecoderStepSaveDuration, | ||||||
|  |     ScherKhanDecoderStepCheckDuration, | ||||||
|  | } ScherKhanDecoderStep; | ||||||
|  | 
 | ||||||
|  | SubGhzProtocolScherKhan* subghz_protocol_scher_khan_alloc(void) { | ||||||
|  |     SubGhzProtocolScherKhan* instance = furi_alloc(sizeof(SubGhzProtocolScherKhan)); | ||||||
|  | 
 | ||||||
|  |     instance->common.name = "Scher-Khan"; | ||||||
|  |     instance->common.code_min_count_bit_for_found = 35; | ||||||
|  |     instance->common.te_short = 750; | ||||||
|  |     instance->common.te_long = 1100; | ||||||
|  |     instance->common.te_delta = 150; | ||||||
|  |     instance->common.type_protocol = SubGhzProtocolCommonTypeDynamic; | ||||||
|  |     instance->common.to_string = (SubGhzProtocolCommonToStr)subghz_protocol_scher_khan_to_str; | ||||||
|  |     instance->common.to_load_protocol = | ||||||
|  |         (SubGhzProtocolCommonLoadFromRAW)subghz_decoder_scher_khan_to_load_protocol; | ||||||
|  | 
 | ||||||
|  |     return instance; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_protocol_scher_khan_free(SubGhzProtocolScherKhan* instance) { | ||||||
|  |     furi_assert(instance); | ||||||
|  |     free(instance); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Send bit 
 | ||||||
|  |  *  | ||||||
|  |  * @param instance - SubGhzProtocolScherKhan instance | ||||||
|  |  * @param bit - bit | ||||||
|  |  */ | ||||||
|  | // void subghz_protocol_scher_khan_send_bit(SubGhzProtocolScherKhan* instance, uint8_t bit) {
 | ||||||
|  | //     if(bit) {
 | ||||||
|  | //         //send bit 1
 | ||||||
|  | //         SUBGHZ_TX_PIN_HIGH();
 | ||||||
|  | //         delay_us(instance->common.te_long);
 | ||||||
|  | //         SUBGHZ_TX_PIN_LOW();
 | ||||||
|  | //         delay_us(instance->common.te_short);
 | ||||||
|  | //     } else {
 | ||||||
|  | //         //send bit 0
 | ||||||
|  | //         SUBGHZ_TX_PIN_HIGH();
 | ||||||
|  | //         delay_us(instance->common.te_short);
 | ||||||
|  | //         SUBGHZ_TX_PIN_LOW();
 | ||||||
|  | //         delay_us(instance->common.te_long);
 | ||||||
|  | //     }
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | // void subghz_protocol_scher_khan_send_key(
 | ||||||
|  | //     SubGhzProtocolScherKhan* instance,
 | ||||||
|  | //     uint64_t key,
 | ||||||
|  | //     uint8_t bit,
 | ||||||
|  | //     uint8_t repeat) {
 | ||||||
|  | //     while(repeat--) {
 | ||||||
|  | //         SUBGHZ_TX_PIN_HIGH();
 | ||||||
|  | //         //Send header
 | ||||||
|  | //         delay_us(instance->common.te_long * 2);
 | ||||||
|  | //         SUBGHZ_TX_PIN_LOW();
 | ||||||
|  | //         delay_us(instance->common.te_long * 2);
 | ||||||
|  | //         //Send key data
 | ||||||
|  | //         for(uint8_t i = bit; i > 0; i--) {
 | ||||||
|  | //             subghz_protocol_scher_khan_send_bit(instance, bit_read(key, i - 1));
 | ||||||
|  | //         }
 | ||||||
|  | //     }
 | ||||||
|  | // }
 | ||||||
|  | 
 | ||||||
|  | void subghz_protocol_scher_khan_reset(SubGhzProtocolScherKhan* instance) { | ||||||
|  |     instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** Analysis of received data
 | ||||||
|  |  *  | ||||||
|  |  * @param instance SubGhzProtocolScherKhan instance | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_check_remote_controller(SubGhzProtocolScherKhan* instance) { | ||||||
|  |     /* 
 | ||||||
|  |     * MAGICAR 51 bit 00000001A99121DE83C3 MAGIC CODE, Dinamic | ||||||
|  |     * 0E8C1619E830C -> 000011101000110000010110 0001 1001 1110 1000001100001100 | ||||||
|  |     * 0E8C1629D830D -> 000011101000110000010110 0010 1001 1101 1000001100001101 | ||||||
|  |     * 0E8C1649B830E -> 000011101000110000010110 0100 1001 1011 1000001100001110 | ||||||
|  |     * 0E8C16897830F -> 000011101000110000010110 1000 1001 0111 1000001100001111 | ||||||
|  |     *                             Serial         Key  Ser ~Key   CNT | ||||||
|  |     */ | ||||||
|  | 
 | ||||||
|  |     switch(instance->common.code_last_count_bit) { | ||||||
|  |     // case 35: //MAGIC CODE, Static
 | ||||||
|  |     //     instance->protocol_name = "MAGIC CODE, Static";
 | ||||||
|  |     //     break;
 | ||||||
|  |     case 51: //MAGIC CODE, Dinamic
 | ||||||
|  |         instance->protocol_name = "MAGIC CODE, Dinamic"; | ||||||
|  |         instance->common.serial = ((instance->common.code_last_found >> 24) & 0xFFFFFF0) | | ||||||
|  |                                   ((instance->common.code_last_found >> 20) & 0x0F); | ||||||
|  |         instance->common.btn = (instance->common.code_last_found >> 24) & 0x0F; | ||||||
|  |         instance->common.cnt = instance->common.code_last_found & 0xFFFF; | ||||||
|  |         break; | ||||||
|  |         // case 57: //MAGIC CODE PRO / PRO2
 | ||||||
|  |         //     instance->protocol_name = "MAGIC CODE PRO / PRO2";
 | ||||||
|  |         //     break;
 | ||||||
|  | 
 | ||||||
|  |     default: | ||||||
|  |         instance->protocol_name = "Unknown"; | ||||||
|  |         instance->common.serial = 0; | ||||||
|  |         instance->common.btn = 0; | ||||||
|  |         instance->common.cnt = 0; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_protocol_scher_khan_parse( | ||||||
|  |     SubGhzProtocolScherKhan* instance, | ||||||
|  |     bool level, | ||||||
|  |     uint32_t duration) { | ||||||
|  |     switch(instance->common.parser_step) { | ||||||
|  |     case ScherKhanDecoderStepReset: | ||||||
|  |         if((level) && | ||||||
|  |            (DURATION_DIFF(duration, instance->common.te_short * 2) < instance->common.te_delta)) { | ||||||
|  |             instance->common.parser_step = ScherKhanDecoderStepCheckPreambula; | ||||||
|  |             instance->common.te_last = duration; | ||||||
|  |             instance->common.header_count = 0; | ||||||
|  |         } else { | ||||||
|  |             instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case ScherKhanDecoderStepCheckPreambula: | ||||||
|  |         if(level) { | ||||||
|  |             if((DURATION_DIFF(duration, instance->common.te_short * 2) < | ||||||
|  |                 instance->common.te_delta) || | ||||||
|  |                (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) { | ||||||
|  |                 instance->common.te_last = duration; | ||||||
|  |             } else { | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |             } | ||||||
|  |         } else if( | ||||||
|  |             (DURATION_DIFF(duration, instance->common.te_short * 2) < instance->common.te_delta) || | ||||||
|  |             (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) { | ||||||
|  |             if(DURATION_DIFF(instance->common.te_last, instance->common.te_short * 2) < | ||||||
|  |                instance->common.te_delta) { | ||||||
|  |                 // Found header
 | ||||||
|  |                 instance->common.header_count++; | ||||||
|  |                 break; | ||||||
|  |             } else if( | ||||||
|  |                 DURATION_DIFF(instance->common.te_last, instance->common.te_short) < | ||||||
|  |                 instance->common.te_delta) { | ||||||
|  |                 // Found start bit
 | ||||||
|  |                 if(instance->common.header_count >= 2) { | ||||||
|  |                     instance->common.parser_step = ScherKhanDecoderStepSaveDuration; | ||||||
|  |                     instance->common.code_found = 0; | ||||||
|  |                     instance->common.code_count_bit = 1; | ||||||
|  |                 } else { | ||||||
|  |                     instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case ScherKhanDecoderStepSaveDuration: | ||||||
|  |         if(level) { | ||||||
|  |             if(duration >= (instance->common.te_long + instance->common.te_delta * 2)) { | ||||||
|  |                 //Found stop bit
 | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |                 if(instance->common.code_count_bit >= | ||||||
|  |                    instance->common.code_min_count_bit_for_found) { | ||||||
|  |                     instance->common.code_last_found = instance->common.code_found; | ||||||
|  |                     instance->common.code_last_count_bit = instance->common.code_count_bit; | ||||||
|  |                     if(instance->common.callback) | ||||||
|  |                         instance->common.callback( | ||||||
|  |                             (SubGhzProtocolCommon*)instance, instance->common.context); | ||||||
|  |                 } | ||||||
|  |                 instance->common.code_found = 0; | ||||||
|  |                 instance->common.code_count_bit = 0; | ||||||
|  |                 break; | ||||||
|  |             } else { | ||||||
|  |                 instance->common.te_last = duration; | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepCheckDuration; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         } else { | ||||||
|  |             instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     case ScherKhanDecoderStepCheckDuration: | ||||||
|  |         if(!level) { | ||||||
|  |             if((DURATION_DIFF(instance->common.te_last, instance->common.te_short) < | ||||||
|  |                 instance->common.te_delta) && | ||||||
|  |                (DURATION_DIFF(duration, instance->common.te_short) < instance->common.te_delta)) { | ||||||
|  |                 subghz_protocol_common_add_bit(&instance->common, 0); | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepSaveDuration; | ||||||
|  |             } else if( | ||||||
|  |                 (DURATION_DIFF(instance->common.te_last, instance->common.te_long) < | ||||||
|  |                  instance->common.te_delta) && | ||||||
|  |                 (DURATION_DIFF(duration, instance->common.te_long) < instance->common.te_delta)) { | ||||||
|  |                 subghz_protocol_common_add_bit(&instance->common, 1); | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepSaveDuration; | ||||||
|  |             } else { | ||||||
|  |                 instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             instance->common.parser_step = ScherKhanDecoderStepReset; | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_protocol_scher_khan_to_str(SubGhzProtocolScherKhan* instance, string_t output) { | ||||||
|  |     subghz_protocol_scher_khan_check_remote_controller(instance); | ||||||
|  | 
 | ||||||
|  |     string_cat_printf( | ||||||
|  |         output, | ||||||
|  |         "%s %dbit\r\n" | ||||||
|  |         "Key:0x%lX%08lX\r\n" | ||||||
|  |         "Sn:%07lX Btn:%lX Cnt:%04X\r\n" | ||||||
|  |         "Pt: %s\r\n", | ||||||
|  |         instance->common.name, | ||||||
|  |         instance->common.code_last_count_bit, | ||||||
|  |         (uint32_t)(instance->common.code_last_found >> 32), | ||||||
|  |         (uint32_t)instance->common.code_last_found, | ||||||
|  |         instance->common.serial, | ||||||
|  |         instance->common.btn, | ||||||
|  |         instance->common.cnt, | ||||||
|  |         instance->protocol_name); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void subghz_decoder_scher_khan_to_load_protocol(SubGhzProtocolScherKhan* instance, void* context) { | ||||||
|  |     furi_assert(context); | ||||||
|  |     furi_assert(instance); | ||||||
|  |     SubGhzProtocolCommonLoad* data = context; | ||||||
|  |     instance->common.code_last_found = data->code_found; | ||||||
|  |     instance->common.code_last_count_bit = data->code_count_bit; | ||||||
|  |     subghz_protocol_scher_khan_check_remote_controller(instance); | ||||||
|  | } | ||||||
							
								
								
									
										58
									
								
								lib/subghz/protocols/subghz_protocol_scher_khan.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								lib/subghz/protocols/subghz_protocol_scher_khan.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include "subghz_protocol_common.h" | ||||||
|  | 
 | ||||||
|  | typedef struct SubGhzProtocolScherKhan SubGhzProtocolScherKhan; | ||||||
|  | 
 | ||||||
|  | /** Allocate SubGhzProtocolScherKhan
 | ||||||
|  |  *  | ||||||
|  |  * @return SubGhzProtocolScherKhan*  | ||||||
|  |  */ | ||||||
|  | SubGhzProtocolScherKhan* subghz_protocol_scher_khan_alloc(); | ||||||
|  | 
 | ||||||
|  | /** Free SubGhzProtocolScherKhan
 | ||||||
|  |  *  | ||||||
|  |  * @param instance  | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_free(SubGhzProtocolScherKhan* instance); | ||||||
|  | 
 | ||||||
|  | /** Sends the key on the air
 | ||||||
|  |  *  | ||||||
|  |  * @param instance - SubGhzProtocolScherKhan instance | ||||||
|  |  * @param key - key send | ||||||
|  |  * @param bit - count bit key | ||||||
|  |  * @param repeat - repeat send key | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_send_key(SubGhzProtocolScherKhan* instance, uint64_t key, uint8_t bit, uint8_t repeat); | ||||||
|  | 
 | ||||||
|  | /** Reset internal state
 | ||||||
|  |  * @param instance - SubGhzProtocolScherKhan instance | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_reset(SubGhzProtocolScherKhan* instance); | ||||||
|  | 
 | ||||||
|  | /** Analysis of received data
 | ||||||
|  |  *  | ||||||
|  |  * @param instance SubGhzProtocolScherKhan instance | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_check_remote_controller(SubGhzProtocolScherKhan* instance); | ||||||
|  | 
 | ||||||
|  | /** Parse accepted duration
 | ||||||
|  |  *  | ||||||
|  |  * @param instance - SubGhzProtocolScherKhan instance | ||||||
|  |  * @param data - LevelDuration level_duration | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_parse(SubGhzProtocolScherKhan* instance, bool level, uint32_t duration); | ||||||
|  | 
 | ||||||
|  | /** Outputting information from the parser
 | ||||||
|  |  *  | ||||||
|  |  * @param instance - SubGhzProtocolScherKhan* instance | ||||||
|  |  * @param output   - output string | ||||||
|  |  */ | ||||||
|  | void subghz_protocol_scher_khan_to_str(SubGhzProtocolScherKhan* instance, string_t output); | ||||||
|  | 
 | ||||||
|  | /** Loading protocol from bin data
 | ||||||
|  |  *  | ||||||
|  |  * @param instance - SubGhzProtocolScherKhan instance | ||||||
|  |  * @param context - SubGhzProtocolCommonLoad context | ||||||
|  |  */ | ||||||
|  | void subghz_decoder_scher_khan_to_load_protocol(SubGhzProtocolScherKhan* instance, void* context); | ||||||
| @ -12,6 +12,7 @@ | |||||||
| #include "protocols/subghz_protocol_nero_sketch.h" | #include "protocols/subghz_protocol_nero_sketch.h" | ||||||
| #include "protocols/subghz_protocol_star_line.h" | #include "protocols/subghz_protocol_star_line.h" | ||||||
| #include "protocols/subghz_protocol_nero_radio.h" | #include "protocols/subghz_protocol_nero_radio.h" | ||||||
|  | #include "protocols/subghz_protocol_scher_khan.h" | ||||||
| 
 | 
 | ||||||
| #include "subghz_keystore.h" | #include "subghz_keystore.h" | ||||||
| 
 | 
 | ||||||
| @ -30,6 +31,7 @@ typedef enum { | |||||||
|     SubGhzProtocolTypeNeroSketch, |     SubGhzProtocolTypeNeroSketch, | ||||||
|     SubGhzProtocolTypeStarLine, |     SubGhzProtocolTypeStarLine, | ||||||
|     SubGhzProtocolTypeNeroRadio, |     SubGhzProtocolTypeNeroRadio, | ||||||
|  |     SubGhzProtocolTypeScherKhan, | ||||||
| 
 | 
 | ||||||
|     SubGhzProtocolTypeMax, |     SubGhzProtocolTypeMax, | ||||||
| } SubGhzProtocolType; | } SubGhzProtocolType; | ||||||
| @ -93,6 +95,8 @@ SubGhzParser* subghz_parser_alloc() { | |||||||
|         (SubGhzProtocolCommon*)subghz_protocol_star_line_alloc(instance->keystore); |         (SubGhzProtocolCommon*)subghz_protocol_star_line_alloc(instance->keystore); | ||||||
|     instance->protocols[SubGhzProtocolTypeNeroRadio] = |     instance->protocols[SubGhzProtocolTypeNeroRadio] = | ||||||
|         (SubGhzProtocolCommon*)subghz_protocol_nero_radio_alloc(); |         (SubGhzProtocolCommon*)subghz_protocol_nero_radio_alloc(); | ||||||
|  |     instance->protocols[SubGhzProtocolTypeScherKhan] = | ||||||
|  |         (SubGhzProtocolCommon*)subghz_protocol_scher_khan_alloc(); | ||||||
| 
 | 
 | ||||||
|     return instance; |     return instance; | ||||||
| } | } | ||||||
| @ -120,6 +124,8 @@ void subghz_parser_free(SubGhzParser* instance) { | |||||||
|         (SubGhzProtocolStarLine*)instance->protocols[SubGhzProtocolTypeStarLine]); |         (SubGhzProtocolStarLine*)instance->protocols[SubGhzProtocolTypeStarLine]); | ||||||
|     subghz_protocol_nero_radio_free( |     subghz_protocol_nero_radio_free( | ||||||
|         (SubGhzProtocolNeroRadio*)instance->protocols[SubGhzProtocolTypeNeroRadio]); |         (SubGhzProtocolNeroRadio*)instance->protocols[SubGhzProtocolTypeNeroRadio]); | ||||||
|  |     subghz_protocol_scher_khan_free( | ||||||
|  |         (SubGhzProtocolScherKhan*)instance->protocols[SubGhzProtocolTypeScherKhan]); | ||||||
| 
 | 
 | ||||||
|     subghz_keystore_free(instance->keystore); |     subghz_keystore_free(instance->keystore); | ||||||
| 
 | 
 | ||||||
| @ -199,6 +205,8 @@ void subghz_parser_reset(SubGhzParser* instance) { | |||||||
|         (SubGhzProtocolStarLine*)instance->protocols[SubGhzProtocolTypeStarLine]); |         (SubGhzProtocolStarLine*)instance->protocols[SubGhzProtocolTypeStarLine]); | ||||||
|     subghz_protocol_nero_radio_reset( |     subghz_protocol_nero_radio_reset( | ||||||
|         (SubGhzProtocolNeroRadio*)instance->protocols[SubGhzProtocolTypeNeroRadio]); |         (SubGhzProtocolNeroRadio*)instance->protocols[SubGhzProtocolTypeNeroRadio]); | ||||||
|  |     subghz_protocol_scher_khan_reset( | ||||||
|  |         (SubGhzProtocolScherKhan*)instance->protocols[SubGhzProtocolTypeScherKhan]); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void subghz_parser_parse(SubGhzParser* instance, bool level, uint32_t duration) { | void subghz_parser_parse(SubGhzParser* instance, bool level, uint32_t duration) { | ||||||
| @ -232,4 +240,8 @@ void subghz_parser_parse(SubGhzParser* instance, bool level, uint32_t duration) | |||||||
|         (SubGhzProtocolNeroRadio*)instance->protocols[SubGhzProtocolTypeNeroRadio], |         (SubGhzProtocolNeroRadio*)instance->protocols[SubGhzProtocolTypeNeroRadio], | ||||||
|         level, |         level, | ||||||
|         duration); |         duration); | ||||||
|  |     subghz_protocol_scher_khan_parse( | ||||||
|  |         (SubGhzProtocolScherKhan*)instance->protocols[SubGhzProtocolTypeScherKhan], | ||||||
|  |         level, | ||||||
|  |         duration); | ||||||
| } | } | ||||||
|  | |||||||
| @ -10,6 +10,10 @@ struct SubGhzWorker { | |||||||
|     volatile bool running; |     volatile bool running; | ||||||
|     volatile bool overrun; |     volatile bool overrun; | ||||||
| 
 | 
 | ||||||
|  |     LevelDuration filter_level_duration; | ||||||
|  |     bool filter_running; | ||||||
|  |     uint16_t filter_duration; | ||||||
|  | 
 | ||||||
|     SubGhzWorkerOverrunCallback overrun_callback; |     SubGhzWorkerOverrunCallback overrun_callback; | ||||||
|     SubGhzWorkerPairCallback pair_callback; |     SubGhzWorkerPairCallback pair_callback; | ||||||
|     void* context; |     void* context; | ||||||
| @ -30,8 +34,8 @@ void subghz_worker_rx_callback(bool level, uint32_t duration, void* context) { | |||||||
|         instance->overrun = false; |         instance->overrun = false; | ||||||
|         level_duration = level_duration_reset(); |         level_duration = level_duration_reset(); | ||||||
|     } |     } | ||||||
|     size_t ret = |     size_t ret = xStreamBufferSendFromISR( | ||||||
|         xStreamBufferSendFromISR(instance->stream, &level_duration, sizeof(LevelDuration), &xHigherPriorityTaskWoken); |         instance->stream, &level_duration, sizeof(LevelDuration), &xHigherPriorityTaskWoken); | ||||||
|     if(sizeof(LevelDuration) != ret) instance->overrun = true; |     if(sizeof(LevelDuration) != ret) instance->overrun = true; | ||||||
|     portYIELD_FROM_ISR(xHigherPriorityTaskWoken); |     portYIELD_FROM_ISR(xHigherPriorityTaskWoken); | ||||||
| } | } | ||||||
| @ -46,15 +50,35 @@ static int32_t subghz_worker_thread_callback(void* context) { | |||||||
| 
 | 
 | ||||||
|     LevelDuration level_duration; |     LevelDuration level_duration; | ||||||
|     while(instance->running) { |     while(instance->running) { | ||||||
|         int ret = xStreamBufferReceive(instance->stream, &level_duration, sizeof(LevelDuration), 10); |         int ret = | ||||||
|  |             xStreamBufferReceive(instance->stream, &level_duration, sizeof(LevelDuration), 10); | ||||||
|         if(ret == sizeof(LevelDuration)) { |         if(ret == sizeof(LevelDuration)) { | ||||||
|             if(level_duration_is_reset(level_duration)) { |             if(level_duration_is_reset(level_duration)) { | ||||||
|                 printf("."); |                 printf("."); | ||||||
|                 if (instance->overrun_callback) instance->overrun_callback(instance->context); |                 if(instance->overrun_callback) instance->overrun_callback(instance->context); | ||||||
|             } else { |             } else { | ||||||
|                 bool level = level_duration_get_level(level_duration); |                 bool level = level_duration_get_level(level_duration); | ||||||
|                 uint32_t duration = level_duration_get_duration(level_duration); |                 uint32_t duration = level_duration_get_duration(level_duration); | ||||||
|                 if (instance->pair_callback) instance->pair_callback(instance->context, level, duration); | 
 | ||||||
|  |                 if(instance->filter_running) { | ||||||
|  |                     if((duration < instance->filter_duration) || | ||||||
|  |                        (instance->filter_level_duration.level == level)) { | ||||||
|  |                         instance->filter_level_duration.duration += duration; | ||||||
|  | 
 | ||||||
|  |                     } else if(instance->filter_level_duration.level != level) { | ||||||
|  |                         if(instance->pair_callback) | ||||||
|  |                             instance->pair_callback( | ||||||
|  |                                 instance->context, | ||||||
|  |                                 instance->filter_level_duration.level, | ||||||
|  |                                 instance->filter_level_duration.duration); | ||||||
|  | 
 | ||||||
|  |                         instance->filter_level_duration.duration = duration; | ||||||
|  |                         instance->filter_level_duration.level = level; | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     if(instance->pair_callback) | ||||||
|  |                         instance->pair_callback(instance->context, level, duration); | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @ -73,6 +97,10 @@ SubGhzWorker* subghz_worker_alloc() { | |||||||
| 
 | 
 | ||||||
|     instance->stream = xStreamBufferCreate(sizeof(LevelDuration) * 1024, sizeof(LevelDuration)); |     instance->stream = xStreamBufferCreate(sizeof(LevelDuration) * 1024, sizeof(LevelDuration)); | ||||||
| 
 | 
 | ||||||
|  |     //setting filter
 | ||||||
|  |     instance->filter_running = true; | ||||||
|  |     instance->filter_duration = 20; | ||||||
|  | 
 | ||||||
|     return instance; |     return instance; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -85,7 +113,9 @@ void subghz_worker_free(SubGhzWorker* instance) { | |||||||
|     free(instance); |     free(instance); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void subghz_worker_set_overrun_callback(SubGhzWorker* instance, SubGhzWorkerOverrunCallback callback) { | void subghz_worker_set_overrun_callback( | ||||||
|  |     SubGhzWorker* instance, | ||||||
|  |     SubGhzWorkerOverrunCallback callback) { | ||||||
|     furi_assert(instance); |     furi_assert(instance); | ||||||
|     instance->overrun_callback = callback; |     instance->overrun_callback = callback; | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Skorpionm
						Skorpionm