gen4_poller_i.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. #include "gen4_poller_i.h"
  2. #include "bit_buffer.h"
  3. #include "protocols/gen4/gen4_poller.h"
  4. #include <nfc/protocols/iso14443_3a/iso14443_3a_poller.h>
  5. #include <bit_lib.h>
  6. #define GEN4_CMD_PREFIX (0xCF)
  7. #define GEN4_CMD_SET_SHD_MODE (0x32)
  8. #define GEN4_CMD_GET_CFG (0xC6)
  9. #define GEN4_CMD_GET_REVISION (0xCC)
  10. #define GEN4_CMD_WRITE (0xCD)
  11. #define GEN4_CMD_READ (0xCE)
  12. #define GEN4_CMD_SET_DW_BLOCK_0 (0xCF)
  13. #define GEN4_CMD_SET_CFG (0xF0)
  14. #define GEN4_CMD_FUSE_CFG (0xF1)
  15. #define GEN4_CMD_SET_PWD (0xFE)
  16. #define GEM4_RESPONSE_SUCCESS (0x02)
  17. #define CONFIG_SIZE_MAX (32)
  18. #define CONFIG_SIZE_MIN (30)
  19. #define REVISION_SIZE (5)
  20. static Gen4PollerError gen4_poller_process_error(Iso14443_3aError error) {
  21. Gen4PollerError ret = Gen4PollerErrorNone;
  22. if(error == Iso14443_3aErrorNone) {
  23. ret = Gen4PollerErrorNone;
  24. } else {
  25. ret = Gen4PollerErrorTimeout;
  26. }
  27. return ret;
  28. }
  29. Gen4PollerError gen4_poller_set_shadow_mode(
  30. Gen4Poller* instance,
  31. uint32_t password,
  32. Gen4PollerShadowMode mode) {
  33. Gen4PollerError ret = Gen4PollerErrorNone;
  34. bit_buffer_reset(instance->tx_buffer);
  35. do {
  36. uint8_t password_arr[4] = {};
  37. bit_lib_num_to_bytes_be(password, COUNT_OF(password_arr), password_arr);
  38. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  39. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  40. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_SET_SHD_MODE);
  41. bit_buffer_append_byte(instance->tx_buffer, mode);
  42. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  43. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  44. if(error != Iso14443_3aErrorNone) {
  45. ret = gen4_poller_process_error(error);
  46. break;
  47. }
  48. size_t response = bit_buffer_get_size_bytes(instance->rx_buffer);
  49. FURI_LOG_D(TAG, "Card response: 0x%02X, Shadow mode set: 0x%02X", response, mode);
  50. if(response != GEM4_RESPONSE_SUCCESS) {
  51. ret = Gen4PollerErrorProtocol;
  52. break;
  53. }
  54. } while(false);
  55. return ret;
  56. }
  57. Gen4PollerError gen4_poller_set_direct_write_block_0_mode(
  58. Gen4Poller* instance,
  59. uint32_t password,
  60. Gen4PollerDirectWriteBlock0Mode mode) {
  61. Gen4PollerError ret = Gen4PollerErrorNone;
  62. bit_buffer_reset(instance->tx_buffer);
  63. do {
  64. uint8_t password_arr[4] = {};
  65. bit_lib_num_to_bytes_be(password, COUNT_OF(password_arr), password_arr);
  66. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  67. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  68. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_SET_DW_BLOCK_0);
  69. bit_buffer_append_byte(instance->tx_buffer, mode);
  70. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  71. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  72. if(error != Iso14443_3aErrorNone) {
  73. ret = gen4_poller_process_error(error);
  74. break;
  75. }
  76. size_t response = bit_buffer_get_size_bytes(instance->rx_buffer);
  77. FURI_LOG_D(
  78. TAG, "Card response: 0x%02X, Direct write to block 0 mode set: 0x%02X", response, mode);
  79. if(response != GEM4_RESPONSE_SUCCESS) {
  80. ret = Gen4PollerErrorProtocol;
  81. break;
  82. }
  83. } while(false);
  84. return ret;
  85. }
  86. Gen4PollerError
  87. gen4_poller_get_config(Gen4Poller* instance, uint32_t password, uint8_t* config_result) {
  88. Gen4PollerError ret = Gen4PollerErrorNone;
  89. bit_buffer_reset(instance->tx_buffer);
  90. do {
  91. uint8_t password_arr[4] = {};
  92. bit_lib_num_to_bytes_be(password, COUNT_OF(password_arr), password_arr);
  93. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  94. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  95. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_GET_CFG);
  96. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  97. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  98. if(error != Iso14443_3aErrorNone) {
  99. ret = gen4_poller_process_error(error);
  100. break;
  101. }
  102. size_t rx_bytes = bit_buffer_get_size_bytes(instance->rx_buffer);
  103. if((rx_bytes != CONFIG_SIZE_MAX) && (rx_bytes != CONFIG_SIZE_MIN)) {
  104. ret = Gen4PollerErrorProtocol;
  105. break;
  106. }
  107. bit_buffer_write_bytes(instance->rx_buffer, config_result, CONFIG_SIZE_MAX);
  108. } while(false);
  109. return ret;
  110. }
  111. Gen4PollerError
  112. gen4_poller_get_revision(Gen4Poller* instance, uint32_t password, uint8_t* revision_result) {
  113. Gen4PollerError ret = Gen4PollerErrorNone;
  114. bit_buffer_reset(instance->tx_buffer);
  115. do {
  116. uint8_t password_arr[4] = {};
  117. bit_lib_num_to_bytes_be(password, COUNT_OF(password_arr), password_arr);
  118. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  119. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  120. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_GET_REVISION);
  121. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  122. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  123. if(error != Iso14443_3aErrorNone) {
  124. ret = gen4_poller_process_error(error);
  125. break;
  126. }
  127. size_t rx_bytes = bit_buffer_get_size_bytes(instance->rx_buffer);
  128. if(rx_bytes != REVISION_SIZE) {
  129. ret = Gen4PollerErrorProtocol;
  130. break;
  131. }
  132. bit_buffer_write_bytes(instance->rx_buffer, revision_result, REVISION_SIZE);
  133. } while(false);
  134. return ret;
  135. }
  136. Gen4PollerError gen4_poller_set_config(
  137. Gen4Poller* instance,
  138. uint32_t password,
  139. const uint8_t* config,
  140. size_t config_size,
  141. bool fuse) {
  142. Gen4PollerError ret = Gen4PollerErrorNone;
  143. bit_buffer_reset(instance->tx_buffer);
  144. do {
  145. uint8_t password_arr[4] = {};
  146. bit_lib_num_to_bytes_be(password, COUNT_OF(password_arr), password_arr);
  147. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  148. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  149. uint8_t fuse_config = fuse ? GEN4_CMD_FUSE_CFG : GEN4_CMD_SET_CFG;
  150. bit_buffer_append_byte(instance->tx_buffer, fuse_config);
  151. bit_buffer_append_bytes(instance->tx_buffer, config, config_size);
  152. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  153. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  154. if(error != Iso14443_3aErrorNone) {
  155. ret = gen4_poller_process_error(error);
  156. break;
  157. }
  158. size_t response = bit_buffer_get_size_bytes(instance->rx_buffer);
  159. FURI_LOG_D(TAG, "Card response to set default config command: 0x%02X", response);
  160. if(response != GEM4_RESPONSE_SUCCESS) {
  161. ret = Gen4PollerErrorProtocol;
  162. break;
  163. }
  164. } while(false);
  165. return ret;
  166. }
  167. Gen4PollerError gen4_poller_write_block(
  168. Gen4Poller* instance,
  169. uint32_t password,
  170. uint8_t block_num,
  171. const uint8_t* data) {
  172. Gen4PollerError ret = Gen4PollerErrorNone;
  173. bit_buffer_reset(instance->tx_buffer);
  174. do {
  175. uint8_t password_arr[4] = {};
  176. bit_lib_num_to_bytes_be(password, COUNT_OF(password_arr), password_arr);
  177. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  178. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  179. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_WRITE);
  180. bit_buffer_append_byte(instance->tx_buffer, block_num);
  181. bit_buffer_append_bytes(instance->tx_buffer, data, GEN4_POLLER_BLOCK_SIZE);
  182. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  183. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  184. if(error != Iso14443_3aErrorNone) {
  185. ret = gen4_poller_process_error(error);
  186. break;
  187. }
  188. size_t rx_bytes = bit_buffer_get_size_bytes(instance->rx_buffer);
  189. if(rx_bytes != 2) {
  190. ret = Gen4PollerErrorProtocol;
  191. break;
  192. }
  193. } while(false);
  194. return ret;
  195. }
  196. Gen4PollerError
  197. gen4_poller_change_password(Gen4Poller* instance, uint32_t pwd_current, uint32_t pwd_new) {
  198. Gen4PollerError ret = Gen4PollerErrorNone;
  199. bit_buffer_reset(instance->tx_buffer);
  200. do {
  201. uint8_t password_arr[4] = {};
  202. bit_lib_num_to_bytes_be(pwd_current, COUNT_OF(password_arr), password_arr);
  203. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_PREFIX);
  204. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  205. bit_buffer_append_byte(instance->tx_buffer, GEN4_CMD_SET_PWD);
  206. bit_lib_num_to_bytes_be(pwd_new, COUNT_OF(password_arr), password_arr);
  207. bit_buffer_append_bytes(instance->tx_buffer, password_arr, COUNT_OF(password_arr));
  208. Iso14443_3aError error = iso14443_3a_poller_send_standard_frame(
  209. instance->iso3_poller, instance->tx_buffer, instance->rx_buffer, GEN4_POLLER_MAX_FWT);
  210. if(error != Iso14443_3aErrorNone) {
  211. ret = gen4_poller_process_error(error);
  212. break;
  213. }
  214. size_t response = bit_buffer_get_size_bytes(instance->rx_buffer);
  215. FURI_LOG_D(
  216. TAG,
  217. "Trying to change password from 0x%08lX to 0x%08lX. Card response: 0x%02X",
  218. pwd_current,
  219. pwd_new,
  220. response);
  221. if(response != GEM4_RESPONSE_SUCCESS) {
  222. ret = Gen4PollerErrorProtocol;
  223. break;
  224. }
  225. } while(false);
  226. return ret;
  227. }