 49dcf81743
			
		
	
	
		49dcf81743
		
			
		
	
	
	
	
		
			
			* Do not load all signals at once (Draft) * Minor cleanup * Refactor remote renaming * Improve function signatures * Rename infrared_remote functions * Optimise signal loading * Implement adding signals to remote * Add read_name() method * Deprecate a function * Partially implement deleting signals (draft) * Use m-array instead of m-list for signal name directory * Use plain C strings instead of furi_string * Implement deleting signals * Implement deleting signals via generalised callback * Implement renaming signals * Rename some types * Some more renaming * Remove unused type * Implement inserting signals (internal use) * Improve InfraredMoveView * Send an event to move a signal * Remove unused type * Implement moving signals * Implement creating new remotes with one signal * Un-deprecate and rename a function * Add InfraredRemote API docs * Add InfraredSignal API docs * Better error messages * Show progress pop-up when moving buttons in a remote * Copy labels to the InfraredMoveView to avoid pointer invalidation * Improve file selection scene * Show progress pop-up when renaming buttons in a remote * Refactor a scene * Show progress when deleting a button from remote * Use a random name for temp files * Add docs to infrared_brute_force.h * Rename Infrared type to InfraredApp * Add docs to infrared_app_i.h * Deliver event data via a callback * Bundle event data together with event type * Change DataExchange behaviour * Adapt RPC debug app to new API * Remove rogue output * Add Doxygen comments to rpc_app.h * Simplify rpc_app.c code * Remove superflous parameter * Do not allocate protobuf messages on the stack * Fix GetError response * Support for button indices * Comment out shallow submodules * Fix F18 api * Fix logical error and add more debug output * fbt: testing unshallow for protobuf * github: lint&checks: unshallow prior to checks * Fix a TODO * github: do not unshallow the unshallowed * fbt: assets: only attempt to unshallow if cannot describe * Do not use the name when loading a signal by index (duh) * Simplify loading infrared signals by name * Sync with protobuf release * Infrared: use compact furi_crash macros Co-authored-by: hedger <hedger@nanode.su> Co-authored-by: hedger <hedger@users.noreply.github.com> Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
		
			
				
	
	
		
			156 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "infrared_brute_force.h"
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <m-dict.h>
 | |
| #include <flipper_format/flipper_format.h>
 | |
| 
 | |
| #include "infrared_signal.h"
 | |
| 
 | |
| typedef struct {
 | |
|     uint32_t index;
 | |
|     uint32_t count;
 | |
| } InfraredBruteForceRecord;
 | |
| 
 | |
| DICT_DEF2(
 | |
|     InfraredBruteForceRecordDict,
 | |
|     FuriString*,
 | |
|     FURI_STRING_OPLIST,
 | |
|     InfraredBruteForceRecord,
 | |
|     M_POD_OPLIST);
 | |
| 
 | |
| struct InfraredBruteForce {
 | |
|     FlipperFormat* ff;
 | |
|     const char* db_filename;
 | |
|     FuriString* current_record_name;
 | |
|     InfraredSignal* current_signal;
 | |
|     InfraredBruteForceRecordDict_t records;
 | |
|     bool is_started;
 | |
| };
 | |
| 
 | |
| InfraredBruteForce* infrared_brute_force_alloc() {
 | |
|     InfraredBruteForce* brute_force = malloc(sizeof(InfraredBruteForce));
 | |
|     brute_force->ff = NULL;
 | |
|     brute_force->db_filename = NULL;
 | |
|     brute_force->current_signal = NULL;
 | |
|     brute_force->is_started = false;
 | |
|     brute_force->current_record_name = furi_string_alloc();
 | |
|     InfraredBruteForceRecordDict_init(brute_force->records);
 | |
|     return brute_force;
 | |
| }
 | |
| 
 | |
| void infrared_brute_force_free(InfraredBruteForce* brute_force) {
 | |
|     furi_assert(!brute_force->is_started);
 | |
|     InfraredBruteForceRecordDict_clear(brute_force->records);
 | |
|     furi_string_free(brute_force->current_record_name);
 | |
|     free(brute_force);
 | |
| }
 | |
| 
 | |
