 c00776ca22
			
		
	
	
		c00776ca22
		
			
		
	
	
	
	
		
			
			* drivers: expose st25r3916 driver API * nfc poller: add start with custom callback * mf classic: rework sync API with poller custom start * mf ultralight: rework sync API with poller custom start * iso14443_3a poller: remove unused col res state * nfc: rework nfc poller custom start * mf ultralight: rename sync API * mf classic: rename sync API * iso14443-3a: rename sync API * nfc: remove async prefix in internal functions * nfc: expose internal API * nfc: fix sync api include and docs * targets: fix f18 build * nfc: rework NfcGenericEventEx type * nfc poller: add documentation * iso14443-3a poller: add documentation * felica poller: add documentation * iso14443_3b poller: add documentation * so14443_4a poller: add documentation * iso14443_4b poller: add documentation * iso15693 poller: add documentation * slix poller: add documentation * mf desfire poller: add documentation * mf ultralight poller: fix API and add documentation * mf classic poller: add documentation Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			307 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			307 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "mf_ultralight_poller_i.h"
 | |
| 
 | |
| #include <furi.h>
 | |
| 
 | |
| #define TAG "MfUltralightPoller"
 | |
| 
 | |
| MfUltralightError mf_ultralight_process_error(Iso14443_3aError error) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
| 
 | |
|     switch(error) {
 | |
|     case Iso14443_3aErrorNone:
 | |
|         ret = MfUltralightErrorNone;
 | |
|         break;
 | |
|     case Iso14443_3aErrorNotPresent:
 | |
|         ret = MfUltralightErrorNotPresent;
 | |
|         break;
 | |
|     case Iso14443_3aErrorColResFailed:
 | |
|     case Iso14443_3aErrorCommunication:
 | |
|     case Iso14443_3aErrorWrongCrc:
 | |
|         ret = MfUltralightErrorProtocol;
 | |
|         break;
 | |
|     case Iso14443_3aErrorTimeout:
 | |
|         ret = MfUltralightErrorTimeout;
 | |
|         break;
 | |
|     default:
 | |
|         ret = MfUltralightErrorProtocol;
 | |
|         break;
 | |
|     }
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_auth_pwd(
 | |
|     MfUltralightPoller* instance,
 | |
|     MfUltralightPollerAuthContext* data) {
 | |
|     uint8_t auth_cmd[5] = {MF_ULTRALIGHT_CMD_PWD_AUTH}; //-V1009
 | |
|     memccpy(&auth_cmd[1], data->password.data, 0, MF_ULTRALIGHT_AUTH_PASSWORD_SIZE);
 | |
|     bit_buffer_copy_bytes(instance->tx_buffer, auth_cmd, sizeof(auth_cmd));
 | |
| 
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
|     do {
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) != MF_ULTRALIGHT_AUTH_PACK_SIZE) {
 | |
|             ret = MfUltralightErrorAuth;
 | |
|             break;
 | |
|         }
 | |
