lfrfid_app.cpp 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. #include "lfrfid_app.h"
  2. #include "assets_icons.h"
  3. #include <core/common_defines.h>
  4. #include "m-string.h"
  5. #include "scene/lfrfid_app_scene_start.h"
  6. #include "scene/lfrfid_app_scene_read.h"
  7. #include "scene/lfrfid_app_scene_read_success.h"
  8. #include "scene/lfrfid_app_scene_retry_confirm.h"
  9. #include "scene/lfrfid_app_scene_exit_confirm.h"
  10. #include "scene/lfrfid_app_scene_read_menu.h"
  11. #include "scene/lfrfid_app_scene_write.h"
  12. #include "scene/lfrfid_app_scene_write_success.h"
  13. #include "scene/lfrfid_app_scene_emulate.h"
  14. #include "scene/lfrfid_app_scene_save_name.h"
  15. #include "scene/lfrfid_app_scene_save_success.h"
  16. #include "scene/lfrfid_app_scene_select_key.h"
  17. #include "scene/lfrfid_app_scene_saved_key_menu.h"
  18. #include "scene/lfrfid_app_scene_save_data.h"
  19. #include "scene/lfrfid_app_scene_save_type.h"
  20. #include "scene/lfrfid_app_scene_saved_info.h"
  21. #include "scene/lfrfid_app_scene_delete_confirm.h"
  22. #include "scene/lfrfid_app_scene_delete_success.h"
  23. #include "scene/lfrfid_app_scene_rpc.h"
  24. #include <toolbox/path.h>
  25. #include <flipper_format/flipper_format.h>
  26. #include <rpc/rpc_app.h>
  27. const char* LfRfidApp::app_folder = ANY_PATH("lfrfid");
  28. const char* LfRfidApp::app_extension = ".rfid";
  29. const char* LfRfidApp::app_filetype = "Flipper RFID key";
  30. LfRfidApp::LfRfidApp()
  31. : scene_controller{this}
  32. , notification{"notification"}
  33. , storage{"storage"}
  34. , dialogs{"dialogs"}
  35. , text_store(40) {
  36. string_init_set_str(file_path, app_folder);
  37. }
  38. LfRfidApp::~LfRfidApp() {
  39. string_clear(file_path);
  40. if(rpc_ctx) {
  41. rpc_system_app_set_callback(rpc_ctx, NULL, NULL);
  42. rpc_system_app_send_exited(rpc_ctx);
  43. }
  44. }
  45. static void rpc_command_callback(RpcAppSystemEvent rpc_event, void* context) {
  46. furi_assert(context);
  47. LfRfidApp* app = static_cast<LfRfidApp*>(context);
  48. if(rpc_event == RpcAppEventSessionClose) {
  49. LfRfidApp::Event event;
  50. event.type = LfRfidApp::EventType::RpcSessionClose;
  51. app->view_controller.send_event(&event);
  52. // Detach RPC
  53. rpc_system_app_set_callback(app->rpc_ctx, NULL, NULL);
  54. app->rpc_ctx = NULL;
  55. } else if(rpc_event == RpcAppEventAppExit) {
  56. LfRfidApp::Event event;
  57. event.type = LfRfidApp::EventType::Exit;
  58. app->view_controller.send_event(&event);
  59. } else if(rpc_event == RpcAppEventLoadFile) {
  60. LfRfidApp::Event event;
  61. event.type = LfRfidApp::EventType::RpcLoadFile;
  62. app->view_controller.send_event(&event);
  63. } else {
  64. rpc_system_app_confirm(app->rpc_ctx, rpc_event, false);
  65. }
  66. }
  67. void LfRfidApp::run(void* _args) {
  68. const char* args = reinterpret_cast<const char*>(_args);
  69. make_app_folder();
  70. if(args && strlen(args)) {
  71. uint32_t rpc_ctx_ptr = 0;
  72. if(sscanf(args, "RPC %lX", &rpc_ctx_ptr) == 1) {
  73. rpc_ctx = (RpcAppSystem*)rpc_ctx_ptr;
  74. rpc_system_app_set_callback(rpc_ctx, rpc_command_callback, this);
  75. rpc_system_app_send_started(rpc_ctx);
  76. view_controller.attach_to_gui(ViewDispatcherTypeDesktop);
  77. scene_controller.add_scene(SceneType::Rpc, new LfRfidAppSceneRpc());
  78. scene_controller.process(100, SceneType::Rpc);
  79. } else {
  80. string_set_str(file_path, args);
  81. load_key_data(file_path, &worker.key, true);
  82. view_controller.attach_to_gui(ViewDispatcherTypeFullscreen);
  83. scene_controller.add_scene(SceneType::Emulate, new LfRfidAppSceneEmulate());
  84. scene_controller.process(100, SceneType::Emulate);
  85. }
  86. } else {
  87. view_controller.attach_to_gui(ViewDispatcherTypeFullscreen);
  88. scene_controller.add_scene(SceneType::Start, new LfRfidAppSceneStart());
  89. scene_controller.add_scene(SceneType::Read, new LfRfidAppSceneRead());
  90. scene_controller.add_scene(SceneType::RetryConfirm, new LfRfidAppSceneRetryConfirm());
  91. scene_controller.add_scene(SceneType::ExitConfirm, new LfRfidAppSceneExitConfirm());
  92. scene_controller.add_scene(SceneType::ReadSuccess, new LfRfidAppSceneReadSuccess());
  93. scene_controller.add_scene(SceneType::ReadKeyMenu, new LfRfidAppSceneReadKeyMenu());
  94. scene_controller.add_scene(SceneType::Write, new LfRfidAppSceneWrite());
  95. scene_controller.add_scene(SceneType::WriteSuccess, new LfRfidAppSceneWriteSuccess());
  96. scene_controller.add_scene(SceneType::Emulate, new LfRfidAppSceneEmulate());
  97. scene_controller.add_scene(SceneType::SaveName, new LfRfidAppSceneSaveName());
  98. scene_controller.add_scene(SceneType::SaveSuccess, new LfRfidAppSceneSaveSuccess());
  99. scene_controller.add_scene(SceneType::SelectKey, new LfRfidAppSceneSelectKey());
  100. scene_controller.add_scene(SceneType::SavedKeyMenu, new LfRfidAppSceneSavedKeyMenu());
  101. scene_controller.add_scene(SceneType::SaveData, new LfRfidAppSceneSaveData());
  102. scene_controller.add_scene(SceneType::SaveType, new LfRfidAppSceneSaveType());
  103. scene_controller.add_scene(SceneType::SavedInfo, new LfRfidAppSceneSavedInfo());
  104. scene_controller.add_scene(SceneType::DeleteConfirm, new LfRfidAppSceneDeleteConfirm());
  105. scene_controller.add_scene(SceneType::DeleteSuccess, new LfRfidAppSceneDeleteSuccess());
  106. scene_controller.process(100);
  107. }
  108. }
  109. bool LfRfidApp::save_key(RfidKey* key) {
  110. bool result = false;
  111. make_app_folder();
  112. if(string_end_with_str_p(file_path, app_extension)) {
  113. size_t filename_start = string_search_rchar(file_path, '/');
  114. string_left(file_path, filename_start);
  115. }
  116. string_cat_printf(file_path, "/%s%s", key->get_name(), app_extension);
  117. result = save_key_data(file_path, key);
  118. return result;
  119. }
  120. bool LfRfidApp::load_key_from_file_select(bool need_restore) {
  121. if(!need_restore) {
  122. string_set_str(file_path, app_folder);
  123. }
  124. bool result = dialog_file_browser_show(
  125. dialogs, file_path, file_path, app_extension, true, &I_125_10px, true);
  126. if(result) {
  127. result = load_key_data(file_path, &worker.key, true);
  128. }
  129. return result;
  130. }
  131. bool LfRfidApp::delete_key(RfidKey* key) {
  132. UNUSED(key);
  133. return storage_simply_remove(storage, string_get_cstr(file_path));
  134. }
  135. bool LfRfidApp::load_key_data(string_t path, RfidKey* key, bool show_dialog) {
  136. FlipperFormat* file = flipper_format_file_alloc(storage);
  137. bool result = false;
  138. string_t str_result;
  139. string_init(str_result);
  140. do {
  141. if(!flipper_format_file_open_existing(file, string_get_cstr(path))) break;
  142. // header
  143. uint32_t version;
  144. if(!flipper_format_read_header(file, str_result, &version)) break;
  145. if(string_cmp_str(str_result, app_filetype) != 0) break;
  146. if(version != 1) break;
  147. // key type
  148. LfrfidKeyType type;
  149. RfidKey loaded_key;
  150. if(!flipper_format_read_string(file, "Key type", str_result)) break;
  151. if(!lfrfid_key_get_string_type(string_get_cstr(str_result), &type)) break;
  152. loaded_key.set_type(type);
  153. // key data
  154. uint8_t key_data[loaded_key.get_type_data_count()] = {};
  155. if(!flipper_format_read_hex(file, "Data", key_data, loaded_key.get_type_data_count()))
  156. break;
  157. loaded_key.set_data(key_data, loaded_key.get_type_data_count());
  158. path_extract_filename(path, str_result, true);
  159. loaded_key.set_name(string_get_cstr(str_result));
  160. *key = loaded_key;
  161. result = true;
  162. } while(0);
  163. flipper_format_free(file);
  164. string_clear(str_result);
  165. if((!result) && (show_dialog)) {
  166. dialog_message_show_storage_error(dialogs, "Cannot load\nkey file");
  167. }
  168. return result;
  169. }
  170. bool LfRfidApp::save_key_data(string_t path, RfidKey* key) {
  171. FlipperFormat* file = flipper_format_file_alloc(storage);
  172. bool result = false;
  173. do {
  174. if(!flipper_format_file_open_always(file, string_get_cstr(path))) break;
  175. if(!flipper_format_write_header_cstr(file, app_filetype, 1)) break;
  176. if(!flipper_format_write_comment_cstr(file, "Key type can be EM4100, H10301 or I40134"))
  177. break;
  178. if(!flipper_format_write_string_cstr(
  179. file, "Key type", lfrfid_key_get_type_string(key->get_type())))
  180. break;
  181. if(!flipper_format_write_comment_cstr(
  182. file, "Data size for EM4100 is 5, for H10301 is 3, for I40134 is 3"))
  183. break;
  184. if(!flipper_format_write_hex(file, "Data", key->get_data(), key->get_type_data_count()))
  185. break;
  186. result = true;
  187. } while(0);
  188. flipper_format_free(file);
  189. if(!result) {
  190. dialog_message_show_storage_error(dialogs, "Cannot save\nkey file");
  191. }
  192. return result;
  193. }
  194. void LfRfidApp::make_app_folder() {
  195. if(!storage_simply_mkdir(storage, app_folder)) {
  196. dialog_message_show_storage_error(dialogs, "Cannot create\napp folder");
  197. }
  198. }