flipperzero-firmware/lib/subghz/receiver.c
Skorpionm 163be139eb
SubGhz: add protocol BinRAW (binarization of data quantized by the minimum correlated duration) (#2322)
* 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>
2023-02-09 13:48:06 +09:00

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;
}