 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>
		
			
				
	
	
		
			129 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			129 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "felica_poller_i.h"
 | |
| 
 | |
| #include <nfc/helpers/felica_crc.h>
 | |
| 
 | |
| #define TAG "FelicaPoller"
 | |
| 
 | |
| static FelicaError felica_poller_process_error(NfcError error) {
 | |
|     switch(error) {
 | |
|     case NfcErrorNone:
 | |
|         return FelicaErrorNone;
 | |
|     case NfcErrorTimeout:
 | |
|         return FelicaErrorTimeout;
 | |
|     default:
 | |
|         return FelicaErrorNotPresent;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static FelicaError felica_poller_frame_exchange(
 | |
|     FelicaPoller* instance,
 | |
|     const BitBuffer* tx_buffer,
 | |
|     BitBuffer* rx_buffer,
 | |
|     uint32_t fwt) {
 | |
|     furi_assert(instance);
 | |
| 
 | |
|     const size_t tx_bytes = bit_buffer_get_size_bytes(tx_buffer);
 | |
|     furi_assert(tx_bytes <= bit_buffer_get_capacity_bytes(instance->tx_buffer) - FELICA_CRC_SIZE);
 | |
| 
 | |
|     felica_crc_append(instance->tx_buffer);
 | |
| 
 | |
|     FelicaError ret = FelicaErrorNone;
 | |
| 
 | |
|     do {
 | |
|         NfcError error =
 | |
|             nfc_poller_trx(instance->nfc, instance->tx_buffer, instance->rx_buffer, fwt);
 | |
|         if(error != NfcErrorNone) {
 | |
|             ret = felica_poller_process_error(error);
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         bit_buffer_copy(rx_buffer, instance->rx_buffer);
 | |
|         if(!felica_crc_check(instance->rx_buffer)) {
 | |
|             ret = FelicaErrorWrongCrc;
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         felica_crc_trim(rx_buffer);
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| FelicaError felica_poller_polling(
 | |
|     FelicaPoller* instance,
 | |
|     const FelicaPollerPollingCommand* cmd,
 | |
|     FelicaPollerPollingResponse* resp) {
 | |
|     furi_assert(instance);
 | |
|     furi_assert(cmd);
 | |
|     furi_assert(resp);
 | |
| 
 | |
|     FelicaError error = FelicaErrorNone;
 | |
| 
 | |
|     do {
 | |
|         bit_buffer_set_size_bytes(instance->tx_buffer, 2);
 | |
|         // Set frame len
 | |
|         bit_buffer_set_byte(
 | |
|             instance->tx_buffer, 0, sizeof(FelicaPollerPollingCommand) + FELICA_CRC_SIZE);
 | |
|         // Set command code
 | |
|         bit_buffer_set_byte(instance->tx_buffer, 1, FELICA_POLLER_CMD_POLLING_REQ_CODE);
 | |
|         // Set other data
 | |
|         bit_buffer_append_bytes(
 | |
|             instance->tx_buffer, (uint8_t*)cmd, sizeof(FelicaPollerPollingCommand));
 | |
| 
 | |
|         error = felica_poller_frame_exchange(
 | |
|             instance, instance->tx_buffer, instance->rx_buffer, FELICA_POLLER_POLLING_FWT);
 | |
| 
 | |
|         if(error != FelicaErrorNone) break;
 | |
|         if(bit_buffer_get_byte(instance->rx_buffer, 1) != FELICA_POLLER_CMD_POLLING_RESP_CODE) {
 | |
|             error = FelicaErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
|         if(bit_buffer_get_size_bytes(instance->rx_buffer) <
 | |
|            sizeof(FelicaIDm) + sizeof(FelicaPMm) + 1) {
 | |
|             error = FelicaErrorProtocol;
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         bit_buffer_write_bytes_mid(instance->rx_buffer, resp->idm.data, 2, sizeof(FelicaIDm));
 | |
|         bit_buffer_write_bytes_mid(
 | |
|             instance->rx_buffer, resp->pmm.data, sizeof(FelicaIDm) + 2, sizeof(FelicaPMm));
 | |
| 
 | |
|     } while(false);
 | |
| 
 | |
|     return error;
 | |
| }
 | |
| 
 | |
| FelicaError felica_poller_activate(FelicaPoller* instance, FelicaData* data) {
 | |
|     furi_assert(instance);
 | |
| 
 | |
|     felica_reset(data);
 | |
| 
 | |
|     FelicaError ret;
 | |
| 
 | |
|     do {
 | |
|         bit_buffer_reset(instance->tx_buffer);
 | |
|         bit_buffer_reset(instance->rx_buffer);
 | |
| 
 | |
|         // Send Polling command
 | |
|         const FelicaPollerPollingCommand polling_cmd = {
 | |
|             .system_code = FELICA_SYSTEM_CODE_CODE,
 | |
|             .request_code = 0,
 | |
|             .time_slot = FELICA_TIME_SLOT_1,
 | |
|         };
 | |
|         FelicaPollerPollingResponse polling_resp = {};
 | |
| 
 | |
|         ret = felica_poller_polling(instance, &polling_cmd, &polling_resp);
 | |
| 
 | |
|         if(ret != FelicaErrorNone) {
 | |
|             FURI_LOG_T(TAG, "Activation failed error: %d", ret);
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         data->idm = polling_resp.idm;
 | |
|         data->pmm = polling_resp.pmm;
 | |
|         instance->state = FelicaPollerStateActivated;
 | |
|     } while(false);
 | |
| 
 | |
|     return ret;
 | |
| }
 |