 163be139eb
			
		
	
	
		163be139eb
		
			
		
	
	
	
	
		
			
			* SubGhz: add protocol DataRAW (binarization of data quantized by the minimum correlated duration) * SubGhz: fix name history * SubGhz: add encoder Data_RAW protocol * SubGhz: decreasing the size of the LevelDuration structure * SubGhz: history, added check that there is free RAM * SubGhz: checking for free memory, support to pass without gap * SubGhz: add running average to average the result, auto cut noise at the end of a burst * SubGhz: support for repeating sequences * SubGhz: fix secplus_v2 decoder * SubGhz: bin_RAW fix add history * SubGhz: add debug * SubGhz: debug refactoring * FURI_LOG: add FURI_LOG_RAW_x formatted string output like printf * SubGhz: fix new FURI_LOG metod * FURI_LOG: fix unit test * SubGhz: add enable/disable BinRAW protocol decoding * SubGhz: fix PVS * SubGhz: forcibly turn off the speaker when exiting SubGhz * SubGhz: adaptive adjustment to the noise level Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
		
			
				
	
	
		
			125 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			125 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "receiver.h"
 | |
| 
 | |
| #include "registry.h"
 | |
| #include "protocols/protocol_items.h"
 | |
| 
 | |
| #include <m-array.h>
 | |
| 
 | |
| typedef struct {
 | |
|     SubGhzProtocolEncoderBase* base;
 | |
| } SubGhzReceiverSlot;
 | |
| 
 | |
| ARRAY_DEF(SubGhzReceiverSlotArray, SubGhzReceiverSlot, M_POD_OPLIST);
 | |
| #define M_OPL_SubGhzReceiverSlotArray_t() ARRAY_OPLIST(SubGhzReceiverSlotArray, M_POD_OPLIST)
 | |
| 
 | |
| struct SubGhzReceiver {
 | |
|     SubGhzReceiverSlotArray_t slots;
 | |
|     SubGhzProtocolFlag filter;
 | |
| 
 | |
|     SubGhzReceiverCallback callback;
 | |
|     void* context;
 | |
| };
 | |
| 
 | |
| SubGhzReceiver* subghz_receiver_alloc_init(SubGhzEnvironment* environment) {
 | |
|     SubGhzReceiver* instance = malloc(sizeof(SubGhzReceiver));
 | |
|     SubGhzReceiverSlotArray_init(instance->slots);
 | |
|     const SubGhzProtocolRegistry* protocol_registry_items =
 | |
|         subghz_environment_get_protocol_registry(environment);
 | |
| 
 | |
|     for(size_t i = 0; i < subghz_protocol_registry_count(protocol_registry_items); ++i) {
 | |
|         const SubGhzProtocol* protocol =
 | |
|             subghz_protocol_registry_get_by_index(protocol_registry_items, i);
 | |
| 
 | |
|         if(protocol->decoder && protocol->decoder->alloc) {
 | |
|             SubGhzReceiverSlot* slot = SubGhzReceiverSlotArray_push_new(instance->slots);
 | |
|             slot->base = protocol->decoder->alloc(environment);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     instance->callback = NULL;
 | |
|     instance->context = NULL;
 | |
|     return instance;
 | |
| }
 | |
| 
 | |
| void subghz_receiver_free(SubGhzReceiver* instance) {
 | |
|     furi_assert(instance);
 | |
| 
 | |
|     instance->callback = NULL;
 | |
|     instance->context = NULL;
 | |
| 
 | |
|     // Release allocated slots
 | |
|     for
 | |
|         M_EACH(slot, instance->slots, SubGhzReceiverSlotArray_t) {
 | |
|             slot->base->protocol->decoder->free(slot->base);
 | |
|             slot->base = NULL;
 | |
|         }
 | |
|     SubGhzReceiverSlotArray_clear(instance->slots);
 | |
| 
 | |
|     free(instance);
 | |
| }
 | |
| 
 | |
| void subghz_receiver_decode(SubGhzReceiver* instance, bool level, uint32_t duration) {
 | |
|     furi_assert(instance);
 | |
|     furi_assert(instance->slots);
 | |
| 
 | |
|     for
 | |
|         M_EACH(slot, instance->slots, SubGhzReceiverSlotArray_t) {
 | |
|             if((slot->base->protocol->flag & instance->filter) != 0) {
 | |
|                 slot->base->protocol->decoder->feed(slot->base, level, duration);
 | |
|             }
 | |
|         }
 | |
| }
 | |
| 
 | |
| void subghz_receiver_reset(SubGhzReceiver* instance) {
 | |
|     furi_assert(instance);
 | |
|     furi_assert(instance->slots);
 | |
| 
 | |
|     for
 | |
|         M_EACH(slot, instance->slots, SubGhzReceiverSlotArray_t) {
 | |
|             slot->base->protocol->decoder->reset(slot->base);
 | |
|         }
 | |
| }
 | |
| 
 | |
| static void subghz_receiver_rx_callback(SubGhzProtocolDecoderBase* decoder_base, void* context) {
 | |
|     SubGhzReceiver* instance = context;
 | |
|     if(instance->callback) {
 | |
|         instance->callback(instance, decoder_base, instance->context);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void subghz_receiver_set_rx_callback(
 | |
|     SubGhzReceiver* instance,
 | |
|     SubGhzReceiverCallback callback,
 | |
|     void* context) {
 | |
|     furi_assert(instance);
 | |
| 
 | |
|     for
 | |
|         M_EACH(slot, instance->slots, SubGhzReceiverSlotArray_t) {
 | |
|             subghz_protocol_decoder_base_set_decoder_callback(
 | |
|                 (SubGhzProtocolDecoderBase*)slot->base, subghz_receiver_rx_callback, instance);
 | |
|         }
 | |
| 
 | |
|     instance->callback = callback;
 | |
|     instance->context = context;
 | |
| }
 | |
| 
 | |
| void subghz_receiver_set_filter(SubGhzReceiver* instance, SubGhzProtocolFlag filter) {
 | |
|     furi_assert(instance);
 | |
|     instance->filter = filter;
 | |
| }
 | |
| 
 | |
| SubGhzProtocolDecoderBase* subghz_receiver_search_decoder_base_by_name(
 | |
|     SubGhzReceiver* instance,
 | |
|     const char* decoder_name) {
 | |
|     SubGhzProtocolDecoderBase* result = NULL;
 | |
| 
 | |
|     for
 | |
|         M_EACH(slot, instance->slots, SubGhzReceiverSlotArray_t) {
 | |
|             if(strcmp(slot->base->protocol->name, decoder_name) == 0) {
 | |
|                 result = (SubGhzProtocolDecoderBase*)slot->base;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     return result;
 | |
| }
 |