2022-06-21 15:45:50 +03:00
|
|
|
#include "infrared_brute_force.h"
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <m-dict.h>
|
2025-02-20 03:58:55 +04:00
|
|
|
#include <m-array.h>
|
2022-06-21 15:45:50 +03:00
|
|
|
#include <flipper_format/flipper_format.h>
|
|
|
|
|
|
|
|
|
|
#include "infrared_signal.h"
|
|
|
|
|
|
2025-02-20 03:58:55 +04:00
|
|
|
ARRAY_DEF(SignalPositionArray, size_t, M_DEFAULT_OPLIST);
|
|
|
|
|
|
2022-06-21 15:45:50 +03:00
|
|
|
typedef struct {
|
2025-02-20 03:58:55 +04:00
|
|
|
size_t index;
|
|
|
|
|
SignalPositionArray_t signals;
|
2022-06-21 15:45:50 +03:00
|
|
|
} InfraredBruteForceRecord;
|
|
|
|
|
|
2025-02-20 03:58:55 +04:00
|
|
|
static inline void ir_bf_record_init(InfraredBruteForceRecord* record) {
|
|
|
|
|
record->index = 0;
|
|
|
|
|
SignalPositionArray_init(record->signals);
|
|
|
|
|
}
|
|
|
|
|
#define IR_BF_RECORD_INIT(r) (ir_bf_record_init(&(r)))
|
|
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
|
ir_bf_record_init_set(InfraredBruteForceRecord* dest, const InfraredBruteForceRecord* src) {
|
|
|
|
|
dest->index = src->index;
|
|
|
|
|
SignalPositionArray_init_set(dest->signals, src->signals);
|
|
|
|
|
}
|
|
|
|
|
#define IR_BF_RECORD_INIT_SET(d, s) (ir_bf_record_init_set(&(d), &(s)))
|
|
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
|
ir_bf_record_set(InfraredBruteForceRecord* dest, const InfraredBruteForceRecord* src) {
|
|
|
|
|
dest->index = src->index;
|
|
|
|
|
SignalPositionArray_set(dest->signals, src->signals);
|
|
|
|
|
}
|
|
|
|
|
#define IR_BF_RECORD_SET(d, s) (ir_bf_record_set(&(d), &(s)))
|
|
|
|
|
|
|
|
|
|
static inline void ir_bf_record_clear(InfraredBruteForceRecord* record) {
|
|
|
|
|
SignalPositionArray_clear(record->signals);
|
|
|
|
|
}
|
|
|
|
|
#define IR_BF_RECORD_CLEAR(r) (ir_bf_record_clear(&(r)))
|
|
|
|
|
|
|
|
|
|
#define IR_BF_RECORD_OPLIST \
|
|
|
|
|
(INIT(IR_BF_RECORD_INIT), \
|
|
|
|
|
INIT_SET(IR_BF_RECORD_INIT_SET), \
|
|
|
|
|
SET(IR_BF_RECORD_SET), \
|
|
|
|
|
CLEAR(IR_BF_RECORD_CLEAR))
|
|
|
|
|
|
2022-06-21 15:45:50 +03:00
|
|
|
DICT_DEF2(
|
|
|
|
|
InfraredBruteForceRecordDict,
|
2022-10-06 01:15:23 +10:00
|
|
|
FuriString*,
|
|
|
|
|
FURI_STRING_OPLIST,
|
2022-06-21 15:45:50 +03:00
|
|
|
InfraredBruteForceRecord,
|
2025-02-20 03:58:55 +04:00
|
|
|
IR_BF_RECORD_OPLIST);
|
2022-06-21 15:45:50 +03:00
|
|
|
|
|
|
|
|
struct InfraredBruteForce {
|
|
|
|
|
FlipperFormat* ff;
|
|
|
|
|
const char* db_filename;
|
2022-10-06 01:15:23 +10:00
|
|
|
FuriString* current_record_name;
|
2025-02-20 03:58:55 +04:00
|
|
|
InfraredBruteForceRecord current_record;
|
2022-09-27 20:11:28 +03:00
|
|
|
InfraredSignal* current_signal;
|
2022-06-21 15:45:50 +03:00
|
|
|
InfraredBruteForceRecordDict_t records;
|
2022-09-27 20:11:28 +03:00
|
|
|
bool is_started;
|
2022-06-21 15:45:50 +03:00
|
|
|
};
|
|
|
|
|
|
2024-03-19 23:43:52 +09:00
|
|
|
InfraredBruteForce* infrared_brute_force_alloc(void) {
|
2022-06-21 15:45:50 +03:00
|
|
|
InfraredBruteForce* brute_force = malloc(sizeof(InfraredBruteForce));
|
|
|
|
|
brute_force->ff = NULL;
|
|
|
|
|
brute_force->db_filename = NULL;
|
2022-09-27 20:11:28 +03:00
|
|
|
brute_force->current_signal = NULL;
|
|
|
|
|
brute_force->is_started = false;
|
2022-10-06 01:15:23 +10:00
|
|
|
brute_force->current_record_name = furi_string_alloc();
|
2022-06-21 15:45:50 +03:00
|
|
|
InfraredBruteForceRecordDict_init(brute_force->records);
|
|
|
|
|
return brute_force;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void infrared_brute_force_free(InfraredBruteForce* brute_force) {
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(!brute_force->is_started);
|
2022-06-21 15:45:50 +03:00
|
|
|
InfraredBruteForceRecordDict_clear(brute_force->records);
|
2022-10-06 01:15:23 +10:00
|
|
|
furi_string_free(brute_force->current_record_name);
|
2022-06-21 15:45:50 +03:00
|
|
|
free(brute_force);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void infrared_brute_force_set_db_filename(InfraredBruteForce* brute_force, const char* db_filename) {
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(!brute_force->is_started);
|
2022-06-21 15:45:50 +03:00
|
|
|
brute_force->db_filename = db_filename;
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-06 12:52:00 +03:00
|
|
|
InfraredErrorCode infrared_brute_force_calculate_messages(InfraredBruteForce* brute_force) {
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(!brute_force->is_started);
|
2022-06-21 15:45:50 +03:00
|
|
|
furi_assert(brute_force->db_filename);
|
2024-09-06 12:52:00 +03:00
|
|
|
InfraredErrorCode error = InfraredErrorCodeNone;
|
2022-06-21 15:45:50 +03:00
|
|
|
|
2022-07-26 15:21:51 +03:00
|
|
|
Storage* storage = furi_record_open(RECORD_STORAGE);
|
2022-07-22 19:00:25 +03:00
|
|
|
FlipperFormat* ff = flipper_format_buffered_file_alloc(storage);
|
2024-02-12 05:30:10 +03:00
|
|
|
FuriString* signal_name = furi_string_alloc();
|
|
|
|
|
InfraredSignal* signal = infrared_signal_alloc();
|
|
|
|
|
|
|
|
|
|
do {
|
2024-09-06 12:52:00 +03:00
|
|
|
if(!flipper_format_buffered_file_open_existing(ff, brute_force->db_filename)) {
|
|
|
|
|
error = InfraredErrorCodeFileOperationFailed;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2024-02-12 05:30:10 +03:00
|
|
|
|
2025-02-20 03:58:55 +04:00
|
|
|
bool signal_valid = false;
|
2024-09-06 12:52:00 +03:00
|
|
|
while(infrared_signal_read_name(ff, signal_name) == InfraredErrorCodeNone) {
|
2025-03-31 16:52:15 +00:00
|
|
|
size_t signal_start = flipper_format_tell(ff);
|
2024-09-06 12:52:00 +03:00
|
|
|
error = infrared_signal_read_body(signal, ff);
|
2025-02-20 03:58:55 +04:00
|
|
|
signal_valid = (!INFRARED_ERROR_PRESENT(error)) && infrared_signal_is_valid(signal);
|
|
|
|
|
if(!signal_valid) break;
|
2022-06-21 15:45:50 +03:00
|
|
|
|
|
|
|
|
InfraredBruteForceRecord* record =
|
|
|
|
|
InfraredBruteForceRecordDict_get(brute_force->records, signal_name);
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_assert(record);
|
|
|
|
|
SignalPositionArray_push_back(record->signals, signal_start);
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
2025-02-20 03:58:55 +04:00
|
|
|
if(!signal_valid) break;
|
2024-02-12 05:30:10 +03:00
|
|
|
} while(false);
|
|
|
|
|
|
|
|
|
|
infrared_signal_free(signal);
|
|
|
|
|
furi_string_free(signal_name);
|
2022-06-21 15:45:50 +03:00
|
|
|
|
|
|
|
|
flipper_format_free(ff);
|
2022-07-26 15:21:51 +03:00
|
|
|
furi_record_close(RECORD_STORAGE);
|
2024-09-06 12:52:00 +03:00
|
|
|
return error;
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool infrared_brute_force_start(
|
|
|
|
|
InfraredBruteForce* brute_force,
|
|
|
|
|
uint32_t index,
|
|
|
|
|
uint32_t* record_count) {
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(!brute_force->is_started);
|
2022-06-21 15:45:50 +03:00
|
|
|
bool success = false;
|
|
|
|
|
*record_count = 0;
|
|
|
|
|
|
|
|
|
|
InfraredBruteForceRecordDict_it_t it;
|
|
|
|
|
for(InfraredBruteForceRecordDict_it(it, brute_force->records);
|
|
|
|
|
!InfraredBruteForceRecordDict_end_p(it);
|
|
|
|
|
InfraredBruteForceRecordDict_next(it)) {
|
|
|
|
|
const InfraredBruteForceRecordDict_itref_t* record = InfraredBruteForceRecordDict_cref(it);
|
|
|
|
|
if(record->value.index == index) {
|
2025-02-20 03:58:55 +04:00
|
|
|
*record_count = SignalPositionArray_size(record->value.signals);
|
2022-06-21 15:45:50 +03:00
|
|
|
if(*record_count) {
|
2022-10-06 01:15:23 +10:00
|
|
|
furi_string_set(brute_force->current_record_name, record->key);
|
2025-02-20 03:58:55 +04:00
|
|
|
brute_force->current_record = record->value;
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(*record_count) {
|
2022-07-26 15:21:51 +03:00
|
|
|
Storage* storage = furi_record_open(RECORD_STORAGE);
|
2022-07-22 19:00:25 +03:00
|
|
|
brute_force->ff = flipper_format_buffered_file_alloc(storage);
|
2022-09-27 20:11:28 +03:00
|
|
|
brute_force->current_signal = infrared_signal_alloc();
|
|
|
|
|
brute_force->is_started = true;
|
2022-07-22 19:00:25 +03:00
|
|
|
success =
|
|
|
|
|
flipper_format_buffered_file_open_existing(brute_force->ff, brute_force->db_filename);
|
2022-09-27 20:11:28 +03:00
|
|
|
if(!success) infrared_brute_force_stop(brute_force);
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
return success;
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-30 19:20:35 +03:00
|
|
|
bool infrared_brute_force_is_started(const InfraredBruteForce* brute_force) {
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
return brute_force->is_started;
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void infrared_brute_force_stop(InfraredBruteForce* brute_force) {
|
2025-02-20 03:58:55 +04:00
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(brute_force->is_started);
|
2022-10-06 01:15:23 +10:00
|
|
|
furi_string_reset(brute_force->current_record_name);
|
2022-09-27 20:11:28 +03:00
|
|
|
infrared_signal_free(brute_force->current_signal);
|
2022-06-21 15:45:50 +03:00
|
|
|
flipper_format_free(brute_force->ff);
|
2022-09-27 20:11:28 +03:00
|
|
|
brute_force->current_signal = NULL;
|
2022-06-21 15:45:50 +03:00
|
|
|
brute_force->ff = NULL;
|
2022-09-27 20:11:28 +03:00
|
|
|
brute_force->is_started = false;
|
|
|
|
|
furi_record_close(RECORD_STORAGE);
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
|
2025-02-20 03:58:55 +04:00
|
|
|
bool infrared_brute_force_send(InfraredBruteForce* brute_force, uint32_t signal_index) {
|
|
|
|
|
furi_check(brute_force);
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(brute_force->is_started);
|
2024-09-06 12:52:00 +03:00
|
|
|
|
2025-02-20 03:58:55 +04:00
|
|
|
if(signal_index >= SignalPositionArray_size(brute_force->current_record.signals)) return false;
|
|
|
|
|
|
|
|
|
|
size_t signal_start =
|
|
|
|
|
*SignalPositionArray_cget(brute_force->current_record.signals, signal_index);
|
|
|
|
|
if(!flipper_format_seek(brute_force->ff, signal_start, FlipperFormatOffsetFromStart))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
if(INFRARED_ERROR_PRESENT(
|
|
|
|
|
infrared_signal_read_body(brute_force->current_signal, brute_force->ff)))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
infrared_signal_transmit(brute_force->current_signal);
|
|
|
|
|
return true;
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void infrared_brute_force_add_record(
|
|
|
|
|
InfraredBruteForce* brute_force,
|
|
|
|
|
uint32_t index,
|
|
|
|
|
const char* name) {
|
2025-02-20 03:58:55 +04:00
|
|
|
InfraredBruteForceRecord value;
|
|
|
|
|
ir_bf_record_init(&value);
|
|
|
|
|
value.index = index;
|
2022-10-06 01:15:23 +10:00
|
|
|
FuriString* key;
|
|
|
|
|
key = furi_string_alloc_set(name);
|
2022-06-21 15:45:50 +03:00
|
|
|
InfraredBruteForceRecordDict_set_at(brute_force->records, key, value);
|
2022-10-06 01:15:23 +10:00
|
|
|
furi_string_free(key);
|
2022-06-21 15:45:50 +03:00
|
|
|
}
|
2022-09-22 19:13:00 +03:00
|
|
|
|
|
|
|
|
void infrared_brute_force_reset(InfraredBruteForce* brute_force) {
|
2022-09-27 20:11:28 +03:00
|
|
|
furi_assert(!brute_force->is_started);
|
2022-09-22 19:13:00 +03:00
|
|
|
InfraredBruteForceRecordDict_reset(brute_force->records);
|
|
|
|
|
}
|