picopass_scene_read_card_success.c 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. #include "../picopass_i.h"
  2. #include <dolphin/dolphin.h>
  3. #include <picopass_keys.h>
  4. #define TAG "PicopassSceneReadCardSuccess"
  5. void picopass_scene_read_card_success_widget_callback(
  6. GuiButtonType result,
  7. InputType type,
  8. void* context) {
  9. furi_assert(context);
  10. Picopass* picopass = context;
  11. if(type == InputTypeShort) {
  12. view_dispatcher_send_custom_event(picopass->view_dispatcher, result);
  13. }
  14. }
  15. void picopass_scene_read_card_success_on_enter(void* context) {
  16. Picopass* picopass = context;
  17. PicopassDeviceAuthMethod auth = picopass->dev->dev_data.auth;
  18. FuriString* csn_str = furi_string_alloc_set("CSN:");
  19. FuriString* credential_str = furi_string_alloc();
  20. FuriString* info_str = furi_string_alloc();
  21. FuriString* key_str = furi_string_alloc();
  22. dolphin_deed(DolphinDeedNfcReadSuccess);
  23. // Send notification
  24. notification_message(picopass->notifications, &sequence_success);
  25. // Setup view
  26. PicopassBlock* card_data = picopass->dev->dev_data.card_data;
  27. PicopassPacs* pacs = &picopass->dev->dev_data.pacs;
  28. Widget* widget = picopass->widget;
  29. uint8_t csn[PICOPASS_BLOCK_LEN] = {0};
  30. memcpy(csn, card_data[PICOPASS_CSN_BLOCK_INDEX].data, PICOPASS_BLOCK_LEN);
  31. for(uint8_t i = 0; i < PICOPASS_BLOCK_LEN; i++) {
  32. furi_string_cat_printf(csn_str, "%02X", csn[i]);
  33. }
  34. bool empty = picopass_is_memset(
  35. card_data[PICOPASS_ICLASS_PACS_CFG_BLOCK_INDEX].data, 0xFF, PICOPASS_BLOCK_LEN);
  36. bool SE = pacs->se_enabled;
  37. bool configCard = (card_data[PICOPASS_ICLASS_PACS_CFG_BLOCK_INDEX].data[7] >> 2 & 3) == 2;
  38. bool secured = (card_data[PICOPASS_CONFIG_BLOCK_INDEX].data[7] & PICOPASS_FUSE_CRYPT10) !=
  39. PICOPASS_FUSE_CRYPT0;
  40. bool hid_csn = picopass_device_hid_csn(picopass->dev);
  41. if(!secured) {
  42. furi_string_cat_printf(info_str, "Non-Secured Chip");
  43. if(!hid_csn) {
  44. furi_string_cat_printf(credential_str, "Non-HID CSN");
  45. }
  46. widget_add_button_element(
  47. widget,
  48. GuiButtonTypeRight,
  49. "More",
  50. picopass_scene_read_card_success_widget_callback,
  51. picopass);
  52. } else if(auth == PicopassDeviceAuthMethodFailed) {
  53. furi_string_cat_printf(info_str, "Read Failed");
  54. if(pacs->se_enabled) {
  55. furi_string_cat_printf(credential_str, "SE enabled");
  56. } else if(!hid_csn) {
  57. furi_string_cat_printf(credential_str, "Non-HID CSN");
  58. }
  59. widget_add_button_element(
  60. widget,
  61. GuiButtonTypeCenter,
  62. "Menu",
  63. picopass_scene_read_card_success_widget_callback,
  64. picopass);
  65. widget_add_button_element(
  66. widget,
  67. GuiButtonTypeRight,
  68. "More",
  69. picopass_scene_read_card_success_widget_callback,
  70. picopass);
  71. } else if(pacs->se_enabled) {
  72. furi_string_cat_printf(credential_str, "SE enabled");
  73. furi_string_cat_printf(info_str, "SIO");
  74. widget_add_button_element(
  75. widget,
  76. GuiButtonTypeRight,
  77. "More",
  78. picopass_scene_read_card_success_widget_callback,
  79. picopass);
  80. } else if(configCard) {
  81. furi_string_cat_printf(credential_str, "Config Card");
  82. } else if(empty) {
  83. furi_string_cat_printf(credential_str, "Empty");
  84. widget_add_button_element(
  85. widget,
  86. GuiButtonTypeCenter,
  87. "Menu",
  88. picopass_scene_read_card_success_widget_callback,
  89. picopass);
  90. } else if(pacs->bitLength == 0 || pacs->bitLength == 255) {
  91. // Neither of these are valid. Indicates the block was all 0x00 or all 0xff
  92. if(SE) {
  93. furi_string_cat_printf(info_str, "SIO");
  94. } else if(auth == PicopassDeviceAuthMethodFailed) {
  95. furi_string_cat_printf(info_str, "Auth Failed");
  96. } else {
  97. furi_string_cat_printf(info_str, "Invalid PACS");
  98. }
  99. widget_add_button_element(
  100. widget,
  101. GuiButtonTypeCenter,
  102. "Menu",
  103. picopass_scene_read_card_success_widget_callback,
  104. picopass);
  105. widget_add_button_element(
  106. widget,
  107. GuiButtonTypeRight,
  108. "More",
  109. picopass_scene_read_card_success_widget_callback,
  110. picopass);
  111. } else {
  112. size_t bytesLength = 1 + pacs->bitLength / 8;
  113. furi_string_set(credential_str, "");
  114. furi_string_cat_printf(credential_str, "(%d) ", pacs->bitLength);
  115. for(uint8_t i = PICOPASS_BLOCK_LEN - bytesLength; i < PICOPASS_BLOCK_LEN; i++) {
  116. furi_string_cat_printf(credential_str, "%02X", pacs->credential[i]);
  117. }
  118. if(pacs->sio) {
  119. furi_string_cat_printf(info_str, " +SIO");
  120. }
  121. widget_add_button_element(
  122. widget,
  123. GuiButtonTypeRight,
  124. "More",
  125. picopass_scene_read_card_success_widget_callback,
  126. picopass);
  127. }
  128. if(auth == PicopassDeviceAuthMethodUnset) {
  129. furi_string_cat_printf(key_str, "Error: Auth Unset");
  130. } else if(auth == PicopassDeviceAuthMethodNone) {
  131. furi_string_cat_printf(key_str, "Unsecure card");
  132. } else if(auth == PicopassDeviceAuthMethodNrMac) {
  133. furi_string_cat_printf(key_str, "No Key: used NR-MAC");
  134. } else if(auth == PicopassDeviceAuthMethodFailed) {
  135. furi_string_cat_printf(key_str, "Auth Failed");
  136. } else if(auth == PicopassDeviceAuthMethodKey) {
  137. furi_string_cat_printf(key_str, "Key: ");
  138. uint8_t key[PICOPASS_BLOCK_LEN];
  139. memcpy(key, &pacs->key, PICOPASS_BLOCK_LEN);
  140. bool standard_key = true;
  141. // Handle DES key being 56bits with parity in LSB
  142. for(uint8_t i = 0; i < PICOPASS_BLOCK_LEN; i++) {
  143. if((key[i] & 0xFE) != (picopass_iclass_key[i] & 0xFE)) {
  144. standard_key = false;
  145. break;
  146. }
  147. }
  148. if(standard_key) {
  149. furi_string_cat_printf(key_str, "Standard");
  150. } else {
  151. for(uint8_t i = 0; i < PICOPASS_BLOCK_LEN; i++) {
  152. furi_string_cat_printf(key_str, "%02X", key[i]);
  153. }
  154. }
  155. }
  156. widget_add_button_element(
  157. widget,
  158. GuiButtonTypeLeft,
  159. "Retry",
  160. picopass_scene_read_card_success_widget_callback,
  161. picopass);
  162. widget_add_string_element(
  163. widget, 64, 5, AlignCenter, AlignCenter, FontSecondary, furi_string_get_cstr(csn_str));
  164. widget_add_string_element(
  165. widget,
  166. 64,
  167. 20,
  168. AlignCenter,
  169. AlignCenter,
  170. FontPrimary,
  171. furi_string_get_cstr(credential_str));
  172. widget_add_string_element(
  173. widget, 64, 36, AlignCenter, AlignCenter, FontSecondary, furi_string_get_cstr(info_str));
  174. widget_add_string_element(
  175. widget, 64, 46, AlignCenter, AlignCenter, FontSecondary, furi_string_get_cstr(key_str));
  176. furi_string_free(csn_str);
  177. furi_string_free(credential_str);
  178. furi_string_free(info_str);
  179. furi_string_free(key_str);
  180. view_dispatcher_switch_to_view(picopass->view_dispatcher, PicopassViewWidget);
  181. }
  182. bool picopass_scene_read_card_success_on_event(void* context, SceneManagerEvent event) {
  183. Picopass* picopass = context;
  184. bool consumed = false;
  185. if(event.type == SceneManagerEventTypeCustom) {
  186. if(event.event == GuiButtonTypeLeft) {
  187. consumed = scene_manager_previous_scene(picopass->scene_manager);
  188. } else if(event.event == GuiButtonTypeRight) {
  189. // Clear device name
  190. picopass_device_set_name(picopass->dev, "");
  191. scene_manager_next_scene(picopass->scene_manager, PicopassSceneCardMenu);
  192. consumed = true;
  193. } else if(event.event == GuiButtonTypeCenter) {
  194. consumed = scene_manager_search_and_switch_to_another_scene(
  195. picopass->scene_manager, PicopassSceneStart);
  196. }
  197. } else if(event.type == SceneManagerEventTypeBack) {
  198. scene_manager_search_and_switch_to_previous_scene(
  199. picopass->scene_manager, PicopassSceneStart);
  200. consumed = true;
  201. }
  202. return consumed;
  203. }
  204. void picopass_scene_read_card_success_on_exit(void* context) {
  205. Picopass* picopass = context;
  206. // Clear view
  207. widget_reset(picopass->widget);
  208. }