mifare_ultralight.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. #include "mifare_ultralight.h"
  2. bool mf_ul_check_card_type(uint8_t ATQA0, uint8_t ATQA1, uint8_t SAK) {
  3. if((ATQA0 == 0x44) && (ATQA1 == 0x00) && (SAK == 0x00)) {
  4. return true;
  5. }
  6. return false;
  7. }
  8. uint16_t mf_ul_prepare_get_version(uint8_t* dest) {
  9. dest[0] = MF_UL_GET_VERSION_CMD;
  10. return 1;
  11. }
  12. void mf_ul_parse_get_version_response(uint8_t* buff, MifareUlDevice* mf_ul_read) {
  13. MfUltralightVersion* version = (MfUltralightVersion*)buff;
  14. memcpy(&mf_ul_read->data.version, version, sizeof(MfUltralightVersion));
  15. if(version->storage_size == 0x0B || version->storage_size == 0x00) {
  16. mf_ul_read->data.type = MfUltralightTypeUL11;
  17. mf_ul_read->pages_to_read = 20;
  18. mf_ul_read->support_fast_read = true;
  19. } else if(version->storage_size == 0x0E) {
  20. mf_ul_read->data.type = MfUltralightTypeUL21;
  21. mf_ul_read->pages_to_read = 41;
  22. mf_ul_read->support_fast_read = true;
  23. } else if(version->storage_size == 0x0F) {
  24. mf_ul_read->data.type = MfUltralightTypeNTAG213;
  25. mf_ul_read->pages_to_read = 45;
  26. mf_ul_read->support_fast_read = false;
  27. } else if(version->storage_size == 0x11) {
  28. mf_ul_read->data.type = MfUltralightTypeNTAG215;
  29. mf_ul_read->pages_to_read = 135;
  30. mf_ul_read->support_fast_read = false;
  31. } else if(version->storage_size == 0x13) {
  32. mf_ul_read->data.type = MfUltralightTypeNTAG216;
  33. mf_ul_read->pages_to_read = 231;
  34. mf_ul_read->support_fast_read = false;
  35. } else {
  36. mf_ul_set_default_version(mf_ul_read);
  37. }
  38. }
  39. void mf_ul_set_default_version(MifareUlDevice* mf_ul_read) {
  40. mf_ul_read->data.type = MfUltralightTypeUnknown;
  41. mf_ul_read->pages_to_read = 16;
  42. mf_ul_read->support_fast_read = false;
  43. }
  44. uint16_t mf_ul_prepare_read(uint8_t* dest, uint8_t start_page) {
  45. dest[0] = MF_UL_READ_CMD;
  46. dest[1] = start_page;
  47. return 2;
  48. }
  49. void mf_ul_parse_read_response(uint8_t* buff, uint16_t page_addr, MifareUlDevice* mf_ul_read) {
  50. uint8_t pages_read = 4;
  51. uint8_t page_read_count = mf_ul_read->pages_readed + pages_read;
  52. if(page_read_count > mf_ul_read->pages_to_read) {
  53. pages_read -= page_read_count - mf_ul_read->pages_to_read;
  54. }
  55. mf_ul_read->pages_readed += pages_read;
  56. mf_ul_read->data.data_size = mf_ul_read->pages_readed * 4;
  57. memcpy(&mf_ul_read->data.data[page_addr * 4], buff, pages_read * 4);
  58. }
  59. uint16_t mf_ul_prepare_fast_read(uint8_t* dest, uint8_t start_page, uint8_t end_page) {
  60. dest[0] = MF_UL_FAST_READ_CMD;
  61. dest[1] = start_page;
  62. dest[2] = end_page;
  63. return 3;
  64. }
  65. void mf_ul_parse_fast_read_response(
  66. uint8_t* buff,
  67. uint8_t start_page,
  68. uint8_t end_page,
  69. MifareUlDevice* mf_ul_read) {
  70. mf_ul_read->pages_readed = end_page - start_page + 1;
  71. mf_ul_read->data.data_size = mf_ul_read->pages_readed * 4;
  72. memcpy(mf_ul_read->data.data, buff, mf_ul_read->data.data_size);
  73. }
  74. uint16_t mf_ul_prepare_read_signature(uint8_t* dest) {
  75. dest[0] = MF_UL_READ_SIG;
  76. dest[1] = 0;
  77. return 2;
  78. }
  79. void mf_ul_parse_read_signature_response(uint8_t* buff, MifareUlDevice* mf_ul_read) {
  80. memcpy(mf_ul_read->data.signature, buff, sizeof(mf_ul_read->data.signature));
  81. }
  82. uint16_t mf_ul_prepare_read_cnt(uint8_t* dest, uint8_t cnt_index) {
  83. if(cnt_index > 2) {
  84. return 0;
  85. }
  86. dest[0] = MF_UL_READ_CNT;
  87. dest[1] = cnt_index;
  88. return 2;
  89. }
  90. void mf_ul_parse_read_cnt_response(uint8_t* buff, uint8_t cnt_index, MifareUlDevice* mf_ul_read) {
  91. // Reverse LSB sequence
  92. if(cnt_index < 3) {
  93. mf_ul_read->data.counter[cnt_index] = (buff[2] << 16) | (buff[1] << 8) | (buff[0]);
  94. }
  95. }
  96. uint16_t mf_ul_prepare_inc_cnt(uint8_t* dest, uint8_t cnt_index, uint32_t value) {
  97. if(cnt_index > 2) {
  98. return 0;
  99. }
  100. dest[0] = MF_UL_INC_CNT;
  101. dest[1] = cnt_index;
  102. dest[2] = (uint8_t)value;
  103. dest[3] = (uint8_t)(value >> 8);
  104. dest[4] = (uint8_t)(value >> 16);
  105. dest[5] = 0;
  106. return 6;
  107. }
  108. uint16_t mf_ul_prepare_check_tearing(uint8_t* dest, uint8_t cnt_index) {
  109. if(cnt_index > 2) {
  110. return 0;
  111. }
  112. dest[0] = MF_UL_CHECK_TEARING;
  113. dest[1] = cnt_index;
  114. return 2;
  115. }
  116. void mf_ul_parse_check_tearing_response(
  117. uint8_t* buff,
  118. uint8_t cnt_index,
  119. MifareUlDevice* mf_ul_read) {
  120. if(cnt_index < 2) {
  121. mf_ul_read->data.tearing[cnt_index] = buff[0];
  122. }
  123. }
  124. uint16_t mf_ul_prepare_write(uint8_t* dest, uint16_t page_addr, uint32_t data) {
  125. if(page_addr < 2) {
  126. return 0;
  127. }
  128. dest[0] = MF_UL_WRITE;
  129. dest[1] = page_addr;
  130. dest[2] = (uint8_t)(data >> 24);
  131. dest[3] = (uint8_t)(data >> 16);
  132. dest[4] = (uint8_t)(data >> 8);
  133. dest[5] = (uint8_t)data;
  134. return 6;
  135. }
  136. void mf_ul_prepare_emulation(MifareUlDevice* mf_ul_emulate, MifareUlData* data) {
  137. mf_ul_emulate->data = *data;
  138. mf_ul_emulate->auth_data = NULL;
  139. mf_ul_emulate->data_changed = false;
  140. if(data->version.storage_size == 0) {
  141. mf_ul_emulate->data.type = MfUltralightTypeUnknown;
  142. mf_ul_emulate->support_fast_read = false;
  143. } else if(data->version.storage_size == 0x0B) {
  144. mf_ul_emulate->data.type = MfUltralightTypeUL11;
  145. mf_ul_emulate->support_fast_read = true;
  146. } else if(data->version.storage_size == 0x0E) {
  147. mf_ul_emulate->data.type = MfUltralightTypeUL21;
  148. mf_ul_emulate->support_fast_read = true;
  149. } else if(data->version.storage_size == 0x0F) {
  150. mf_ul_emulate->data.type = MfUltralightTypeNTAG213;
  151. mf_ul_emulate->support_fast_read = true;
  152. } else if(data->version.storage_size == 0x11) {
  153. mf_ul_emulate->data.type = MfUltralightTypeNTAG215;
  154. mf_ul_emulate->support_fast_read = true;
  155. } else if(data->version.storage_size == 0x13) {
  156. mf_ul_emulate->data.type = MfUltralightTypeNTAG216;
  157. mf_ul_emulate->support_fast_read = true;
  158. }
  159. if(mf_ul_emulate->data.type >= MfUltralightTypeNTAG213) {
  160. uint16_t pwd_page = (data->data_size / 4) - 2;
  161. mf_ul_emulate->auth_data = (MifareUlAuthData*)&data->data[pwd_page * 4];
  162. }
  163. }
  164. void mf_ul_protect_auth_data_on_read_command(
  165. uint8_t* tx_buff,
  166. uint8_t start_page,
  167. uint8_t end_page,
  168. MifareUlDevice* mf_ul_emulate) {
  169. if(mf_ul_emulate->data.type >= MfUltralightTypeNTAG213) {
  170. uint8_t pwd_page = (mf_ul_emulate->data.data_size / 4) - 2;
  171. uint8_t pack_page = pwd_page + 1;
  172. if((start_page <= pwd_page) && (end_page >= pwd_page)) {
  173. memset(&tx_buff[(pwd_page - start_page) * 4], 0, 4);
  174. }
  175. if((start_page <= pack_page) && (end_page >= pack_page)) {
  176. memset(&tx_buff[(pack_page - start_page) * 4], 0, 2);
  177. }
  178. }
  179. }
  180. uint16_t mf_ul_prepare_emulation_response(
  181. uint8_t* buff_rx,
  182. uint16_t len_rx,
  183. uint8_t* buff_tx,
  184. MifareUlDevice* mf_ul_emulate) {
  185. uint8_t cmd = buff_rx[0];
  186. uint16_t page_num = mf_ul_emulate->data.data_size / 4;
  187. uint16_t tx_bytes = 0;
  188. uint16_t tx_bits = 0;
  189. bool command_parsed = false;
  190. // Check composite commands
  191. if(mf_ul_emulate->comp_write_cmd_started) {
  192. // Compatibility write is the only one composit command
  193. if(len_rx == 16) {
  194. memcpy(&mf_ul_emulate->data.data[mf_ul_emulate->comp_write_page_addr * 4], buff_rx, 4);
  195. mf_ul_emulate->data_changed = true;
  196. // Send ACK message
  197. buff_tx[0] = 0x0A;
  198. tx_bits = 4;
  199. command_parsed = true;
  200. }
  201. mf_ul_emulate->comp_write_cmd_started = false;
  202. } else if(cmd == MF_UL_GET_VERSION_CMD) {
  203. if(mf_ul_emulate->data.type != MfUltralightTypeUnknown) {
  204. tx_bytes = sizeof(mf_ul_emulate->data.version);
  205. memcpy(buff_tx, &mf_ul_emulate->data.version, tx_bytes);
  206. command_parsed = true;
  207. }
  208. } else if(cmd == MF_UL_READ_CMD) {
  209. uint8_t start_page = buff_rx[1];
  210. if(start_page < page_num) {
  211. tx_bytes = 16;
  212. if(start_page + 4 > page_num) {
  213. // Handle roll-over mechanism
  214. uint8_t end_pages_num = page_num - start_page;
  215. memcpy(buff_tx, &mf_ul_emulate->data.data[start_page * 4], end_pages_num * 4);
  216. memcpy(
  217. &buff_tx[end_pages_num * 4],
  218. mf_ul_emulate->data.data,
  219. (4 - end_pages_num) * 4);
  220. } else {
  221. memcpy(buff_tx, &mf_ul_emulate->data.data[start_page * 4], tx_bytes);
  222. }
  223. mf_ul_protect_auth_data_on_read_command(
  224. buff_tx, start_page, (start_page + 4), mf_ul_emulate);
  225. command_parsed = true;
  226. }
  227. } else if(cmd == MF_UL_FAST_READ_CMD) {
  228. if(mf_ul_emulate->support_fast_read) {
  229. uint8_t start_page = buff_rx[1];
  230. uint8_t end_page = buff_rx[2];
  231. if((start_page < page_num) && (end_page < page_num) && (start_page < (end_page + 1))) {
  232. tx_bytes = ((end_page + 1) - start_page) * 4;
  233. memcpy(buff_tx, &mf_ul_emulate->data.data[start_page * 4], tx_bytes);
  234. mf_ul_protect_auth_data_on_read_command(
  235. buff_tx, start_page, end_page, mf_ul_emulate);
  236. command_parsed = true;
  237. }
  238. }
  239. } else if(cmd == MF_UL_WRITE) {
  240. uint8_t write_page = buff_rx[1];
  241. if((write_page > 1) && (write_page < page_num - 2)) {
  242. memcpy(&mf_ul_emulate->data.data[write_page * 4], &buff_rx[2], 4);
  243. mf_ul_emulate->data_changed = true;
  244. // ACK
  245. buff_tx[0] = 0x0A;
  246. tx_bits = 4;
  247. command_parsed = true;
  248. }
  249. } else if(cmd == MF_UL_COMP_WRITE) {
  250. uint8_t write_page = buff_rx[1];
  251. if((write_page > 1) && (write_page < page_num - 2)) {
  252. mf_ul_emulate->comp_write_cmd_started = true;
  253. mf_ul_emulate->comp_write_page_addr = write_page;
  254. // ACK
  255. buff_tx[0] = 0x0A;
  256. tx_bits = 4;
  257. command_parsed = true;
  258. }
  259. } else if(cmd == MF_UL_READ_CNT) {
  260. uint8_t cnt_num = buff_rx[1];
  261. if(cnt_num < 3) {
  262. buff_tx[0] = mf_ul_emulate->data.counter[cnt_num] >> 16;
  263. buff_tx[1] = mf_ul_emulate->data.counter[cnt_num] >> 8;
  264. buff_tx[2] = mf_ul_emulate->data.counter[cnt_num];
  265. tx_bytes = 3;
  266. command_parsed = true;
  267. }
  268. } else if(cmd == MF_UL_INC_CNT) {
  269. uint8_t cnt_num = buff_rx[1];
  270. uint32_t inc = (buff_rx[2] | (buff_rx[3] << 8) | (buff_rx[4] << 16));
  271. if((cnt_num < 3) && (mf_ul_emulate->data.counter[cnt_num] + inc < 0x00FFFFFF)) {
  272. mf_ul_emulate->data.counter[cnt_num] += inc;
  273. mf_ul_emulate->data_changed = true;
  274. // ACK
  275. buff_tx[0] = 0x0A;
  276. tx_bits = 4;
  277. command_parsed = true;
  278. }
  279. } else if(cmd == MF_UL_AUTH) {
  280. if(mf_ul_emulate->data.type >= MfUltralightTypeNTAG213) {
  281. if(memcmp(&buff_rx[1], mf_ul_emulate->auth_data->pwd, 4) == 0) {
  282. buff_tx[0] = mf_ul_emulate->auth_data->pack.raw[0];
  283. buff_tx[1] = mf_ul_emulate->auth_data->pack.raw[1];
  284. tx_bytes = 2;
  285. command_parsed = true;
  286. } else if(!mf_ul_emulate->auth_data->pack.value) {
  287. buff_tx[0] = 0x80;
  288. buff_tx[1] = 0x80;
  289. tx_bytes = 2;
  290. command_parsed = true;
  291. }
  292. }
  293. } else if(cmd == MF_UL_READ_SIG) {
  294. // Check 2nd byte = 0x00 - RFU
  295. if(buff_rx[1] == 0x00) {
  296. tx_bytes = sizeof(mf_ul_emulate->data.signature);
  297. memcpy(buff_tx, mf_ul_emulate->data.signature, tx_bytes);
  298. command_parsed = true;
  299. }
  300. } else if(cmd == MF_UL_CHECK_TEARING) {
  301. uint8_t cnt_num = buff_rx[1];
  302. if(cnt_num < 3) {
  303. buff_tx[0] = mf_ul_emulate->data.tearing[cnt_num];
  304. tx_bytes = 1;
  305. command_parsed = true;
  306. }
  307. } else if(cmd == MF_UL_HALT_START) {
  308. tx_bits = 0;
  309. command_parsed = true;
  310. }
  311. if(!command_parsed) {
  312. // Send NACK
  313. buff_tx[0] = 0x00;
  314. tx_bits = 4;
  315. }
  316. // Return tx buffer size in bits
  317. if(tx_bytes) {
  318. tx_bits = tx_bytes * 8;
  319. }
  320. return tx_bits;
  321. }