 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>
		
			
				
	
	
		
			428 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			428 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "infrared_remote.h"
 | |
| 
 | |
| #include <m-array.h>
 | |
| 
 | |
| #include <toolbox/m_cstr_dup.h>
 | |
| #include <toolbox/path.h>
 | |
| #include <storage/storage.h>
 | |
| 
 | |
| #define TAG "InfraredRemote"
 | |
| 
 | |
| #define INFRARED_FILE_HEADER "IR signals file"
 | |
| #define INFRARED_FILE_VERSION (1)
 | |
| 
 | |
| ARRAY_DEF(StringArray, const char*, M_CSTR_DUP_OPLIST); //-V575
 | |
| 
 | |
| struct InfraredRemote {
 | |
|     StringArray_t signal_names;
 | |
|     FuriString* name;
 | |
|     FuriString* path;
 | |
| };
 | |
| 
 | |
| typedef struct {
 | |
|     InfraredRemote* remote;
 | |
|     FlipperFormat* ff_in;
 | |
|     FlipperFormat* ff_out;
 | |
|     FuriString* signal_name;
 | |
|     InfraredSignal* signal;
 | |
|     size_t signal_index;
 | |
| } InfraredBatch;
 | |
| 
 | |
| typedef struct {
 | |
|     size_t signal_index;
 | |
|     const char* signal_name;
 | |
|     const InfraredSignal* signal;
 | |
| } InfraredBatchTarget;
 | |
| 
 | |
| typedef bool (
 | |
|     *InfraredBatchCallback)(const InfraredBatch* batch, const InfraredBatchTarget* target);
 | |
| 
 | |
| InfraredRemote* infrared_remote_alloc() {
 | |
|     InfraredRemote* remote = malloc(sizeof(InfraredRemote));
 | |
|     StringArray_init(remote->signal_names);
 | |
|     remote->name = furi_string_alloc();
 | |
|     remote->path = furi_string_alloc();
 | |
|     return remote;
 | |
| }
 | |
| 
 | |
| void infrared_remote_free(InfraredRemote* remote) {
 | |
|     StringArray_clear(remote->signal_names);
 | |
|     furi_string_free(remote->path);
 | |
|     furi_string_free(remote->name);
 | |
|     free(remote);
 | |
| }
 | |
| 
 | |
| void infrared_remote_reset(InfraredRemote* remote) {
 | |
|     StringArray_reset(remote->signal_names);
 | |
|     furi_string_reset(remote->name);
 | |
|     furi_string_reset(remote->path);
 | |
| }
 | |
| 
 | |
| const char* infrared_remote_get_name(const InfraredRemote* remote) {
 | |
|     return furi_string_get_cstr(remote->name);
 | |
| }
 | |
| 
 | |
| static void infrared_remote_set_path(InfraredRemote* remote, const char* path) {
 | |
|     furi_string_set(remote->path, path);
 | |
|     path_extract_filename(remote->path, remote->name, true);
 | |
| }
 | |
| 
 | |
| const char* infrared_remote_get_path(const InfraredRemote* remote) {
 | |
|     return furi_string_get_cstr(remote->path);
 | |
| }
 | |
| 
 | |
| size_t infrared_remote_get_signal_count(const InfraredRemote* remote) {
 | |
|     return StringArray_size(remote->signal_names);
 | |
| }
 | |
| 
 | |
| const char* infrared_remote_get_signal_name(const InfraredRemote* remote, size_t index) {
 | |
|     furi_assert(index < infrared_remote_get_signal_count(remote));
 | |
|     return *StringArray_cget(remote->signal_names, index);
 | |
| }
 | |
| 
 | |