|         bit_buffer_write_bytes(instance->rx_buffer, data->pack.data, MF_ULTRALIGHT_AUTH_PACK_SIZE);
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_authenticate(MfUltralightPoller* instance) {
 | |
|     uint8_t auth_cmd[2] = {MF_ULTRALIGHT_CMD_AUTH, 0x00};
 | |
|     bit_buffer_copy_bytes(instance->tx_buffer, auth_cmd, sizeof(auth_cmd));
 | |
| 
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
|     do {
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if((bit_buffer_get_size_bytes(instance->rx_buffer) != MF_ULTRALIGHT_AUTH_RESPONSE_SIZE) &&
 | |
|            (bit_buffer_get_byte(instance->rx_buffer, 0) != 0xAF)) {
 | |
|             ret = MfUltralightErrorAuth;
 | |
|             break;
 | |
|         }
 | |
|         //Save encrypted PICC random number RndB here if needed
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_read_page_from_sector(
 | |
|     MfUltralightPoller* instance,
 | |
|     uint8_t sector,
 | |
|     uint8_t tag,
 | |
|     MfUltralightPageReadCommandData* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         const uint8_t select_sector_cmd[2] = {MF_ULTRALIGHT_CMD_SECTOR_SELECT, 0xff};
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, select_sector_cmd, sizeof(select_sector_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorWrongCrc) {
 | |
|             FURI_LOG_D(TAG, "Failed to issue sector select command");
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         const uint8_t read_sector_cmd[4] = {sector, 0x00, 0x00, 0x00};
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, read_sector_cmd, sizeof(read_sector_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorTimeout) {
 | |
|             // This is NOT a typo! The tag ACKs by not sending a response within 1ms.
 | |
|             FURI_LOG_D(TAG, "Sector %u select NAK'd", sector);
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         ret = mf_ultralight_poller_read_page(instance, tag, data);
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_read_page(
 | |
|     MfUltralightPoller* instance,
 | |
|     uint8_t start_page,
 | |
|     MfUltralightPageReadCommandData* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         uint8_t read_page_cmd[2] = {MF_ULTRALIGHT_CMD_READ_PAGE, start_page};
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, read_page_cmd, sizeof(read_page_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) !=
 | |
|            sizeof(MfUltralightPageReadCommandData)) {
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         bit_buffer_write_bytes(instance->rx_buffer, data, sizeof(MfUltralightPageReadCommandData));
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_write_page(
 | |
|     MfUltralightPoller* instance,
 | |
|     uint8_t page,
 | |
|     const MfUltralightPage* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         uint8_t write_page_cmd[MF_ULTRALIGHT_PAGE_SIZE + 2] = {MF_ULTRALIGHT_CMD_WRITE_PAGE, page};
 | |
|         memcpy(&write_page_cmd[2], data->data, MF_ULTRALIGHT_PAGE_SIZE);
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, write_page_cmd, sizeof(write_page_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorWrongCrc) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size(instance->rx_buffer) != 4) {
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         if(!bit_buffer_starts_with_byte(instance->rx_buffer, MF_ULTRALIGHT_CMD_ACK)) {
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError
 | |
|     mf_ultralight_poller_read_version(MfUltralightPoller* instance, MfUltralightVersion* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         const uint8_t get_version_cmd = MF_ULTRALIGHT_CMD_GET_VERSION;
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, &get_version_cmd, sizeof(get_version_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) != sizeof(MfUltralightVersion)) {
 | |
|             FURI_LOG_I(
 | |
|                 TAG, "Read Version failed: %zu", bit_buffer_get_size_bytes(instance->rx_buffer));
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         bit_buffer_write_bytes(instance->rx_buffer, data, sizeof(MfUltralightVersion));
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError
 | |
|     mf_ultralight_poller_read_signature(MfUltralightPoller* instance, MfUltralightSignature* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         const uint8_t read_signature_cmd[2] = {MF_ULTRALIGHT_CMD_READ_SIG, 0x00};
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, read_signature_cmd, sizeof(read_signature_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) != sizeof(MfUltralightSignature)) {
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         bit_buffer_write_bytes(instance->rx_buffer, data, sizeof(MfUltralightSignature));
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_read_counter(
 | |
|     MfUltralightPoller* instance,
 | |
|     uint8_t counter_num,
 | |
|     MfUltralightCounter* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         uint8_t read_counter_cmd[2] = {MF_ULTRALIGHT_CMD_READ_CNT, counter_num};
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, read_counter_cmd, sizeof(read_counter_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) != MF_ULTRALIGHT_COUNTER_SIZE) {
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         bit_buffer_write_bytes(instance->rx_buffer, data->data, MF_ULTRALIGHT_COUNTER_SIZE);
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| MfUltralightError mf_ultralight_poller_read_tearing_flag(
 | |
|     MfUltralightPoller* instance,
 | |
|     uint8_t tearing_falg_num,
 | |
|     MfUltralightTearingFlag* data) {
 | |
|     MfUltralightError ret = MfUltralightErrorNone;
 | |
|     Iso14443_3aError error = Iso14443_3aErrorNone;
 | |
| 
 | |
|     do {
 | |
|         uint8_t check_tearing_cmd[2] = {MF_ULTRALIGHT_CMD_CHECK_TEARING, tearing_falg_num};
 | |
|         bit_buffer_copy_bytes(instance->tx_buffer, check_tearing_cmd, sizeof(check_tearing_cmd));
 | |
|         error = iso14443_3a_poller_send_standard_frame(
 | |
|             instance->iso14443_3a_poller,
 | |
|             instance->tx_buffer,
 | |
|             instance->rx_buffer,
 | |
|             MF_ULTRALIGHT_POLLER_STANDARD_FWT_FC);
 | |
|         if(error != Iso14443_3aErrorNone) {
 | |
|             ret = mf_ultralight_process_error(error);
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) != sizeof(MfUltralightTearingFlag)) {
 | |
|             ret = MfUltralightErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         bit_buffer_write_bytes(instance->rx_buffer, data, sizeof(MfUltralightTearingFlag));
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 |