nfc_maker_scene_save_generate.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. #include "../nfc_maker.h"
  2. size_t nfc_maker_scene_save_generate_populate_ndef_buffer(NfcMaker* app) {
  3. // NDEF Docs: https://developer.nordicsemi.com/nRF_Connect_SDK/doc/latest/nrf/protocols/nfc/index.html#nfc-data-exchange-format-ndef
  4. uint8_t tnf = 0x00;
  5. const char* type = "";
  6. uint8_t* payload = NULL;
  7. uint8_t* payload_it = NULL;
  8. size_t payload_len = 0;
  9. size_t data_len = 0;
  10. switch(scene_manager_get_scene_state(app->scene_manager, NfcMakerSceneStart)) {
  11. case NfcMakerSceneBluetooth: {
  12. tnf = 0x02; // Media-type [RFC 2046]
  13. type = "application/vnd.bluetooth.ep.oob";
  14. data_len = MAC_INPUT_LEN;
  15. payload_len = data_len + 2;
  16. payload = payload_it = malloc(payload_len);
  17. *payload_it++ = 0x08;
  18. *payload_it++ = 0x00;
  19. memcpy(payload_it, app->mac_buf, data_len);
  20. payload_it += data_len;
  21. break;
  22. }
  23. case NfcMakerSceneContact: {
  24. tnf = 0x02; // Media-type [RFC 2046]
  25. type = "text/vcard";
  26. FuriString* vcard = furi_string_alloc_set("BEGIN:VCARD\r\nVERSION:3.0\r\n");
  27. furi_string_cat_printf(
  28. vcard, "PRODID:-//Flipper Xtreme//%s//EN\r\n", version_get_version(NULL));
  29. furi_string_cat_printf(vcard, "N:%s;%s;;;\r\n", app->small_buf2, app->small_buf1);
  30. furi_string_cat_printf(
  31. vcard,
  32. "FN:%s%s%s\r\n",
  33. app->small_buf1,
  34. strnlen(app->small_buf2, SMALL_INPUT_LEN) ? " " : "",
  35. app->small_buf2);
  36. if(strnlen(app->mail_buf, MAIL_INPUT_LEN)) {
  37. furi_string_cat_printf(vcard, "EMAIL:%s\r\n", app->mail_buf);
  38. }
  39. if(strnlen(app->phone_buf, PHONE_INPUT_LEN)) {
  40. furi_string_cat_printf(vcard, "TEL:%s\r\n", app->phone_buf);
  41. }
  42. if(strnlen(app->big_buf, BIG_INPUT_LEN)) {
  43. furi_string_cat_printf(vcard, "URL:%s\r\n", app->big_buf);
  44. }
  45. furi_string_cat_printf(vcard, "END:VCARD\r\n");
  46. payload_len = furi_string_size(vcard);
  47. payload = payload_it = malloc(payload_len);
  48. memcpy(payload_it, furi_string_get_cstr(vcard), payload_len);
  49. payload_it += payload_len;
  50. furi_string_free(vcard);
  51. break;
  52. }
  53. case NfcMakerSceneHttps: {
  54. tnf = 0x01; // NFC Forum well-known type [NFC RTD]
  55. type = "U";
  56. data_len = strnlen(app->big_buf, BIG_INPUT_LEN);
  57. payload_len = data_len + 1;
  58. payload = payload_it = malloc(payload_len);
  59. *payload_it++ = 0x04; // Prepend "https://"
  60. memcpy(payload_it, app->big_buf, data_len);
  61. payload_it += data_len;
  62. break;
  63. }
  64. case NfcMakerSceneMail: {
  65. tnf = 0x01; // NFC Forum well-known type [NFC RTD]
  66. type = "U";
  67. data_len = strnlen(app->mail_buf, MAIL_INPUT_LEN);
  68. payload_len = data_len + 1;
  69. payload = payload_it = malloc(payload_len);
  70. *payload_it++ = 0x06; // Prepend "mailto:"
  71. memcpy(payload_it, app->mail_buf, data_len);
  72. payload_it += data_len;
  73. break;
  74. }
  75. case NfcMakerScenePhone: {
  76. tnf = 0x01; // NFC Forum well-known type [NFC RTD]
  77. type = "U";
  78. data_len = strnlen(app->phone_buf, PHONE_INPUT_LEN);
  79. payload_len = data_len + 1;
  80. payload = payload_it = malloc(payload_len);
  81. *payload_it++ = 0x05; // Prepend "tel:"
  82. memcpy(payload_it, app->phone_buf, data_len);
  83. payload_it += data_len;
  84. break;
  85. }
  86. case NfcMakerSceneText: {
  87. tnf = 0x01; // NFC Forum well-known type [NFC RTD]
  88. type = "T";
  89. data_len = strnlen(app->big_buf, BIG_INPUT_LEN);
  90. payload_len = data_len + 3;
  91. payload = payload_it = malloc(payload_len);
  92. *payload_it++ = 0x02;
  93. *payload_it++ = 'e';
  94. *payload_it++ = 'n';
  95. memcpy(payload_it, app->big_buf, data_len);
  96. payload_it += data_len;
  97. break;
  98. }
  99. case NfcMakerSceneUrl: {
  100. tnf = 0x01; // NFC Forum well-known type [NFC RTD]
  101. type = "U";
  102. data_len = strnlen(app->big_buf, BIG_INPUT_LEN);
  103. payload_len = data_len + 1;
  104. payload = payload_it = malloc(payload_len);
  105. *payload_it++ = 0x00; // No prepend
  106. memcpy(payload_it, app->big_buf, data_len);
  107. payload_it += data_len;
  108. break;
  109. }
  110. case NfcMakerSceneWifi: {
  111. tnf = 0x02; // Media-type [RFC 2046]
  112. type = "application/vnd.wfa.wsc";
  113. uint8_t ssid_len = strnlen(app->small_buf1, SMALL_INPUT_LEN);
  114. uint8_t pass_len = strnlen(app->small_buf2, SMALL_INPUT_LEN);
  115. uint8_t data_len = ssid_len + pass_len;
  116. payload_len = data_len + 39;
  117. payload = payload_it = malloc(payload_len);
  118. *payload_it++ = 0x10;
  119. *payload_it++ = 0x0E;
  120. *payload_it++ = 0x00;
  121. *payload_it++ = data_len + 43;
  122. *payload_it++ = 0x10;
  123. *payload_it++ = 0x26;
  124. *payload_it++ = 0x00;
  125. *payload_it++ = 0x01;
  126. *payload_it++ = 0x01;
  127. *payload_it++ = 0x10;
  128. *payload_it++ = 0x45;
  129. *payload_it++ = 0x00;
  130. *payload_it++ = ssid_len;
  131. memcpy(payload_it, app->small_buf1, ssid_len);
  132. payload_it += ssid_len;
  133. *payload_it++ = 0x10;
  134. *payload_it++ = 0x03;
  135. *payload_it++ = 0x00;
  136. *payload_it++ = 0x02;
  137. *payload_it++ = 0x00;
  138. *payload_it++ = scene_manager_get_scene_state(app->scene_manager, NfcMakerSceneWifiAuth);
  139. *payload_it++ = 0x10;
  140. *payload_it++ = 0x0F;
  141. *payload_it++ = 0x00;
  142. *payload_it++ = 0x02;
  143. *payload_it++ = 0x00;
  144. *payload_it++ = scene_manager_get_scene_state(app->scene_manager, NfcMakerSceneWifiEncr);
  145. *payload_it++ = 0x10;
  146. *payload_it++ = 0x27;
  147. *payload_it++ = 0x00;
  148. *payload_it++ = pass_len;
  149. memcpy(payload_it, app->small_buf2, pass_len);
  150. payload_it += pass_len;
  151. *payload_it++ = 0x10;
  152. *payload_it++ = 0x20;
  153. *payload_it++ = 0x00;
  154. *payload_it++ = 0x06;
  155. *payload_it++ = 0xFF;
  156. *payload_it++ = 0xFF;
  157. *payload_it++ = 0xFF;
  158. *payload_it++ = 0xFF;
  159. *payload_it++ = 0xFF;
  160. *payload_it++ = 0xFF;
  161. break;
  162. }
  163. default:
  164. break;
  165. }
  166. // Setup header
  167. uint8_t flags = 0;
  168. flags |= 1 << 7; // MB (Message Begin)
  169. flags |= 1 << 6; // ME (Message End)
  170. flags |= tnf; // TNF (Type Name Format)
  171. size_t type_len = strlen(type);
  172. size_t header_len = 0;
  173. header_len += 1; // Flags and TNF
  174. header_len += 1; // Type length
  175. if(payload_len < 0xFF) {
  176. flags |= 1 << 4; // SR (Short Record)
  177. header_len += 1; // Payload length
  178. } else {
  179. header_len += 4; // Payload length
  180. }
  181. header_len += type_len; // Payload type
  182. // Start consolidating into NDEF buffer
  183. memset(app->ndef_buffer, 0, MAX_NDEF_LEN);
  184. uint8_t* buf = app->ndef_buffer;
  185. // NDEF TLV block
  186. *buf++ = 0x03; // TLV type
  187. size_t record_len = header_len + payload_len;
  188. if(record_len < 0xFF) {
  189. *buf++ = record_len; // TLV length
  190. } else {
  191. *buf++ = 0xFF; // TLV length
  192. *buf++ = record_len >> 8; // ...
  193. *buf++ = record_len & 0xFF; // ...
  194. }
  195. // Record header
  196. *buf++ = flags; // Flags and TNF
  197. *buf++ = type_len; // Type length
  198. if(flags & (1 << 4)) { // SR (Short Record)
  199. *buf++ = payload_len; // Payload length
  200. } else {
  201. *buf++ = payload_len >> 24 & 0xFF; // Payload length
  202. *buf++ = payload_len >> 16 & 0xFF; // ...
  203. *buf++ = payload_len >> 8 & 0xFF; // ...
  204. *buf++ = payload_len & 0xFF; // ...
  205. }
  206. memcpy(buf, type, type_len); // Payload type
  207. buf += type_len;
  208. // Record payload
  209. memcpy(buf, payload, payload_len);
  210. buf += payload_len;
  211. free(payload);
  212. // Record terminator
  213. *buf++ = 0xFE;
  214. return buf - app->ndef_buffer; // Size of NDEF data
  215. }
  216. void nfc_maker_scene_save_generate_submenu_callback(void* context, uint32_t index) {
  217. NfcMaker* app = context;
  218. view_dispatcher_send_custom_event(app->view_dispatcher, index);
  219. }
  220. void nfc_maker_scene_save_generate_on_enter(void* context) {
  221. NfcMaker* app = context;
  222. Submenu* submenu = app->submenu;
  223. size_t ndef_size = nfc_maker_scene_save_generate_populate_ndef_buffer(app);
  224. submenu_set_header(submenu, "NTAG Type:");
  225. for(Ntag ntag = 0; ntag < NtagMAX; ntag++) {
  226. submenu_add_lockable_item(
  227. submenu,
  228. ntag_names[ntag],
  229. ntag,
  230. nfc_maker_scene_save_generate_submenu_callback,
  231. app,
  232. ndef_size > ntag_sizes[ntag],
  233. "Data is\ntoo large!");
  234. }
  235. submenu_set_selected_item(
  236. submenu, scene_manager_get_scene_state(app->scene_manager, NfcMakerSceneSaveGenerate));
  237. view_dispatcher_switch_to_view(app->view_dispatcher, NfcMakerViewSubmenu);
  238. }
  239. bool nfc_maker_scene_save_generate_on_event(void* context, SceneManagerEvent event) {
  240. NfcMaker* app = context;
  241. bool consumed = false;
  242. if(event.type == SceneManagerEventTypeCustom) {
  243. scene_manager_set_scene_state(app->scene_manager, NfcMakerSceneSaveGenerate, event.event);
  244. if(event.event >= NtagMAX) return consumed;
  245. consumed = true;
  246. nfc_data_generator_fill_data(ntag_generators[event.event], app->nfc_device);
  247. MfUltralightData* data = mf_ultralight_alloc();
  248. nfc_device_copy_data(app->nfc_device, NfcProtocolMfUltralight, data);
  249. size_t size =
  250. MIN(ntag_sizes[event.event], // Known size
  251. data->page[3].data[2] * NTAG_DATA_AREA_UNIT_SIZE // Capability Container
  252. );
  253. memcpy(&data->page[4].data[0], app->ndef_buffer, size);
  254. nfc_device_set_data(app->nfc_device, NfcProtocolMfUltralight, data);
  255. mf_ultralight_free(data);
  256. scene_manager_next_scene(app->scene_manager, NfcMakerSceneSaveName);
  257. }
  258. return consumed;
  259. }
  260. void nfc_maker_scene_save_generate_on_exit(void* context) {
  261. NfcMaker* app = context;
  262. submenu_reset(app->submenu);
  263. }