| bool infrared_remote_load_signal(
 | |
|     const InfraredRemote* remote,
 | |
|     InfraredSignal* signal,
 | |
|     size_t index) {
 | |
|     furi_assert(index < infrared_remote_get_signal_count(remote));
 | |
| 
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     FlipperFormat* ff = flipper_format_buffered_file_alloc(storage);
 | |
| 
 | |
|     bool success = false;
 | |
| 
 | |
|     do {
 | |
|         const char* path = furi_string_get_cstr(remote->path);
 | |
|         if(!flipper_format_buffered_file_open_existing(ff, path)) break;
 | |
| 
 | |
|         if(!infrared_signal_search_by_index_and_read(signal, ff, index)) {
 | |
|             const char* signal_name = infrared_remote_get_signal_name(remote, index);
 | |
|             FURI_LOG_E(TAG, "Failed to load signal '%s' from file '%s'", signal_name, path);
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         success = true;
 | |
|     } while(false);
 | |
| 
 | |
|     flipper_format_free(ff);
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_get_signal_index(
 | |
|     const InfraredRemote* remote,
 | |
|     const char* name,
 | |
|     size_t* index) {
 | |
|     uint32_t i = 0;
 | |
|     StringArray_it_t it;
 | |
| 
 | |
|     for(StringArray_it(it, remote->signal_names); !StringArray_end_p(it);
 | |
|         StringArray_next(it), ++i) {
 | |
|         if(strcmp(*StringArray_cref(it), name) == 0) {
 | |
|             *index = i;
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_append_signal(
 | |
|     InfraredRemote* remote,
 | |
|     const InfraredSignal* signal,
 | |
|     const char* name) {
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     FlipperFormat* ff = flipper_format_file_alloc(storage);
 | |
| 
 | |
|     bool success = false;
 | |
|     const char* path = furi_string_get_cstr(remote->path);
 | |
| 
 | |
|     do {
 | |
|         if(!flipper_format_file_open_append(ff, path)) break;
 | |
|         if(!infrared_signal_save(signal, ff, name)) break;
 | |
| 
 | |
|         StringArray_push_back(remote->signal_names, name);
 | |
|         success = true;
 | |
|     } while(false);
 | |
| 
 | |
|     flipper_format_free(ff);
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| static bool infrared_remote_batch_start(
 | |
|     InfraredRemote* remote,
 | |
|     InfraredBatchCallback batch_callback,
 | |
|     const InfraredBatchTarget* target) {
 | |
|     FuriString* tmp = furi_string_alloc();
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
| 
 | |
|     InfraredBatch batch_context = {
 | |
|         .remote = remote,
 | |
|         .ff_in = flipper_format_buffered_file_alloc(storage),
 | |
|         .ff_out = flipper_format_buffered_file_alloc(storage),
 | |
|         .signal_name = furi_string_alloc(),
 | |
|         .signal = infrared_signal_alloc(),
 | |
|         .signal_index = 0,
 | |
|     };
 | |
| 
 | |
|     const char* path_in = furi_string_get_cstr(remote->path);
 | |
|     const char* path_out;
 | |
| 
 | |
|     FS_Error status;
 | |
| 
 | |
|     do {
 | |
|         furi_string_printf(tmp, "%s.temp%08x.swp", path_in, rand());
 | |
|         path_out = furi_string_get_cstr(tmp);
 | |
|         status = storage_common_stat(storage, path_out, NULL);
 | |
|     } while(status == FSE_OK || status == FSE_EXIST);
 | |
| 
 | |
|     bool success = false;
 | |
| 
 | |
|     do {
 | |
|         if(!flipper_format_buffered_file_open_existing(batch_context.ff_in, path_in)) break;
 | |
|         if(!flipper_format_buffered_file_open_always(batch_context.ff_out, path_out)) break;
 | |
|         if(!flipper_format_write_header_cstr(
 | |
|                batch_context.ff_out, INFRARED_FILE_HEADER, INFRARED_FILE_VERSION))
 | |
|             break;
 | |
| 
 | |
|         const size_t signal_count = infrared_remote_get_signal_count(remote);
 | |
| 
 | |
|         for(; batch_context.signal_index < signal_count; ++batch_context.signal_index) {
 | |
|             if(!infrared_signal_read(
 | |
|                    batch_context.signal, batch_context.ff_in, batch_context.signal_name))
 | |
|                 break;
 | |
|             if(!batch_callback(&batch_context, target)) break;
 | |
|         }
 | |
| 
 | |
|         if(batch_context.signal_index != signal_count) break;
 | |
| 
 | |
|         if(!flipper_format_buffered_file_close(batch_context.ff_out)) break;
 | |
|         if(!flipper_format_buffered_file_close(batch_context.ff_in)) break;
 | |
| 
 | |
|         const FS_Error status = storage_common_rename(storage, path_out, path_in);
 | |
|         success = (status == FSE_OK || status == FSE_EXIST);
 | |
|     } while(false);
 | |
| 
 | |
|     infrared_signal_free(batch_context.signal);
 | |
|     furi_string_free(batch_context.signal_name);
 | |
|     flipper_format_free(batch_context.ff_out);
 | |
|     flipper_format_free(batch_context.ff_in);
 | |
|     furi_string_free(tmp);
 | |
| 
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| static bool infrared_remote_insert_signal_callback(
 | |
|     const InfraredBatch* batch,
 | |
|     const InfraredBatchTarget* target) {
 | |
|     // Insert a signal under the specified index
 | |
|     if(batch->signal_index == target->signal_index) {
 | |
|         if(!infrared_signal_save(target->signal, batch->ff_out, target->signal_name)) return false;
 | |
|         StringArray_push_at(
 | |
|             batch->remote->signal_names, target->signal_index, target->signal_name);
 | |
|     }
 | |
| 
 | |
|     // Write the rest normally
 | |
|     return infrared_signal_save(
 | |
|         batch->signal, batch->ff_out, furi_string_get_cstr(batch->signal_name));
 | |
| }
 | |
| 
 | |
| bool infrared_remote_insert_signal(
 | |
|     InfraredRemote* remote,
 | |
|     const InfraredSignal* signal,
 | |
|     const char* name,
 | |
|     size_t index) {
 | |
|     if(index >= infrared_remote_get_signal_count(remote)) {
 | |
|         return infrared_remote_append_signal(remote, signal, name);
 | |
|     }
 | |
| 
 | |
|     const InfraredBatchTarget insert_target = {
 | |
|         .signal_index = index,
 | |
|         .signal_name = name,
 | |
|         .signal = signal,
 | |
|     };
 | |
| 
 | |
|     return infrared_remote_batch_start(
 | |
|         remote, infrared_remote_insert_signal_callback, &insert_target);
 | |
| }
 | |
| 
 | |
| static bool infrared_remote_rename_signal_callback(
 | |
|     const InfraredBatch* batch,
 | |
|     const InfraredBatchTarget* target) {
 | |
|     const char* signal_name;
 | |
| 
 | |
|     if(batch->signal_index == target->signal_index) {
 | |
|         // Rename the signal at requested index
 | |
|         signal_name = target->signal_name;
 | |
|         StringArray_set_at(batch->remote->signal_names, batch->signal_index, signal_name);
 | |
|     } else {
 | |
|         // Use the original name otherwise
 | |
|         signal_name = furi_string_get_cstr(batch->signal_name);
 | |
|     }
 | |
| 
 | |
|     return infrared_signal_save(batch->signal, batch->ff_out, signal_name);
 | |
| }
 | |
| 
 | |
| bool infrared_remote_rename_signal(InfraredRemote* remote, size_t index, const char* new_name) {
 | |
|     furi_assert(index < infrared_remote_get_signal_count(remote));
 | |
| 
 | |
|     const InfraredBatchTarget rename_target = {
 | |
|         .signal_index = index,
 | |
|         .signal_name = new_name,
 | |
|         .signal = NULL,
 | |
|     };
 | |
| 
 | |
|     return infrared_remote_batch_start(
 | |
|         remote, infrared_remote_rename_signal_callback, &rename_target);
 | |
| }
 | |
| 
 | |
| static bool infrared_remote_delete_signal_callback(
 | |
|     const InfraredBatch* batch,
 | |
|     const InfraredBatchTarget* target) {
 | |
|     if(batch->signal_index == target->signal_index) {
 | |
|         // Do not save the signal to be deleted, remove it from the signal name list instead
 | |
|         StringArray_remove_v(
 | |
|             batch->remote->signal_names, batch->signal_index, batch->signal_index + 1);
 | |
|     } else {
 | |
|         // Pass other signals through
 | |
|         return infrared_signal_save(
 | |
|             batch->signal, batch->ff_out, furi_string_get_cstr(batch->signal_name));
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_delete_signal(InfraredRemote* remote, size_t index) {
 | |
|     furi_assert(index < infrared_remote_get_signal_count(remote));
 | |
| 
 | |
|     const InfraredBatchTarget delete_target = {
 | |
|         .signal_index = index,
 | |
|         .signal_name = NULL,
 | |
|         .signal = NULL,
 | |
|     };
 | |
| 
 | |
|     return infrared_remote_batch_start(
 | |
|         remote, infrared_remote_delete_signal_callback, &delete_target);
 | |
| }
 | |
| 
 | |
| bool infrared_remote_move_signal(InfraredRemote* remote, size_t index, size_t new_index) {
 | |
|     const size_t signal_count = infrared_remote_get_signal_count(remote);
 | |
|     furi_assert(index < signal_count);
 | |
|     furi_assert(new_index < signal_count);
 | |
| 
 | |
|     if(index == new_index) return true;
 | |
| 
 | |
|     InfraredSignal* signal = infrared_signal_alloc();
 | |
|     char* signal_name = strdup(infrared_remote_get_signal_name(remote, index));
 | |
| 
 | |
|     bool success = false;
 | |
| 
 | |
|     do {
 | |
|         if(!infrared_remote_load_signal(remote, signal, index)) break;
 | |
|         if(!infrared_remote_delete_signal(remote, index)) break;
 | |
|         if(!infrared_remote_insert_signal(remote, signal, signal_name, new_index)) break;
 | |
| 
 | |
|         success = true;
 | |
|     } while(false);
 | |
| 
 | |
|     free(signal_name);
 | |
|     infrared_signal_free(signal);
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_create(InfraredRemote* remote, const char* path) {
 | |
|     FURI_LOG_I(TAG, "Creating new file: '%s'", path);
 | |
| 
 | |
|     infrared_remote_reset(remote);
 | |
|     infrared_remote_set_path(remote, path);
 | |
| 
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     FlipperFormat* ff = flipper_format_file_alloc(storage);
 | |
| 
 | |
|     bool success = false;
 | |
| 
 | |
|     do {
 | |
|         if(!flipper_format_file_open_always(ff, path)) break;
 | |
|         if(!flipper_format_write_header_cstr(ff, INFRARED_FILE_HEADER, INFRARED_FILE_VERSION))
 | |
|             break;
 | |
| 
 | |
|         success = true;
 | |
|     } while(false);
 | |
| 
 | |
|     flipper_format_free(ff);
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_load(InfraredRemote* remote, const char* path) {
 | |
|     FURI_LOG_I(TAG, "Loading file: '%s'", path);
 | |
| 
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     FlipperFormat* ff = flipper_format_buffered_file_alloc(storage);
 | |
| 
 | |
|     FuriString* tmp = furi_string_alloc();
 | |
|     bool success = false;
 | |
| 
 | |
|     do {
 | |
|         if(!flipper_format_buffered_file_open_existing(ff, path)) break;
 | |
| 
 | |
|         uint32_t version;
 | |
|         if(!flipper_format_read_header(ff, tmp, &version)) break;
 | |
| 
 | |
|         if(!furi_string_equal(tmp, INFRARED_FILE_HEADER) || (version != INFRARED_FILE_VERSION))
 | |
|             break;
 | |
| 
 | |
|         infrared_remote_set_path(remote, path);
 | |
|         StringArray_reset(remote->signal_names);
 | |
| 
 | |
|         while(infrared_signal_read_name(ff, tmp)) {
 | |
|             StringArray_push_back(remote->signal_names, furi_string_get_cstr(tmp));
 | |
|         }
 | |
| 
 | |
|         success = true;
 | |
|     } while(false);
 | |
| 
 | |
|     furi_string_free(tmp);
 | |
|     flipper_format_free(ff);
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_rename(InfraredRemote* remote, const char* new_path) {
 | |
|     const char* old_path = infrared_remote_get_path(remote);
 | |
| 
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     const FS_Error status = storage_common_rename(storage, old_path, new_path);
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     const bool success = (status == FSE_OK || status == FSE_EXIST);
 | |
| 
 | |
|     if(success) {
 | |
|         infrared_remote_set_path(remote, new_path);
 | |
|     }
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| bool infrared_remote_remove(InfraredRemote* remote) {
 | |
|     Storage* storage = furi_record_open(RECORD_STORAGE);
 | |
|     const FS_Error status = storage_common_remove(storage, infrared_remote_get_path(remote));
 | |
|     furi_record_close(RECORD_STORAGE);
 | |
| 
 | |
|     const bool success = (status == FSE_OK || status == FSE_NOT_EXIST);
 | |
| 
 | |
|     if(success) {
 | |
|         infrared_remote_reset(remote);
 | |
|     }
 | |
| 
 | |
|     return success;
 | |
| }
 |