picopass_scene_read_card_success.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. #include "../picopass_i.h"
  2. #include <dolphin/dolphin.h>
  3. void picopass_scene_read_card_success_widget_callback(
  4. GuiButtonType result,
  5. InputType type,
  6. void* context) {
  7. furi_assert(context);
  8. Picopass* picopass = context;
  9. if(type == InputTypeShort) {
  10. view_dispatcher_send_custom_event(picopass->view_dispatcher, result);
  11. }
  12. }
  13. void picopass_scene_read_card_success_on_enter(void* context) {
  14. Picopass* picopass = context;
  15. FuriString* csn_str = furi_string_alloc_set("CSN:");
  16. FuriString* credential_str = furi_string_alloc();
  17. FuriString* wiegand_str = furi_string_alloc();
  18. FuriString* key_str = furi_string_alloc();
  19. dolphin_deed(DolphinDeedNfcReadSuccess);
  20. // Send notification
  21. notification_message(picopass->notifications, &sequence_success);
  22. // Setup view
  23. PicopassBlock* AA1 = picopass->dev->dev_data.AA1;
  24. PicopassPacs* pacs = &picopass->dev->dev_data.pacs;
  25. Widget* widget = picopass->widget;
  26. uint8_t csn[RFAL_PICOPASS_BLOCK_LEN] = {0};
  27. memcpy(csn, AA1[PICOPASS_CSN_BLOCK_INDEX].data, RFAL_PICOPASS_BLOCK_LEN);
  28. for(uint8_t i = 0; i < RFAL_PICOPASS_BLOCK_LEN; i++) {
  29. furi_string_cat_printf(csn_str, "%02X", csn[i]);
  30. }
  31. bool no_key = picopass_is_memset(pacs->key, 0x00, RFAL_PICOPASS_BLOCK_LEN);
  32. bool empty = picopass_is_memset(
  33. AA1[PICOPASS_ICLASS_PACS_CFG_BLOCK_INDEX].data, 0xFF, RFAL_PICOPASS_BLOCK_LEN);
  34. if(no_key) {
  35. furi_string_cat_printf(wiegand_str, "Read Failed");
  36. if(pacs->se_enabled) {
  37. furi_string_cat_printf(credential_str, "SE enabled");
  38. }
  39. widget_add_button_element(
  40. widget,
  41. GuiButtonTypeCenter,
  42. "Menu",
  43. picopass_scene_read_card_success_widget_callback,
  44. picopass);
  45. } else if(empty) {
  46. furi_string_cat_printf(wiegand_str, "Empty");
  47. widget_add_button_element(
  48. widget,
  49. GuiButtonTypeCenter,
  50. "Menu",
  51. picopass_scene_read_card_success_widget_callback,
  52. picopass);
  53. } else if(pacs->record.bitLength == 0 || pacs->record.bitLength == 255) {
  54. // Neither of these are valid. Indicates the block was all 0x00 or all 0xff
  55. furi_string_cat_printf(wiegand_str, "Invalid PACS");
  56. if(pacs->se_enabled) {
  57. furi_string_cat_printf(credential_str, "SE enabled");
  58. }
  59. widget_add_button_element(
  60. widget,
  61. GuiButtonTypeCenter,
  62. "Menu",
  63. picopass_scene_read_card_success_widget_callback,
  64. picopass);
  65. } else {
  66. size_t bytesLength = 1 + pacs->record.bitLength / 8;
  67. furi_string_set(credential_str, "");
  68. for(uint8_t i = RFAL_PICOPASS_BLOCK_LEN - bytesLength; i < RFAL_PICOPASS_BLOCK_LEN; i++) {
  69. furi_string_cat_printf(credential_str, "%02X", pacs->credential[i]);
  70. }
  71. if(pacs->record.valid) {
  72. furi_string_cat_printf(
  73. wiegand_str, "FC: %u CN: %u", pacs->record.FacilityCode, pacs->record.CardNumber);
  74. } else {
  75. furi_string_cat_printf(wiegand_str, "%d bits", pacs->record.bitLength);
  76. }
  77. if(pacs->sio) {
  78. furi_string_cat_printf(credential_str, " +SIO");
  79. }
  80. if(pacs->key) {
  81. furi_string_cat_printf(key_str, "Key: ");
  82. uint8_t key[RFAL_PICOPASS_BLOCK_LEN];
  83. memcpy(key, &pacs->key, RFAL_PICOPASS_BLOCK_LEN);
  84. bool standard_key = true;
  85. // Handle DES key being 56bits with parity in LSB
  86. for(uint8_t i = 0; i < RFAL_PICOPASS_BLOCK_LEN; i++) {
  87. if((key[i] & 0xFE) != (picopass_iclass_key[i] & 0xFE)) {
  88. standard_key = false;
  89. break;
  90. }
  91. }
  92. if(standard_key) {
  93. furi_string_cat_printf(key_str, "Standard");
  94. } else {
  95. for(uint8_t i = 0; i < RFAL_PICOPASS_BLOCK_LEN; i++) {
  96. furi_string_cat_printf(key_str, "%02X", key[i]);
  97. }
  98. }
  99. }
  100. widget_add_button_element(
  101. widget,
  102. GuiButtonTypeRight,
  103. "More",
  104. picopass_scene_read_card_success_widget_callback,
  105. picopass);
  106. }
  107. widget_add_button_element(
  108. widget,
  109. GuiButtonTypeLeft,
  110. "Retry",
  111. picopass_scene_read_card_success_widget_callback,
  112. picopass);
  113. widget_add_string_element(
  114. widget, 64, 5, AlignCenter, AlignCenter, FontSecondary, furi_string_get_cstr(csn_str));
  115. widget_add_string_element(
  116. widget, 64, 20, AlignCenter, AlignCenter, FontPrimary, furi_string_get_cstr(wiegand_str));
  117. widget_add_string_element(
  118. widget,
  119. 64,
  120. 36,
  121. AlignCenter,
  122. AlignCenter,
  123. FontSecondary,
  124. furi_string_get_cstr(credential_str));
  125. widget_add_string_element(
  126. widget, 64, 46, AlignCenter, AlignCenter, FontSecondary, furi_string_get_cstr(key_str));
  127. furi_string_free(csn_str);
  128. furi_string_free(credential_str);
  129. furi_string_free(wiegand_str);
  130. furi_string_free(key_str);
  131. view_dispatcher_switch_to_view(picopass->view_dispatcher, PicopassViewWidget);
  132. }
  133. bool picopass_scene_read_card_success_on_event(void* context, SceneManagerEvent event) {
  134. Picopass* picopass = context;
  135. bool consumed = false;
  136. if(event.type == SceneManagerEventTypeCustom) {
  137. if(event.event == GuiButtonTypeLeft) {
  138. consumed = scene_manager_previous_scene(picopass->scene_manager);
  139. } else if(event.event == GuiButtonTypeRight) {
  140. // Clear device name
  141. picopass_device_set_name(picopass->dev, "");
  142. scene_manager_next_scene(picopass->scene_manager, PicopassSceneCardMenu);
  143. consumed = true;
  144. } else if(event.event == GuiButtonTypeCenter) {
  145. consumed = scene_manager_search_and_switch_to_another_scene(
  146. picopass->scene_manager, PicopassSceneStart);
  147. }
  148. }
  149. return consumed;
  150. }
  151. void picopass_scene_read_card_success_on_exit(void* context) {
  152. Picopass* picopass = context;
  153. // Clear view
  154. widget_reset(picopass->widget);
  155. }