picopass_scene_elite_dict_attack.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. #include "../picopass_i.h"
  2. #include <dolphin/dolphin.h>
  3. #include "../picopass_keys.h"
  4. #define PICOPASS_SCENE_DICT_ATTACK_KEYS_BATCH_UPDATE (10)
  5. enum {
  6. PicopassSceneEliteDictAttackDictEliteUser,
  7. PicopassSceneEliteDictAttackDictStandard,
  8. PicopassSceneEliteDictAttackDictElite,
  9. };
  10. const char* picopass_dict_name[] = {
  11. [PicopassSceneEliteDictAttackDictEliteUser] = "Elite User Dictionary",
  12. [PicopassSceneEliteDictAttackDictStandard] = "Standard System Dictionary",
  13. [PicopassSceneEliteDictAttackDictElite] = "Elite System Dictionary",
  14. };
  15. static bool picopass_elite_dict_attack_change_dict(Picopass* picopass) {
  16. bool success = false;
  17. do {
  18. uint32_t scene_state =
  19. scene_manager_get_scene_state(picopass->scene_manager, PicopassSceneEliteDictAttack);
  20. keys_dict_free(picopass->dict);
  21. picopass->dict = NULL;
  22. if(scene_state == PicopassSceneEliteDictAttackDictElite) break;
  23. if(scene_state == PicopassSceneEliteDictAttackDictEliteUser) {
  24. if(!keys_dict_check_presence(PICOPASS_ICLASS_STANDARD_DICT_FLIPPER_NAME)) break;
  25. picopass->dict = keys_dict_alloc(
  26. PICOPASS_ICLASS_STANDARD_DICT_FLIPPER_NAME,
  27. KeysDictModeOpenExisting,
  28. PICOPASS_KEY_LEN);
  29. scene_state = PicopassSceneEliteDictAttackDictStandard;
  30. } else if(scene_state == PicopassSceneEliteDictAttackDictStandard) {
  31. if(!keys_dict_check_presence(PICOPASS_ICLASS_ELITE_DICT_FLIPPER_NAME)) break;
  32. picopass->dict = keys_dict_alloc(
  33. PICOPASS_ICLASS_ELITE_DICT_FLIPPER_NAME,
  34. KeysDictModeOpenExisting,
  35. PICOPASS_KEY_LEN);
  36. scene_state = PicopassSceneEliteDictAttackDictElite;
  37. }
  38. picopass->dict_attack_ctx.total_keys = keys_dict_get_total_keys(picopass->dict);
  39. picopass->dict_attack_ctx.current_key = 0;
  40. picopass->dict_attack_ctx.name = picopass_dict_name[scene_state];
  41. scene_manager_set_scene_state(
  42. picopass->scene_manager, PicopassSceneEliteDictAttack, scene_state);
  43. success = true;
  44. } while(false);
  45. return success;
  46. }
  47. NfcCommand picopass_elite_dict_attack_worker_callback(PicopassPollerEvent event, void* context) {
  48. furi_assert(context);
  49. NfcCommand command = NfcCommandContinue;
  50. Picopass* picopass = context;
  51. if(event.type == PicopassPollerEventTypeRequestMode) {
  52. event.data->req_mode.mode = PicopassPollerModeRead;
  53. } else if(event.type == PicopassPollerEventTypeRequestKey) {
  54. uint8_t key[PICOPASS_KEY_LEN] = {};
  55. bool is_key_provided = true;
  56. if(!keys_dict_get_next_key(picopass->dict, key, PICOPASS_KEY_LEN)) {
  57. if(picopass_elite_dict_attack_change_dict(picopass)) {
  58. is_key_provided = keys_dict_get_next_key(picopass->dict, key, PICOPASS_KEY_LEN);
  59. view_dispatcher_send_custom_event(
  60. picopass->view_dispatcher, PicopassCustomEventDictAttackUpdateView);
  61. } else {
  62. is_key_provided = false;
  63. }
  64. }
  65. uint32_t scene_state =
  66. scene_manager_get_scene_state(picopass->scene_manager, PicopassSceneEliteDictAttack);
  67. memcpy(event.data->req_key.key, key, PICOPASS_KEY_LEN);
  68. event.data->req_key.is_elite_key =
  69. (scene_state != PicopassSceneEliteDictAttackDictStandard);
  70. event.data->req_key.is_key_provided = is_key_provided;
  71. if(is_key_provided) {
  72. picopass->dict_attack_ctx.current_key++;
  73. if(picopass->dict_attack_ctx.current_key %
  74. PICOPASS_SCENE_DICT_ATTACK_KEYS_BATCH_UPDATE ==
  75. 0) {
  76. view_dispatcher_send_custom_event(
  77. picopass->view_dispatcher, PicopassCustomEventDictAttackUpdateView);
  78. }
  79. }
  80. } else if(
  81. event.type == PicopassPollerEventTypeSuccess ||
  82. event.type == PicopassPollerEventTypeFail ||
  83. event.type == PicopassPollerEventTypeAuthFail) {
  84. const PicopassDeviceData* data = picopass_poller_get_data(picopass->poller);
  85. memcpy(&picopass->dev->dev_data, data, sizeof(PicopassDeviceData));
  86. view_dispatcher_send_custom_event(
  87. picopass->view_dispatcher, PicopassCustomEventPollerSuccess);
  88. } else if(event.type == PicopassPollerEventTypeCardLost) {
  89. picopass->dict_attack_ctx.card_detected = false;
  90. view_dispatcher_send_custom_event(
  91. picopass->view_dispatcher, PicopassCustomEventDictAttackUpdateView);
  92. } else if(event.type == PicopassPollerEventTypeCardDetected) {
  93. picopass->dict_attack_ctx.card_detected = true;
  94. view_dispatcher_send_custom_event(
  95. picopass->view_dispatcher, PicopassCustomEventDictAttackUpdateView);
  96. }
  97. return command;
  98. }
  99. static void picopass_scene_elite_dict_attack_update_view(Picopass* instance) {
  100. if(instance->dict_attack_ctx.card_detected) {
  101. dict_attack_set_card_detected(instance->dict_attack);
  102. dict_attack_set_header(instance->dict_attack, instance->dict_attack_ctx.name);
  103. dict_attack_set_total_dict_keys(
  104. instance->dict_attack, instance->dict_attack_ctx.total_keys);
  105. dict_attack_set_current_dict_key(
  106. instance->dict_attack, instance->dict_attack_ctx.current_key);
  107. } else {
  108. dict_attack_set_card_removed(instance->dict_attack);
  109. }
  110. }
  111. static void picopass_scene_elite_dict_attack_callback(void* context) {
  112. Picopass* instance = context;
  113. view_dispatcher_send_custom_event(
  114. instance->view_dispatcher, PicopassCustomEventDictAttackSkip);
  115. }
  116. void picopass_scene_elite_dict_attack_on_enter(void* context) {
  117. Picopass* picopass = context;
  118. dolphin_deed(DolphinDeedNfcRead);
  119. // Setup dict attack context
  120. uint32_t state = PicopassSceneEliteDictAttackDictEliteUser;
  121. bool use_user_dict = keys_dict_check_presence(PICOPASS_ICLASS_ELITE_DICT_USER_NAME);
  122. if(use_user_dict) {
  123. picopass->dict = keys_dict_alloc(
  124. PICOPASS_ICLASS_ELITE_DICT_USER_NAME, KeysDictModeOpenExisting, PICOPASS_KEY_LEN);
  125. if(keys_dict_get_total_keys(picopass->dict) == 0) {
  126. keys_dict_free(picopass->dict);
  127. use_user_dict = false;
  128. }
  129. }
  130. if(use_user_dict) {
  131. state = PicopassSceneEliteDictAttackDictEliteUser;
  132. } else {
  133. picopass->dict = keys_dict_alloc(
  134. PICOPASS_ICLASS_STANDARD_DICT_FLIPPER_NAME,
  135. KeysDictModeOpenExisting,
  136. PICOPASS_KEY_LEN);
  137. state = PicopassSceneEliteDictAttackDictStandard;
  138. }
  139. dict_attack_reset(picopass->dict_attack);
  140. picopass->dict_attack_ctx.card_detected = false;
  141. picopass->dict_attack_ctx.total_keys = keys_dict_get_total_keys(picopass->dict);
  142. picopass->dict_attack_ctx.current_key = 0;
  143. picopass->dict_attack_ctx.name = picopass_dict_name[state];
  144. scene_manager_set_scene_state(picopass->scene_manager, PicopassSceneEliteDictAttack, state);
  145. // Setup view
  146. picopass_scene_elite_dict_attack_update_view(picopass);
  147. dict_attack_set_callback(
  148. picopass->dict_attack, picopass_scene_elite_dict_attack_callback, picopass);
  149. // Start worker
  150. picopass->poller = picopass_poller_alloc(picopass->nfc);
  151. picopass_poller_start(picopass->poller, picopass_elite_dict_attack_worker_callback, picopass);
  152. view_dispatcher_switch_to_view(picopass->view_dispatcher, PicopassViewDictAttack);
  153. picopass_blink_start(picopass);
  154. }
  155. bool picopass_scene_elite_dict_attack_on_event(void* context, SceneManagerEvent event) {
  156. Picopass* picopass = context;
  157. bool consumed = false;
  158. if(event.type == SceneManagerEventTypeCustom) {
  159. if(event.event == PicopassCustomEventPollerSuccess) {
  160. if(memcmp(
  161. picopass->dev->dev_data.pacs.key,
  162. picopass_factory_debit_key,
  163. PICOPASS_BLOCK_LEN) == 0) {
  164. scene_manager_next_scene(picopass->scene_manager, PicopassSceneReadFactorySuccess);
  165. } else {
  166. scene_manager_next_scene(picopass->scene_manager, PicopassSceneReadCardSuccess);
  167. }
  168. consumed = true;
  169. } else if(event.event == PicopassCustomEventDictAttackUpdateView) {
  170. picopass_scene_elite_dict_attack_update_view(picopass);
  171. consumed = true;
  172. } else if(event.event == PicopassCustomEventDictAttackSkip) {
  173. uint32_t scene_state = scene_manager_get_scene_state(
  174. picopass->scene_manager, PicopassSceneEliteDictAttack);
  175. if(scene_state != PicopassSceneEliteDictAttackDictElite) {
  176. picopass_elite_dict_attack_change_dict(picopass);
  177. picopass_scene_elite_dict_attack_update_view(picopass);
  178. } else {
  179. if(memcmp(
  180. picopass->dev->dev_data.pacs.key,
  181. picopass_factory_debit_key,
  182. PICOPASS_BLOCK_LEN) == 0) {
  183. scene_manager_next_scene(
  184. picopass->scene_manager, PicopassSceneReadFactorySuccess);
  185. } else {
  186. scene_manager_next_scene(
  187. picopass->scene_manager, PicopassSceneReadCardSuccess);
  188. }
  189. }
  190. consumed = true;
  191. }
  192. }
  193. return consumed;
  194. }
  195. void picopass_scene_elite_dict_attack_on_exit(void* context) {
  196. Picopass* picopass = context;
  197. if(picopass->dict) {
  198. keys_dict_free(picopass->dict);
  199. picopass->dict = NULL;
  200. }
  201. picopass->dict_attack_ctx.current_key = 0;
  202. picopass->dict_attack_ctx.total_keys = 0;
  203. picopass_poller_stop(picopass->poller);
  204. picopass_poller_free(picopass->poller);
  205. // Clear view
  206. popup_reset(picopass->popup);
  207. scene_manager_set_scene_state(
  208. picopass->scene_manager,
  209. PicopassSceneEliteDictAttack,
  210. PicopassSceneEliteDictAttackDictEliteUser);
  211. picopass_blink_stop(picopass);
  212. }