"A long time ago in a galaxy far, far away...." we started NFC subsystem refactoring. Starring: - @gornekich - NFC refactoring project lead, architect, senior developer - @gsurkov - architect, senior developer - @RebornedBrain - senior developer Supporting roles: - @skotopes, @DrZlo13, @hedger - general architecture advisors, code review - @Astrrra, @doomwastaken, @Hellitron, @ImagineVagon333 - quality assurance Special thanks: @bettse, @pcunning, @nxv, @noproto, @AloneLiberty and everyone else who has been helping us all this time and contributing valuable knowledges, ideas and source code.
		
			
				
	
	
		
			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_async_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_async_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_async_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;
 | 
						|
}
 |