| void infrared_brute_force_set_db_filename(InfraredBruteForce* brute_force, const char* db_filename) {
 | |
|     furi_assert(!brute_force->is_started);
 | |
|     brute_force->db_filename = db_filename;
 | |
| }
 | |
| 
 | |
| bool infrared_brute_force_calculate_messages(InfraredBruteForce* brute_force) {
 | |
|     furi_assert(!brute_force->is_started);
 | |
|     furi_assert(brute_force->db_filename);
 | |
|     bool success = false;
 | |
| 
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     FlipperFormat* ff = flipper_format_buffered_file_alloc(storage);
 | |
| 
 | |
|     success = flipper_format_buffered_file_open_existing(ff, brute_force->db_filename);
 | |
|     if(success) {
 | |
|         FuriString* signal_name;
 | |
|         signal_name = furi_string_alloc();
 | |
|         while(flipper_format_read_string(ff, "name", signal_name)) {
 | |
|             InfraredBruteForceRecord* record =
 | |
|                 InfraredBruteForceRecordDict_get(brute_force->records, signal_name);
 | |
|             if(record) { //-V547
 | |
|                 ++(record->count);
 | |
|             }
 | |
|         }
 | |
|         furi_string_free(signal_name);
 | |
|     }
 | |
| 
 | |
|     flipper_format_free(ff);
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_brute_force_start(
 | |
|     InfraredBruteForce* brute_force,
 | |
|     uint32_t index,
 | |
|     uint32_t* record_count) {
 | |
|     furi_assert(!brute_force->is_started);
 | |
|     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) {
 | |
|             *record_count = record->value.count;
 | |
|             if(*record_count) {
 | |
|                 furi_string_set(brute_force->current_record_name, record->key);
 | |
|             }
 | |
|             break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if(*record_count) {
 | |
|         Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|         brute_force->ff = flipper_format_buffered_file_alloc(storage);
 | |
|         brute_force->current_signal = infrared_signal_alloc();
 | |
|         brute_force->is_started = true;
 | |
|         success =
 | |
|             flipper_format_buffered_file_open_existing(brute_force->ff, brute_force->db_filename);
 | |
|         if(!success) infrared_brute_force_stop(brute_force);
 | |
|     }
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_brute_force_is_started(const InfraredBruteForce* brute_force) {
 | |
|     return brute_force->is_started;
 | |
| }
 | |
| 
 | |
| void infrared_brute_force_stop(InfraredBruteForce* brute_force) {
 | |
|     furi_assert(brute_force->is_started);
 | |
|     furi_string_reset(brute_force->current_record_name);
 | |
|     infrared_signal_free(brute_force->current_signal);
 | |
|     flipper_format_free(brute_force->ff);
 | |
|     brute_force->current_signal = NULL;
 | |
|     brute_force->ff = NULL;
 | |
|     brute_force->is_started = false;
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| }
 | |
| 
 | |
| bool infrared_brute_force_send_next(InfraredBruteForce* brute_force) {
 | |
|     furi_assert(brute_force->is_started);
 | |
|     const bool success = infrared_signal_search_by_name_and_read(
 | |
|         brute_force->current_signal,
 | |
|         brute_force->ff,
 | |
|         furi_string_get_cstr(brute_force->current_record_name));
 | |
|     if(success) {
 | |
|         infrared_signal_transmit(brute_force->current_signal);
 | |
|     }
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| void infrared_brute_force_add_record(
 | |
|     InfraredBruteForce* brute_force,
 | |
|     uint32_t index,
 | |
|     const char* name) {
 | |
|     InfraredBruteForceRecord value = {.index = index, .count = 0};
 | |
|     FuriString* key;
 | |
|     key = furi_string_alloc_set(name);
 | |
|     InfraredBruteForceRecordDict_set_at(brute_force->records, key, value);
 | |
|     furi_string_free(key);
 | |
| }
 | |
| 
 | |
| void infrared_brute_force_reset(InfraredBruteForce* brute_force) {
 | |
|     furi_assert(!brute_force->is_started);
 | |
|     InfraredBruteForceRecordDict_reset(brute_force->records);
 | |
| }
 |