mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-12-04 18:29:31 +00:00
389ff92cc1
* Makefile, Scripts: new linter * About: remove ID from IC * Firmware: remove double define for DIVC/DIVR * Scripts: check folder names too. Docker: replace syntax check with make lint. * Reformat Sources and Migrate to new file naming convention * Docker: symlink clang-format-12 to clang-format * Add coding style guide
1978 lines
76 KiB
C
1978 lines
76 KiB
C
/**
|
|
******************************************************************************
|
|
*
|
|
* COPYRIGHT(c) 2020 STMicroelectronics
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without modification,
|
|
* are permitted provided that the following conditions are met:
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
* 3. Neither the name of STMicroelectronics nor the names of its contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
|
|
/*! \file rfal_nfc.c
|
|
*
|
|
* \author Gustavo Patricio
|
|
*
|
|
* \brief RFAL NFC device
|
|
*
|
|
* This module provides the required features to behave as an NFC Poller
|
|
* or Listener device. It grants an easy to use interface for the following
|
|
* activities: Technology Detection, Collision Resollution, Activation,
|
|
* Data Exchange, and Deactivation
|
|
*
|
|
* This layer is influenced by (but not fully aligned with) the NFC Forum
|
|
* specifications, in particular: Activity 2.0 and NCI 2.0
|
|
*
|
|
*/
|
|
|
|
/*
|
|
******************************************************************************
|
|
* INCLUDES
|
|
******************************************************************************
|
|
*/
|
|
#include "rfal_nfc.h"
|
|
#include "utils.h"
|
|
#include "rfal_analogConfig.h"
|
|
|
|
/*
|
|
******************************************************************************
|
|
* GLOBAL DEFINES
|
|
******************************************************************************
|
|
*/
|
|
#define RFAL_NFC_MAX_DEVICES 5U /* Max number of devices supported */
|
|
|
|
/*
|
|
******************************************************************************
|
|
* GLOBAL MACROS
|
|
******************************************************************************
|
|
*/
|
|
|
|
#define rfalNfcNfcNotify(st) \
|
|
if(gNfcDev.disc.notifyCb != NULL) gNfcDev.disc.notifyCb(st)
|
|
|
|
/*
|
|
******************************************************************************
|
|
* GLOBAL TYPES
|
|
******************************************************************************
|
|
*/
|
|
|
|
/*! Buffer union, only one interface is used at a time */
|
|
typedef union { /* PRQA S 0750 # MISRA 19.2 - Members of the union will not be used concurrently, only one interface at a time */
|
|
rfalIsoDepBufFormat isoDepBuf; /*!< ISO-DEP buffer format (with header/prologue) */
|
|
rfalNfcDepBufFormat nfcDepBuf; /*!< NFC-DEP buffer format (with header/prologue) */
|
|
} rfalNfcTmpBuffer;
|
|
|
|
typedef struct {
|
|
rfalNfcState state; /* Main state */
|
|
uint16_t techsFound; /* Technologies found bitmask */
|
|
uint16_t techs2do; /* Technologies still to be performed */
|
|
rfalBitRate ap2pBR; /* Bit rate to poll for AP2P */
|
|
uint8_t selDevIdx; /* Selected device index */
|
|
rfalNfcDevice* activeDev; /* Active device pointer */
|
|
rfalNfcDiscoverParam disc; /* Discovery parameters */
|
|
rfalNfcDevice devList[RFAL_NFC_MAX_DEVICES]; /*!< Location of device list */
|
|
uint8_t devCnt; /* Decices found counter */
|
|
uint32_t discTmr; /* Discovery Total duration timer */
|
|
ReturnCode dataExErr; /* Last Data Exchange error */
|
|
bool discRestart; /* Restart discover after deactivation flag */
|
|
bool isRxChaining; /* Flag indicating Other device is chaining */
|
|
uint32_t lmMask; /* Listen Mode mask */
|
|
bool isTechInit; /* Flag indicating technology has been set */
|
|
bool isOperOngoing; /* Flag indicating opration is ongoing */
|
|
|
|
rfalNfcBuffer txBuf; /* Tx buffer for Data Exchange */
|
|
rfalNfcBuffer rxBuf; /* Rx buffer for Data Exchange */
|
|
uint16_t rxLen; /* Length of received data on Data Exchange */
|
|
|
|
#if RFAL_FEATURE_NFC_DEP || RFAL_FEATURE_ISO_DEP
|
|
rfalNfcTmpBuffer tmpBuf; /* Tmp buffer for Data Exchange */
|
|
#endif /* RFAL_FEATURE_NFC_DEP || RFAL_FEATURE_ISO_DEP */
|
|
|
|
} rfalNfc;
|
|
|
|
/*
|
|
******************************************************************************
|
|
* LOCAL VARIABLES
|
|
******************************************************************************
|
|
*/
|
|
#ifdef RFAL_TEST_MODE
|
|
rfalNfc gNfcDev;
|
|
#else /* RFAL_TEST_MODE */
|
|
static rfalNfc gNfcDev;
|
|
#endif /* RFAL_TEST_MODE */
|
|
|
|
/*
|
|
******************************************************************************
|
|
* LOCAL FUNCTION PROTOTYPES
|
|
******************************************************************************
|
|
*/
|
|
static ReturnCode rfalNfcPollTechDetetection(void);
|
|
static ReturnCode rfalNfcPollCollResolution(void);
|
|
static ReturnCode rfalNfcPollActivation(uint8_t devIt);
|
|
static ReturnCode rfalNfcDeactivation(void);
|
|
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
static ReturnCode rfalNfcNfcDepActivate(
|
|
rfalNfcDevice* device,
|
|
rfalNfcDepCommMode commMode,
|
|
const uint8_t* atrReq,
|
|
uint16_t atrReqLen);
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
static ReturnCode rfalNfcListenActivation(void);
|
|
#endif /* RFAL_FEATURE_LISTEN_MODE*/
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcInitialize(void) {
|
|
ReturnCode err;
|
|
|
|
gNfcDev.state = RFAL_NFC_STATE_NOTINIT;
|
|
|
|
rfalAnalogConfigInitialize(); /* Initialize RFAL's Analog Configs */
|
|
EXIT_ON_ERR(err, rfalInitialize()); /* Initialize RFAL */
|
|
|
|
gNfcDev.state = RFAL_NFC_STATE_IDLE; /* Go to initialized */
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcDiscover(const rfalNfcDiscoverParam* disParams) {
|
|
/* Check if initialization has been performed */
|
|
if(gNfcDev.state != RFAL_NFC_STATE_IDLE) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
/* Check valid parameters */
|
|
if((disParams == NULL) || (disParams->devLimit > RFAL_NFC_MAX_DEVICES) ||
|
|
(disParams->devLimit == 0U) ||
|
|
((disParams->maxBR > RFAL_BR_1695) && (disParams->maxBR != RFAL_BR_KEEP)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_POLL_TECH_F) != 0U) &&
|
|
(disParams->nfcfBR != RFAL_BR_212) && (disParams->nfcfBR != RFAL_BR_424)) ||
|
|
((((disParams->techs2Find & RFAL_NFC_POLL_TECH_AP2P) != 0U) &&
|
|
(disParams->ap2pBR > RFAL_BR_424)) ||
|
|
(disParams->GBLen > RFAL_NFCDEP_GB_MAX_LEN))) {
|
|
return ERR_PARAM;
|
|
}
|
|
|
|
if((((disParams->techs2Find & RFAL_NFC_POLL_TECH_A) != 0U) && !((bool)RFAL_FEATURE_NFCA)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_POLL_TECH_B) != 0U) && !((bool)RFAL_FEATURE_NFCB)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_POLL_TECH_F) != 0U) && !((bool)RFAL_FEATURE_NFCF)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_POLL_TECH_V) != 0U) && !((bool)RFAL_FEATURE_NFCV)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_POLL_TECH_ST25TB) != 0U) &&
|
|
!((bool)RFAL_FEATURE_ST25TB)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_POLL_TECH_AP2P) != 0U) &&
|
|
!((bool)RFAL_FEATURE_NFC_DEP)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_LISTEN_TECH_A) != 0U) && !((bool)RFAL_FEATURE_NFCA)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_LISTEN_TECH_B) != 0U) && !((bool)RFAL_FEATURE_NFCB)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_LISTEN_TECH_F) != 0U) && !((bool)RFAL_FEATURE_NFCF)) ||
|
|
(((disParams->techs2Find & RFAL_NFC_LISTEN_TECH_AP2P) != 0U) &&
|
|
!((bool)RFAL_FEATURE_NFC_DEP))) {
|
|
return ERR_DISABLED; /* PRQA S 2880 # MISRA 2.1 - Unreachable code due to configuration option being set/unset */
|
|
}
|
|
|
|
/* Initialize context for discovery */
|
|
gNfcDev.activeDev = NULL;
|
|
gNfcDev.techsFound = RFAL_NFC_TECH_NONE;
|
|
gNfcDev.devCnt = 0;
|
|
gNfcDev.discRestart = true;
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.disc = *disParams;
|
|
|
|
/* Calculate Listen Mask */
|
|
gNfcDev.lmMask = 0U;
|
|
gNfcDev.lmMask |=
|
|
(((gNfcDev.disc.techs2Find & RFAL_NFC_LISTEN_TECH_A) != 0U) ? RFAL_LM_MASK_NFCA : 0U);
|
|
gNfcDev.lmMask |=
|
|
(((gNfcDev.disc.techs2Find & RFAL_NFC_LISTEN_TECH_B) != 0U) ? RFAL_LM_MASK_NFCB : 0U);
|
|
gNfcDev.lmMask |=
|
|
(((gNfcDev.disc.techs2Find & RFAL_NFC_LISTEN_TECH_F) != 0U) ? RFAL_LM_MASK_NFCF : 0U);
|
|
gNfcDev.lmMask |=
|
|
(((gNfcDev.disc.techs2Find & RFAL_NFC_LISTEN_TECH_AP2P) != 0U) ? RFAL_LM_MASK_ACTIVE_P2P :
|
|
0U);
|
|
|
|
#if !RFAL_FEATURE_LISTEN_MODE
|
|
/* Check if Listen Mode is supported/Enabled */
|
|
if(gNfcDev.lmMask != 0U) {
|
|
return ERR_DISABLED;
|
|
}
|
|
#endif
|
|
|
|
gNfcDev.state = RFAL_NFC_STATE_START_DISCOVERY;
|
|
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcDeactivate(bool discovery) {
|
|
/* Check for valid state */
|
|
if(gNfcDev.state <= RFAL_NFC_STATE_IDLE) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
/* Check if discovery is to continue afterwards */
|
|
if((discovery == true) && (gNfcDev.disc.techs2Find != RFAL_NFC_TECH_NONE)) {
|
|
/* If so let the state machine continue*/
|
|
gNfcDev.discRestart = discovery;
|
|
gNfcDev.state = RFAL_NFC_STATE_DEACTIVATION;
|
|
} else {
|
|
/* Otherwise deactivate immediately and go to IDLE */
|
|
rfalNfcDeactivation();
|
|
gNfcDev.state = RFAL_NFC_STATE_IDLE;
|
|
}
|
|
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcSelect(uint8_t devIdx) {
|
|
/* Check for valid state */
|
|
if(gNfcDev.state != RFAL_NFC_STATE_POLL_SELECT) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
gNfcDev.selDevIdx = devIdx;
|
|
gNfcDev.state = RFAL_NFC_STATE_POLL_ACTIVATION;
|
|
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
rfalNfcState rfalNfcGetState(void) {
|
|
return gNfcDev.state;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcGetDevicesFound(rfalNfcDevice** devList, uint8_t* devCnt) {
|
|
/* Check for valid state */
|
|
if(gNfcDev.state < RFAL_NFC_STATE_POLL_SELECT) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
/* Check valid parameters */
|
|
if((devList == NULL) || (devCnt == NULL)) {
|
|
return ERR_PARAM;
|
|
}
|
|
|
|
*devCnt = gNfcDev.devCnt;
|
|
*devList = gNfcDev.devList;
|
|
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcGetActiveDevice(rfalNfcDevice** dev) {
|
|
/* Check for valid state */
|
|
if(gNfcDev.state < RFAL_NFC_STATE_ACTIVATED) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
/* Check valid parameter */
|
|
if(dev == NULL) {
|
|
return ERR_PARAM;
|
|
}
|
|
|
|
/* Check for valid state */
|
|
if((gNfcDev.devCnt == 0U) || (gNfcDev.activeDev == NULL)) {
|
|
return ERR_REQUEST;
|
|
}
|
|
|
|
*dev = gNfcDev.activeDev;
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
void rfalNfcWorker(void) {
|
|
ReturnCode err;
|
|
|
|
rfalWorker(); /* Execute RFAL process */
|
|
|
|
switch(gNfcDev.state) {
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_NOTINIT:
|
|
case RFAL_NFC_STATE_IDLE:
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_START_DISCOVERY:
|
|
|
|
/* Initialize context for discovery cycle */
|
|
gNfcDev.devCnt = 0;
|
|
gNfcDev.selDevIdx = 0;
|
|
gNfcDev.techsFound = RFAL_NFC_TECH_NONE;
|
|
gNfcDev.techs2do = gNfcDev.disc.techs2Find;
|
|
gNfcDev.state = RFAL_NFC_STATE_POLL_TECHDETECT;
|
|
|
|
#if RFAL_FEATURE_WAKEUP_MODE
|
|
/* Check if Low power Wake-Up is to be performed */
|
|
if(gNfcDev.disc.wakeupEnabled) {
|
|
/* Initialize Low power Wake-up mode and wait */
|
|
err = rfalWakeUpModeStart(
|
|
(gNfcDev.disc.wakeupConfigDefault ? NULL : &gNfcDev.disc.wakeupConfig));
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.state = RFAL_NFC_STATE_WAKEUP_MODE;
|
|
rfalNfcNfcNotify(gNfcDev.state); /* Notify caller that WU was started */
|
|
}
|
|
}
|
|
#endif /* RFAL_FEATURE_WAKEUP_MODE */
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_WAKEUP_MODE:
|
|
|
|
#if RFAL_FEATURE_WAKEUP_MODE
|
|
/* Check if the Wake-up mode has woke */
|
|
if(rfalWakeUpModeHasWoke()) {
|
|
rfalWakeUpModeStop(); /* Disable Wake-up mode */
|
|
gNfcDev.state = RFAL_NFC_STATE_POLL_TECHDETECT; /* Go to Technology detection */
|
|
|
|
rfalNfcNfcNotify(gNfcDev.state); /* Notify caller that WU has woke */
|
|
}
|
|
#endif /* RFAL_FEATURE_WAKEUP_MODE */
|
|
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_POLL_TECHDETECT:
|
|
|
|
/* Start total duration timer */
|
|
platformTimerDestroy(gNfcDev.discTmr);
|
|
gNfcDev.discTmr = (uint32_t)platformTimerCreate(gNfcDev.disc.totalDuration);
|
|
|
|
err =
|
|
rfalNfcPollTechDetetection(); /* Perform Technology Detection */
|
|
if(err != ERR_BUSY) /* Wait until all technologies are performed */
|
|
{
|
|
if((err != ERR_NONE) ||
|
|
(gNfcDev.techsFound ==
|
|
RFAL_NFC_TECH_NONE)) /* Check if any error occurred or no techs were found */
|
|
{
|
|
rfalFieldOff();
|
|
gNfcDev.state =
|
|
RFAL_NFC_STATE_LISTEN_TECHDETECT; /* Nothing found as poller, go to listener */
|
|
break;
|
|
}
|
|
|
|
gNfcDev.techs2do =
|
|
gNfcDev.techsFound; /* Store the found technologies for collision resolution */
|
|
gNfcDev.state =
|
|
RFAL_NFC_STATE_POLL_COLAVOIDANCE; /* One or more devices found, go to Collision Avoidance */
|
|
}
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_POLL_COLAVOIDANCE:
|
|
|
|
err =
|
|
rfalNfcPollCollResolution(); /* Resolve any eventual collision */
|
|
if(err != ERR_BUSY) /* Wait until all technologies are performed */
|
|
{
|
|
if((err != ERR_NONE) ||
|
|
(gNfcDev.devCnt == 0U)) /* Check if any error occurred or no devices were found */
|
|
{
|
|
gNfcDev.state = RFAL_NFC_STATE_DEACTIVATION;
|
|
break; /* Unable to retrieve any device, restart loop */
|
|
}
|
|
|
|
/* Check if more than one device has been found */
|
|
if(gNfcDev.devCnt > 1U) {
|
|
/* If more than one device was found inform upper layer to choose which one to activate */
|
|
if(gNfcDev.disc.notifyCb != NULL) {
|
|
gNfcDev.state = RFAL_NFC_STATE_POLL_SELECT;
|
|
gNfcDev.disc.notifyCb(gNfcDev.state);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* If only one device or no callback has been set, activate the first device found */
|
|
gNfcDev.selDevIdx = 0U;
|
|
gNfcDev.state = RFAL_NFC_STATE_POLL_ACTIVATION;
|
|
}
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_POLL_ACTIVATION:
|
|
|
|
err = rfalNfcPollActivation(gNfcDev.selDevIdx);
|
|
if(err != ERR_BUSY) /* Wait until all Activation is complete */
|
|
{
|
|
if(err != ERR_NONE) /* Activation failed selected device */
|
|
{
|
|
gNfcDev.state =
|
|
RFAL_NFC_STATE_DEACTIVATION; /* If Activation failed, restart loop */
|
|
break;
|
|
}
|
|
|
|
gNfcDev.state = RFAL_NFC_STATE_ACTIVATED; /* Device has been properly activated */
|
|
rfalNfcNfcNotify(
|
|
gNfcDev.state); /* Inform upper layer that a device has been activated */
|
|
}
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_DATAEXCHANGE:
|
|
|
|
rfalNfcDataExchangeGetStatus(); /* Run the internal state machine */
|
|
|
|
if(gNfcDev.dataExErr != ERR_BUSY) /* If Dataexchange has terminated */
|
|
{
|
|
gNfcDev.state = RFAL_NFC_STATE_DATAEXCHANGE_DONE; /* Go to done state */
|
|
rfalNfcNfcNotify(gNfcDev.state); /* And notify caller */
|
|
}
|
|
if(gNfcDev.dataExErr == ERR_SLEEP_REQ) /* Check if Listen mode has to go to Sleep */
|
|
{
|
|
gNfcDev.state = RFAL_NFC_STATE_LISTEN_SLEEP; /* Go to Listen Sleep state */
|
|
rfalNfcNfcNotify(gNfcDev.state); /* And notify caller */
|
|
}
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_DEACTIVATION:
|
|
|
|
rfalNfcDeactivation(); /* Deactivate current device */
|
|
|
|
gNfcDev.state =
|
|
((gNfcDev.discRestart) ? RFAL_NFC_STATE_START_DISCOVERY : RFAL_NFC_STATE_IDLE);
|
|
rfalNfcNfcNotify(gNfcDev.state); /* Notify caller */
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_LISTEN_TECHDETECT:
|
|
|
|
if(platformTimerIsExpired(gNfcDev.discTmr)) {
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
rfalListenStop();
|
|
#else
|
|
rfalFieldOff();
|
|
#endif /* RFAL_FEATURE_LISTEN_MODE */
|
|
|
|
gNfcDev.state = RFAL_NFC_STATE_START_DISCOVERY; /* Restart the discovery loop */
|
|
rfalNfcNfcNotify(gNfcDev.state); /* Notify caller */
|
|
break;
|
|
}
|
|
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
|
|
if(gNfcDev.lmMask != 0U) /* Check if configured to perform Listen mode */
|
|
{
|
|
err = rfalListenStart(
|
|
gNfcDev.lmMask,
|
|
&gNfcDev.disc.lmConfigPA,
|
|
NULL,
|
|
&gNfcDev.disc.lmConfigPF,
|
|
(uint8_t*)&gNfcDev.rxBuf.rfBuf,
|
|
(uint16_t)rfalConvBytesToBits(sizeof(gNfcDev.rxBuf.rfBuf)),
|
|
&gNfcDev.rxLen);
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.state =
|
|
RFAL_NFC_STATE_LISTEN_COLAVOIDANCE; /* Wait for listen mode to be activated */
|
|
}
|
|
}
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_LISTEN_COLAVOIDANCE:
|
|
|
|
if(platformTimerIsExpired(
|
|
gNfcDev.discTmr)) /* Check if the total duration has been reached */
|
|
{
|
|
rfalListenStop();
|
|
gNfcDev.state = RFAL_NFC_STATE_START_DISCOVERY; /* Restart the discovery loop */
|
|
rfalNfcNfcNotify(gNfcDev.state); /* Notify caller */
|
|
break;
|
|
}
|
|
|
|
/* Check for external field */
|
|
if(rfalListenGetState(NULL, NULL) >= RFAL_LM_STATE_IDLE) {
|
|
gNfcDev.state =
|
|
RFAL_NFC_STATE_LISTEN_ACTIVATION; /* Wait for listen mode to be activated */
|
|
}
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_LISTEN_ACTIVATION:
|
|
case RFAL_NFC_STATE_LISTEN_SLEEP:
|
|
|
|
err = rfalNfcListenActivation();
|
|
if(err != ERR_BUSY) {
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.activeDev =
|
|
gNfcDev.devList; /* Assign the active device to be used further on */
|
|
gNfcDev.devCnt++;
|
|
|
|
gNfcDev.state = RFAL_NFC_STATE_ACTIVATED; /* Device has been properly activated */
|
|
rfalNfcNfcNotify(
|
|
gNfcDev.state); /* Inform upper layer that a device has been activated */
|
|
} else {
|
|
rfalListenStop();
|
|
gNfcDev.state = RFAL_NFC_STATE_START_DISCOVERY; /* Restart the discovery loop */
|
|
rfalNfcNfcNotify(gNfcDev.state); /* Notify caller */
|
|
}
|
|
}
|
|
#endif /* RFAL_FEATURE_LISTEN_MODE */
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_STATE_ACTIVATED:
|
|
case RFAL_NFC_STATE_POLL_SELECT:
|
|
case RFAL_NFC_STATE_DATAEXCHANGE_DONE:
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcDataExchangeStart(
|
|
uint8_t* txData,
|
|
uint16_t txDataLen,
|
|
uint8_t** rxData,
|
|
uint16_t** rvdLen,
|
|
uint32_t fwt,
|
|
uint32_t flags) {
|
|
ReturnCode err;
|
|
rfalTransceiveContext ctx;
|
|
|
|
/*******************************************************************************/
|
|
/* The Data Exchange is divided in two different moments, the trigger/Start of *
|
|
* the transfer followed by the check until its completion */
|
|
if((gNfcDev.state >= RFAL_NFC_STATE_ACTIVATED) && (gNfcDev.activeDev != NULL)) {
|
|
/*******************************************************************************/
|
|
/* In Listen mode is the Poller that initiates the communicatation */
|
|
/* Assign output parameters and rfalNfcDataExchangeGetStatus will return */
|
|
/* incoming data from Poller/Initiator */
|
|
if((gNfcDev.state == RFAL_NFC_STATE_ACTIVATED) &&
|
|
rfalNfcIsRemDevPoller(gNfcDev.activeDev->type)) {
|
|
if(txDataLen > 0U) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
*rvdLen = (uint16_t*)&gNfcDev.rxLen;
|
|
*rxData = (uint8_t*)( (gNfcDev.activeDev->rfInterface == RFAL_NFC_INTERFACE_ISODEP) ? gNfcDev.rxBuf.isoDepBuf.apdu :
|
|
((gNfcDev.activeDev->rfInterface == RFAL_NFC_INTERFACE_NFCDEP) ? gNfcDev.rxBuf.nfcDepBuf.pdu : gNfcDev.rxBuf.rfBuf));
|
|
if(gNfcDev.disc.activate_after_sak) {
|
|
gNfcDev.state = RFAL_NFC_STATE_DATAEXCHANGE_DONE;
|
|
}
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
switch(gNfcDev.activeDev
|
|
->rfInterface) /* Check which RF interface shall be used/has been activated */
|
|
{
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_INTERFACE_RF:
|
|
|
|
rfalCreateByteFlagsTxRxContext(
|
|
ctx,
|
|
(uint8_t*)txData,
|
|
txDataLen,
|
|
gNfcDev.rxBuf.rfBuf,
|
|
sizeof(gNfcDev.rxBuf.rfBuf),
|
|
&gNfcDev.rxLen,
|
|
flags,
|
|
fwt);
|
|
if(flags == RFAL_TXRX_FLAGS_RAW) {
|
|
ctx.txBufLen = txDataLen;
|
|
}
|
|
*rxData = (uint8_t*)gNfcDev.rxBuf.rfBuf;
|
|
*rvdLen = (uint16_t*)&gNfcDev.rxLen;
|
|
err = rfalStartTransceive(&ctx);
|
|
break;
|
|
|
|
#if RFAL_FEATURE_ISO_DEP
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_INTERFACE_ISODEP: {
|
|
rfalIsoDepApduTxRxParam isoDepTxRx;
|
|
|
|
if(txDataLen > sizeof(gNfcDev.txBuf.isoDepBuf.apdu)) {
|
|
return ERR_NOMEM;
|
|
}
|
|
|
|
if(txDataLen > 0U) {
|
|
ST_MEMCPY((uint8_t*)gNfcDev.txBuf.isoDepBuf.apdu, txData, txDataLen);
|
|
}
|
|
|
|
isoDepTxRx.DID = RFAL_ISODEP_NO_DID;
|
|
isoDepTxRx.ourFSx = RFAL_ISODEP_FSX_KEEP;
|
|
isoDepTxRx.FSx = gNfcDev.activeDev->proto.isoDep.info.FSx;
|
|
isoDepTxRx.dFWT = gNfcDev.activeDev->proto.isoDep.info.dFWT;
|
|
isoDepTxRx.FWT = gNfcDev.activeDev->proto.isoDep.info.FWT;
|
|
isoDepTxRx.txBuf = &gNfcDev.txBuf.isoDepBuf;
|
|
isoDepTxRx.txBufLen = txDataLen;
|
|
isoDepTxRx.rxBuf = &gNfcDev.rxBuf.isoDepBuf;
|
|
isoDepTxRx.rxLen = &gNfcDev.rxLen;
|
|
isoDepTxRx.tmpBuf = &gNfcDev.tmpBuf.isoDepBuf;
|
|
*rxData = (uint8_t*)gNfcDev.rxBuf.isoDepBuf.apdu;
|
|
*rvdLen = (uint16_t*)&gNfcDev.rxLen;
|
|
|
|
/*******************************************************************************/
|
|
/* Trigger a RFAL ISO-DEP Transceive */
|
|
err = rfalIsoDepStartApduTransceive(isoDepTxRx);
|
|
break;
|
|
}
|
|
#endif /* RFAL_FEATURE_ISO_DEP */
|
|
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_INTERFACE_NFCDEP: {
|
|
rfalNfcDepPduTxRxParam nfcDepTxRx;
|
|
|
|
if(txDataLen > sizeof(gNfcDev.txBuf.nfcDepBuf.pdu)) {
|
|
return ERR_NOMEM;
|
|
}
|
|
|
|
if(txDataLen > 0U) {
|
|
ST_MEMCPY((uint8_t*)gNfcDev.txBuf.nfcDepBuf.pdu, txData, txDataLen);
|
|
}
|
|
|
|
nfcDepTxRx.DID = RFAL_NFCDEP_DID_KEEP;
|
|
nfcDepTxRx.FSx =
|
|
rfalNfcIsRemDevListener(gNfcDev.activeDev->type) ?
|
|
rfalNfcDepLR2FS((uint8_t)rfalNfcDepPP2LR(
|
|
gNfcDev.activeDev->proto.nfcDep.activation.Target.ATR_RES.PPt)) :
|
|
rfalNfcDepLR2FS((uint8_t)rfalNfcDepPP2LR(
|
|
gNfcDev.activeDev->proto.nfcDep.activation.Initiator.ATR_REQ.PPi));
|
|
nfcDepTxRx.dFWT = gNfcDev.activeDev->proto.nfcDep.info.dFWT;
|
|
nfcDepTxRx.FWT = gNfcDev.activeDev->proto.nfcDep.info.FWT;
|
|
nfcDepTxRx.txBuf = &gNfcDev.txBuf.nfcDepBuf;
|
|
nfcDepTxRx.txBufLen = txDataLen;
|
|
nfcDepTxRx.rxBuf = &gNfcDev.rxBuf.nfcDepBuf;
|
|
nfcDepTxRx.rxLen = &gNfcDev.rxLen;
|
|
nfcDepTxRx.tmpBuf = &gNfcDev.tmpBuf.nfcDepBuf;
|
|
*rxData = (uint8_t*)gNfcDev.rxBuf.nfcDepBuf.pdu;
|
|
*rvdLen = (uint16_t*)&gNfcDev.rxLen;
|
|
|
|
/*******************************************************************************/
|
|
/* Trigger a RFAL NFC-DEP Transceive */
|
|
err = rfalNfcDepStartPduTransceive(nfcDepTxRx);
|
|
break;
|
|
}
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
/*******************************************************************************/
|
|
default:
|
|
err = ERR_PARAM;
|
|
break;
|
|
}
|
|
|
|
/* If a transceive has succesfully started flag Data Exchange as ongoing */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.dataExErr = ERR_BUSY;
|
|
gNfcDev.state = RFAL_NFC_STATE_DATAEXCHANGE;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
ReturnCode rfalNfcDataExchangeGetStatus(void) {
|
|
/*******************************************************************************/
|
|
/* Check if it's the first frame received in Listen mode */
|
|
if(gNfcDev.state == RFAL_NFC_STATE_ACTIVATED) {
|
|
/* Continue data exchange as normal */
|
|
gNfcDev.dataExErr = ERR_BUSY;
|
|
gNfcDev.state = RFAL_NFC_STATE_DATAEXCHANGE;
|
|
|
|
/* Check if we performing in T3T CE */
|
|
if((gNfcDev.activeDev->type == RFAL_NFC_POLL_TYPE_NFCF) &&
|
|
(gNfcDev.activeDev->rfInterface == RFAL_NFC_INTERFACE_RF)) {
|
|
/* The first frame has been retrieved by rfalListenMode, flag data immediately */
|
|
/* Can only call rfalGetTransceiveStatus() after starting a transceive with rfalStartTransceive */
|
|
gNfcDev.dataExErr = ERR_NONE;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Check if we are in we have been placed to sleep, and return last error */
|
|
if(gNfcDev.state == RFAL_NFC_STATE_LISTEN_SLEEP) {
|
|
return gNfcDev.dataExErr; /* ERR_SLEEP_REQ */
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Check if Data exchange has been started */
|
|
if((gNfcDev.state != RFAL_NFC_STATE_DATAEXCHANGE) &&
|
|
(gNfcDev.state != RFAL_NFC_STATE_DATAEXCHANGE_DONE)) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
/* Check if Data exchange is still ongoing */
|
|
if(gNfcDev.dataExErr == ERR_BUSY) {
|
|
switch(gNfcDev.activeDev->rfInterface) {
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_INTERFACE_RF:
|
|
gNfcDev.dataExErr = rfalGetTransceiveStatus();
|
|
break;
|
|
|
|
#if RFAL_FEATURE_ISO_DEP
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_INTERFACE_ISODEP:
|
|
gNfcDev.dataExErr = rfalIsoDepGetApduTransceiveStatus();
|
|
break;
|
|
#endif /* RFAL_FEATURE_ISO_DEP */
|
|
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
case RFAL_NFC_INTERFACE_NFCDEP:
|
|
gNfcDev.dataExErr = rfalNfcDepGetPduTransceiveStatus();
|
|
break;
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
/*******************************************************************************/
|
|
default:
|
|
gNfcDev.dataExErr = ERR_PARAM;
|
|
break;
|
|
}
|
|
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
/*******************************************************************************/
|
|
/* If a Sleep request has been received (Listen Mode) go to sleep immediately */
|
|
if(gNfcDev.dataExErr == ERR_SLEEP_REQ) {
|
|
EXIT_ON_ERR(
|
|
gNfcDev.dataExErr,
|
|
rfalListenSleepStart(
|
|
RFAL_LM_STATE_SLEEP_A,
|
|
gNfcDev.rxBuf.rfBuf,
|
|
sizeof(gNfcDev.rxBuf.rfBuf),
|
|
&gNfcDev.rxLen));
|
|
|
|
/* If set Sleep was succesfull keep restore the Sleep request signal */
|
|
gNfcDev.dataExErr = ERR_SLEEP_REQ;
|
|
}
|
|
#endif /* RFAL_FEATURE_LISTEN_MODE */
|
|
}
|
|
|
|
return gNfcDev.dataExErr;
|
|
}
|
|
|
|
/*!
|
|
******************************************************************************
|
|
* \brief Poller Technology Detection
|
|
*
|
|
* This method implements the Technology Detection / Poll for different
|
|
* device technologies.
|
|
*
|
|
* \return ERR_NONE : Operation completed with no error
|
|
* \return ERR_BUSY : Operation ongoing
|
|
* \return ERR_XXXX : Error occurred
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
static ReturnCode rfalNfcPollTechDetetection(void) {
|
|
ReturnCode err;
|
|
|
|
err = ERR_NONE;
|
|
|
|
/* Supress warning when specific RFAL features have been disabled */
|
|
NO_WARNING(err);
|
|
|
|
/*******************************************************************************/
|
|
/* AP2P Technology Detection */
|
|
/*******************************************************************************/
|
|
if(((gNfcDev.disc.techs2Find & RFAL_NFC_POLL_TECH_AP2P) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_AP2P) != 0U)) {
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalSetMode(RFAL_MODE_POLL_ACTIVE_P2P, gNfcDev.disc.ap2pBR, gNfcDev.disc.ap2pBR));
|
|
rfalSetErrorHandling(RFAL_ERRORHANDLING_NFC);
|
|
rfalSetFDTListen(RFAL_FDT_LISTEN_AP2P_POLLER);
|
|
rfalSetFDTPoll(RFAL_TIMING_NONE);
|
|
rfalSetGT(RFAL_GT_AP2P_ADJUSTED);
|
|
EXIT_ON_ERR(err, rfalFieldOnAndStartGT()); /* Turns the Field On and starts GT timer */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(rfalIsGTExpired()) /* Wait until Guard Time is fulfilled */
|
|
{
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_AP2P;
|
|
|
|
err = rfalNfcNfcDepActivate(
|
|
gNfcDev.devList, RFAL_NFCDEP_COMM_ACTIVE, NULL, 0); /* Poll for NFC-A devices */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.techsFound |= RFAL_NFC_POLL_TECH_AP2P;
|
|
|
|
gNfcDev.devList->type = RFAL_NFC_LISTEN_TYPE_AP2P;
|
|
gNfcDev.devList->rfInterface = RFAL_NFC_INTERFACE_NFCDEP;
|
|
gNfcDev.devCnt++;
|
|
|
|
return ERR_NONE;
|
|
}
|
|
|
|
gNfcDev.isTechInit = false;
|
|
rfalFieldOff();
|
|
}
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-A Technology Detection */
|
|
/*******************************************************************************/
|
|
if(((gNfcDev.disc.techs2Find & RFAL_NFC_POLL_TECH_A) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_A) != 0U)) {
|
|
#if RFAL_FEATURE_NFCA
|
|
|
|
rfalNfcaSensRes sensRes;
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalNfcaPollerInitialize()); /* Initialize RFAL for NFC-A */
|
|
EXIT_ON_ERR(err, rfalFieldOnAndStartGT()); /* Turns the Field On and starts GT timer */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(rfalIsGTExpired()) /* Wait until Guard Time is fulfilled */
|
|
{
|
|
err = rfalNfcaPollerTechnologyDetection(
|
|
gNfcDev.disc.compMode, &sensRes); /* Poll for NFC-A devices */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.techsFound |= RFAL_NFC_POLL_TECH_A;
|
|
}
|
|
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_A;
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_NFCA */
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-B Technology Detection */
|
|
/*******************************************************************************/
|
|
if(((gNfcDev.disc.techs2Find & RFAL_NFC_POLL_TECH_B) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_B) != 0U)) {
|
|
#if RFAL_FEATURE_NFCB
|
|
|
|
rfalNfcbSensbRes sensbRes;
|
|
uint8_t sensbResLen;
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalNfcbPollerInitialize()); /* Initialize RFAL for NFC-B */
|
|
EXIT_ON_ERR(
|
|
err, rfalFieldOnAndStartGT()); /* As field is already On only starts GT timer */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(rfalIsGTExpired()) /* Wait until Guard Time is fulfilled */
|
|
{
|
|
err = rfalNfcbPollerTechnologyDetection(
|
|
gNfcDev.disc.compMode, &sensbRes, &sensbResLen); /* Poll for NFC-B devices */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.techsFound |= RFAL_NFC_POLL_TECH_B;
|
|
}
|
|
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_B;
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_NFCB */
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-F Technology Detection */
|
|
/*******************************************************************************/
|
|
if(((gNfcDev.disc.techs2Find & RFAL_NFC_POLL_TECH_F) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_F) != 0U)) {
|
|
#if RFAL_FEATURE_NFCF
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalNfcfPollerInitialize(gNfcDev.disc.nfcfBR)); /* Initialize RFAL for NFC-F */
|
|
EXIT_ON_ERR(
|
|
err, rfalFieldOnAndStartGT()); /* As field is already On only starts GT timer */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(rfalIsGTExpired()) /* Wait until Guard Time is fulfilled */
|
|
{
|
|
err = rfalNfcfPollerCheckPresence(); /* Poll for NFC-F devices */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.techsFound |= RFAL_NFC_POLL_TECH_F;
|
|
}
|
|
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_F;
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_NFCF */
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-V Technology Detection */
|
|
/*******************************************************************************/
|
|
if(((gNfcDev.disc.techs2Find & RFAL_NFC_POLL_TECH_V) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_V) != 0U)) {
|
|
#if RFAL_FEATURE_NFCV
|
|
|
|
rfalNfcvInventoryRes invRes;
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalNfcvPollerInitialize()); /* Initialize RFAL for NFC-V */
|
|
EXIT_ON_ERR(
|
|
err, rfalFieldOnAndStartGT()); /* As field is already On only starts GT timer */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(rfalIsGTExpired()) /* Wait until Guard Time is fulfilled */
|
|
{
|
|
err = rfalNfcvPollerCheckPresence(&invRes); /* Poll for NFC-V devices */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.techsFound |= RFAL_NFC_POLL_TECH_V;
|
|
}
|
|
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_V;
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_NFCV */
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* Passive Proprietary Technology ST25TB */
|
|
/*******************************************************************************/
|
|
if(((gNfcDev.disc.techs2Find & RFAL_NFC_POLL_TECH_ST25TB) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_ST25TB) != 0U)) {
|
|
#if RFAL_FEATURE_ST25TB
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalSt25tbPollerInitialize()); /* Initialize RFAL for NFC-V */
|
|
EXIT_ON_ERR(
|
|
err, rfalFieldOnAndStartGT()); /* As field is already On only starts GT timer */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(rfalIsGTExpired()) /* Wait until Guard Time is fulfilled */
|
|
{
|
|
err = rfalSt25tbPollerCheckPresence(NULL); /* Poll for ST25TB devices */
|
|
if(err == ERR_NONE) {
|
|
gNfcDev.techsFound |= RFAL_NFC_POLL_TECH_ST25TB;
|
|
}
|
|
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_ST25TB;
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_ST25TB */
|
|
}
|
|
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*!
|
|
******************************************************************************
|
|
* \brief Poller Collision Resolution
|
|
*
|
|
* This method implements the Collision Resolution on all technologies that
|
|
* have been detected before.
|
|
*
|
|
* \return ERR_NONE : Operation completed with no error
|
|
* \return ERR_BUSY : Operation ongoing
|
|
* \return ERR_XXXX : Error occurred
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
static ReturnCode rfalNfcPollCollResolution(void) {
|
|
uint8_t i;
|
|
static uint8_t devCnt;
|
|
ReturnCode err;
|
|
|
|
err = ERR_NONE;
|
|
i = 0;
|
|
|
|
/* Supress warning when specific RFAL features have been disabled */
|
|
NO_WARNING(err);
|
|
NO_WARNING(devCnt);
|
|
NO_WARNING(i);
|
|
|
|
/* Check if device limit has been reached */
|
|
if(gNfcDev.devCnt >= gNfcDev.disc.devLimit) {
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*******************************************************************************/
|
|
/* NFC-A Collision Resolution */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCA
|
|
if(((gNfcDev.techsFound & RFAL_NFC_POLL_TECH_A) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_A) !=
|
|
0U)) /* If a NFC-A device was found/detected, perform Collision Resolution */
|
|
{
|
|
static rfalNfcaListenDevice nfcaDevList[RFAL_NFC_MAX_DEVICES];
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalNfcaPollerInitialize()); /* Initialize RFAL for NFC-A */
|
|
EXIT_ON_ERR(err, rfalFieldOnAndStartGT()); /* Turns the Field On and starts GT timer */
|
|
|
|
gNfcDev.isTechInit = true; /* Technology has been initialized */
|
|
gNfcDev.isOperOngoing = false; /* No operation currently ongoing */
|
|
}
|
|
|
|
if(!rfalIsGTExpired()) {
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
if(!gNfcDev.isOperOngoing) {
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalNfcaPollerStartFullCollisionResolution(
|
|
gNfcDev.disc.compMode,
|
|
(gNfcDev.disc.devLimit - gNfcDev.devCnt),
|
|
nfcaDevList,
|
|
&devCnt));
|
|
|
|
gNfcDev.isOperOngoing = true;
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
err = rfalNfcaPollerGetFullCollisionResolutionStatus();
|
|
if(err != ERR_BUSY) {
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_A;
|
|
|
|
if((err == ERR_NONE) && (devCnt != 0U)) {
|
|
for(i = 0; i < devCnt;
|
|
i++) /* Copy devices found form local Nfca list into global device list */
|
|
{
|
|
gNfcDev.devList[gNfcDev.devCnt].type = RFAL_NFC_LISTEN_TYPE_NFCA;
|
|
gNfcDev.devList[gNfcDev.devCnt].dev.nfca = nfcaDevList[i];
|
|
gNfcDev.devCnt++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
}
|
|
#endif /* RFAL_FEATURE_NFCA */
|
|
|
|
/*******************************************************************************/
|
|
/* NFC-B Collision Resolution */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCB
|
|
if(((gNfcDev.techsFound & RFAL_NFC_POLL_TECH_B) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_B) !=
|
|
0U)) /* If a NFC-B device was found/detected, perform Collision Resolution */
|
|
{
|
|
rfalNfcbListenDevice nfcbDevList[RFAL_NFC_MAX_DEVICES];
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalNfcbPollerInitialize()); /* Initialize RFAL for NFC-B */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalFieldOnAndStartGT()); /* Ensure GT again as other technologies have also been polled */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(!rfalIsGTExpired()) {
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
devCnt = 0;
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_B;
|
|
|
|
err = rfalNfcbPollerCollisionResolution(
|
|
gNfcDev.disc.compMode, (gNfcDev.disc.devLimit - gNfcDev.devCnt), nfcbDevList, &devCnt);
|
|
if((err == ERR_NONE) && (devCnt != 0U)) {
|
|
for(i = 0; i < devCnt;
|
|
i++) /* Copy devices found form local Nfcb list into global device list */
|
|
{
|
|
gNfcDev.devList[gNfcDev.devCnt].type = RFAL_NFC_LISTEN_TYPE_NFCB;
|
|
gNfcDev.devList[gNfcDev.devCnt].dev.nfcb = nfcbDevList[i];
|
|
gNfcDev.devCnt++;
|
|
}
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
}
|
|
#endif /* RFAL_FEATURE_NFCB*/
|
|
|
|
/*******************************************************************************/
|
|
/* NFC-F Collision Resolution */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCF
|
|
if(((gNfcDev.techsFound & RFAL_NFC_POLL_TECH_F) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_F) !=
|
|
0U)) /* If a NFC-F device was found/detected, perform Collision Resolution */
|
|
{
|
|
rfalNfcfListenDevice nfcfDevList[RFAL_NFC_MAX_DEVICES];
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalNfcfPollerInitialize(gNfcDev.disc.nfcfBR)); /* Initialize RFAL for NFC-F */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalFieldOnAndStartGT()); /* Ensure GT again as other technologies have also been polled */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(!rfalIsGTExpired()) {
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
devCnt = 0;
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_F;
|
|
|
|
err = rfalNfcfPollerCollisionResolution(
|
|
gNfcDev.disc.compMode, (gNfcDev.disc.devLimit - gNfcDev.devCnt), nfcfDevList, &devCnt);
|
|
if((err == ERR_NONE) && (devCnt != 0U)) {
|
|
for(i = 0; i < devCnt;
|
|
i++) /* Copy devices found form local Nfcf list into global device list */
|
|
{
|
|
gNfcDev.devList[gNfcDev.devCnt].type = RFAL_NFC_LISTEN_TYPE_NFCF;
|
|
gNfcDev.devList[gNfcDev.devCnt].dev.nfcf = nfcfDevList[i];
|
|
gNfcDev.devCnt++;
|
|
}
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
}
|
|
#endif /* RFAL_FEATURE_NFCF */
|
|
|
|
/*******************************************************************************/
|
|
/* NFC-V Collision Resolution */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCV
|
|
if(((gNfcDev.techsFound & RFAL_NFC_POLL_TECH_V) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_V) !=
|
|
0U)) /* If a NFC-V device was found/detected, perform Collision Resolution */
|
|
{
|
|
rfalNfcvListenDevice nfcvDevList[RFAL_NFC_MAX_DEVICES];
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalNfcvPollerInitialize()); /* Initialize RFAL for NFC-V */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalFieldOnAndStartGT()); /* Ensure GT again as other technologies have also been polled */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(!rfalIsGTExpired()) {
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
devCnt = 0;
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_V;
|
|
|
|
err = rfalNfcvPollerCollisionResolution(
|
|
RFAL_COMPLIANCE_MODE_NFC,
|
|
(gNfcDev.disc.devLimit - gNfcDev.devCnt),
|
|
nfcvDevList,
|
|
&devCnt);
|
|
if((err == ERR_NONE) && (devCnt != 0U)) {
|
|
for(i = 0; i < devCnt;
|
|
i++) /* Copy devices found form local Nfcf list into global device list */
|
|
{
|
|
gNfcDev.devList[gNfcDev.devCnt].type = RFAL_NFC_LISTEN_TYPE_NFCV;
|
|
gNfcDev.devList[gNfcDev.devCnt].dev.nfcv = nfcvDevList[i];
|
|
gNfcDev.devCnt++;
|
|
}
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
}
|
|
#endif /* RFAL_FEATURE_NFCV */
|
|
|
|
/*******************************************************************************/
|
|
/* ST25TB Collision Resolution */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_ST25TB
|
|
if(((gNfcDev.techsFound & RFAL_NFC_POLL_TECH_ST25TB) != 0U) &&
|
|
((gNfcDev.techs2do & RFAL_NFC_POLL_TECH_ST25TB) !=
|
|
0U)) /* If a ST25TB device was found/detected, perform Collision Resolution */
|
|
{
|
|
rfalSt25tbListenDevice st25tbDevList[RFAL_NFC_MAX_DEVICES];
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
EXIT_ON_ERR(err, rfalSt25tbPollerInitialize()); /* Initialize RFAL for ST25TB */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalFieldOnAndStartGT()); /* Ensure GT again as other technologies have also been polled */
|
|
gNfcDev.isTechInit = true;
|
|
}
|
|
|
|
if(!rfalIsGTExpired()) {
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
devCnt = 0;
|
|
gNfcDev.isTechInit = false;
|
|
gNfcDev.techs2do &= ~RFAL_NFC_POLL_TECH_ST25TB;
|
|
|
|
err = rfalSt25tbPollerCollisionResolution(
|
|
(gNfcDev.disc.devLimit - gNfcDev.devCnt), st25tbDevList, &devCnt);
|
|
if((err == ERR_NONE) && (devCnt != 0U)) {
|
|
for(i = 0; i < devCnt;
|
|
i++) /* Copy devices found form local Nfcf list into global device list */
|
|
{
|
|
gNfcDev.devList[gNfcDev.devCnt].type = RFAL_NFC_LISTEN_TYPE_ST25TB;
|
|
gNfcDev.devList[gNfcDev.devCnt].dev.st25tb = st25tbDevList[i];
|
|
gNfcDev.devCnt++;
|
|
}
|
|
}
|
|
|
|
return ERR_BUSY;
|
|
}
|
|
#endif /* RFAL_FEATURE_ST25TB */
|
|
|
|
return ERR_NONE; /* All technologies have been performed */
|
|
}
|
|
|
|
/*!
|
|
******************************************************************************
|
|
* \brief Poller Activation
|
|
*
|
|
* This method Activates a given device according to it's type and
|
|
* protocols supported
|
|
*
|
|
* \param[in] devIt : device's position on the list to be activated
|
|
*
|
|
* \return ERR_NONE : Operation completed with no error
|
|
* \return ERR_BUSY : Operation ongoing
|
|
* \return ERR_XXXX : Error occurred
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
static ReturnCode rfalNfcPollActivation(uint8_t devIt) {
|
|
ReturnCode err;
|
|
|
|
err = ERR_NONE;
|
|
|
|
/* Supress warning when specific RFAL features have been disabled */
|
|
NO_WARNING(err);
|
|
|
|
if(devIt > gNfcDev.devCnt) {
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
switch(gNfcDev.devList[devIt].type) {
|
|
/*******************************************************************************/
|
|
/* AP2P Activation */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
case RFAL_NFC_LISTEN_TYPE_AP2P:
|
|
/* Activation has already been perfomed (ATR_REQ) */
|
|
|
|
gNfcDev.devList[devIt].nfcid =
|
|
gNfcDev.devList[devIt].proto.nfcDep.activation.Target.ATR_RES.NFCID3;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_NFCDEP_NFCID3_LEN;
|
|
break;
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-A Activation */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCA
|
|
case RFAL_NFC_LISTEN_TYPE_NFCA:
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
rfalNfcaPollerInitialize();
|
|
gNfcDev.isTechInit = true;
|
|
gNfcDev.isOperOngoing = false;
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
if(gNfcDev.devList[devIt].dev.nfca.isSleep) /* Check if desired device is in Sleep */
|
|
{
|
|
rfalNfcaSensRes sensRes;
|
|
rfalNfcaSelRes selRes;
|
|
|
|
if(!gNfcDev.isOperOngoing) {
|
|
/* Wake up all cards */
|
|
EXIT_ON_ERR(
|
|
err, rfalNfcaPollerCheckPresence(RFAL_14443A_SHORTFRAME_CMD_WUPA, &sensRes));
|
|
gNfcDev.isOperOngoing = true;
|
|
} else {
|
|
/* Select specific device */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalNfcaPollerSelect(
|
|
gNfcDev.devList[devIt].dev.nfca.nfcId1,
|
|
gNfcDev.devList[devIt].dev.nfca.nfcId1Len,
|
|
&selRes));
|
|
gNfcDev.devList[devIt].dev.nfca.isSleep = false;
|
|
gNfcDev.isOperOngoing = false;
|
|
}
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
/* Set NFCID */
|
|
gNfcDev.devList[devIt].nfcid = gNfcDev.devList[devIt].dev.nfca.nfcId1;
|
|
gNfcDev.devList[devIt].nfcidLen = gNfcDev.devList[devIt].dev.nfca.nfcId1Len;
|
|
|
|
/*******************************************************************************/
|
|
/* Perform protocol specific activation */
|
|
switch(gNfcDev.devList[devIt].dev.nfca.type) {
|
|
/*******************************************************************************/
|
|
case RFAL_NFCA_T1T:
|
|
|
|
/* No further activation needed for T1T (RID already performed) */
|
|
|
|
gNfcDev.devList[devIt].nfcid = gNfcDev.devList[devIt].dev.nfca.ridRes.uid;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_T1T_UID_LEN;
|
|
|
|
gNfcDev.devList[devIt].rfInterface = RFAL_NFC_INTERFACE_RF;
|
|
break;
|
|
|
|
case RFAL_NFCA_T2T:
|
|
|
|
/* No further activation needed for a T2T */
|
|
|
|
gNfcDev.devList[devIt].rfInterface = RFAL_NFC_INTERFACE_RF;
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFCA_T4T: /* Device supports ISO-DEP */
|
|
|
|
#if RFAL_FEATURE_ISO_DEP && RFAL_FEATURE_ISO_DEP_POLL
|
|
if(!gNfcDev.isOperOngoing) {
|
|
/* Perform ISO-DEP (ISO14443-4) activation: RATS and PPS if supported */
|
|
rfalIsoDepInitialize();
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalIsoDepPollAStartActivation(
|
|
(rfalIsoDepFSxI)RFAL_ISODEP_FSDI_DEFAULT,
|
|
RFAL_ISODEP_NO_DID,
|
|
gNfcDev.disc.maxBR,
|
|
&gNfcDev.devList[devIt].proto.isoDep));
|
|
|
|
gNfcDev.isOperOngoing = true;
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
err = rfalIsoDepPollAGetActivationStatus();
|
|
if(err != ERR_NONE) {
|
|
return err;
|
|
}
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_ISODEP; /* NFC-A T4T device activated */
|
|
#else
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_RF; /* No ISO-DEP supported activate using RF interface */
|
|
#endif /* RFAL_FEATURE_ISO_DEP_POLL */
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_NFCA_T4T_NFCDEP: /* Device supports both T4T and NFC-DEP */
|
|
case RFAL_NFCA_NFCDEP: /* Device supports NFC-DEP */
|
|
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
/* Perform NFC-DEP (P2P) activation: ATR and PSL if supported */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalNfcNfcDepActivate(&gNfcDev.devList[devIt], RFAL_NFCDEP_COMM_PASSIVE, NULL, 0));
|
|
|
|
gNfcDev.devList[devIt].nfcid =
|
|
gNfcDev.devList[devIt].proto.nfcDep.activation.Target.ATR_RES.NFCID3;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_NFCDEP_NFCID3_LEN;
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_NFCDEP; /* NFC-A P2P device activated */
|
|
#else
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_RF; /* No NFC-DEP supported activate using RF interface */
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
break;
|
|
|
|
/*******************************************************************************/
|
|
default:
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
break;
|
|
#endif /* RFAL_FEATURE_NFCA */
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-B Activation */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCB
|
|
case RFAL_NFC_LISTEN_TYPE_NFCB:
|
|
|
|
if(!gNfcDev.isTechInit) {
|
|
rfalNfcbPollerInitialize();
|
|
gNfcDev.isTechInit = true;
|
|
gNfcDev.isOperOngoing = false;
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
if(gNfcDev.devList[devIt].dev.nfcb.isSleep) /* Check if desired device is in Sleep */
|
|
{
|
|
rfalNfcbSensbRes sensbRes;
|
|
uint8_t sensbResLen;
|
|
|
|
/* Wake up all cards. SENSB_RES may return collision but the NFCID0 is available to explicitly select NFC-B card via ATTRIB; so error will be ignored here */
|
|
rfalNfcbPollerCheckPresence(
|
|
RFAL_NFCB_SENS_CMD_ALLB_REQ, RFAL_NFCB_SLOT_NUM_1, &sensbRes, &sensbResLen);
|
|
}
|
|
|
|
/* Set NFCID */
|
|
gNfcDev.devList[devIt].nfcid = gNfcDev.devList[devIt].dev.nfcb.sensbRes.nfcid0;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_NFCB_NFCID0_LEN;
|
|
|
|
#if RFAL_FEATURE_ISO_DEP && RFAL_FEATURE_ISO_DEP_POLL
|
|
/* Check if device supports ISO-DEP (ISO14443-4) */
|
|
if((gNfcDev.devList[devIt].dev.nfcb.sensbRes.protInfo.FsciProType &
|
|
RFAL_NFCB_SENSB_RES_PROTO_ISO_MASK) != 0U) {
|
|
if(!gNfcDev.isOperOngoing) {
|
|
rfalIsoDepInitialize();
|
|
/* Perform ISO-DEP (ISO14443-4) activation: ATTRIB */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalIsoDepPollBStartActivation(
|
|
(rfalIsoDepFSxI)RFAL_ISODEP_FSDI_DEFAULT,
|
|
RFAL_ISODEP_NO_DID,
|
|
gNfcDev.disc.maxBR,
|
|
0x00,
|
|
&gNfcDev.devList[devIt].dev.nfcb,
|
|
NULL,
|
|
0,
|
|
&gNfcDev.devList[devIt].proto.isoDep));
|
|
|
|
gNfcDev.isOperOngoing = true;
|
|
return ERR_BUSY;
|
|
}
|
|
|
|
err = rfalIsoDepPollBGetActivationStatus();
|
|
if(err != ERR_NONE) {
|
|
return err;
|
|
}
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_ISODEP; /* NFC-B T4T device activated */
|
|
break;
|
|
}
|
|
|
|
#endif /* RFAL_FEATURE_ISO_DEP_POLL */
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_RF; /* NFC-B device activated */
|
|
break;
|
|
|
|
#endif /* RFAL_FEATURE_NFCB */
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-F Activation */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCF
|
|
case RFAL_NFC_LISTEN_TYPE_NFCF:
|
|
|
|
rfalNfcfPollerInitialize(gNfcDev.disc.nfcfBR);
|
|
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
if(rfalNfcfIsNfcDepSupported(&gNfcDev.devList[devIt].dev.nfcf)) {
|
|
/* Perform NFC-DEP (P2P) activation: ATR and PSL if supported */
|
|
EXIT_ON_ERR(
|
|
err,
|
|
rfalNfcNfcDepActivate(&gNfcDev.devList[devIt], RFAL_NFCDEP_COMM_PASSIVE, NULL, 0));
|
|
|
|
/* Set NFCID */
|
|
gNfcDev.devList[devIt].nfcid =
|
|
gNfcDev.devList[devIt].proto.nfcDep.activation.Target.ATR_RES.NFCID3;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_NFCDEP_NFCID3_LEN;
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_NFCDEP; /* NFC-F P2P device activated */
|
|
break;
|
|
}
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
/* Set NFCID */
|
|
gNfcDev.devList[devIt].nfcid = gNfcDev.devList[devIt].dev.nfcf.sensfRes.NFCID2;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_NFCF_NFCID2_LEN;
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_RF; /* NFC-F T3T device activated */
|
|
break;
|
|
#endif /* RFAL_FEATURE_NFCF */
|
|
|
|
/*******************************************************************************/
|
|
/* Passive NFC-V Activation */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFCV
|
|
case RFAL_NFC_LISTEN_TYPE_NFCV:
|
|
|
|
rfalNfcvPollerInitialize();
|
|
|
|
/* No specific activation needed for a T5T */
|
|
|
|
/* Set NFCID */
|
|
gNfcDev.devList[devIt].nfcid = gNfcDev.devList[devIt].dev.nfcv.InvRes.UID;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_NFCV_UID_LEN;
|
|
|
|
gNfcDev.devList[devIt].rfInterface =
|
|
RFAL_NFC_INTERFACE_RF; /* NFC-V T5T device activated */
|
|
break;
|
|
#endif /* RFAL_FEATURE_NFCV */
|
|
|
|
/*******************************************************************************/
|
|
/* Passive ST25TB Activation */
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_ST25TB
|
|
case RFAL_NFC_LISTEN_TYPE_ST25TB:
|
|
|
|
rfalSt25tbPollerInitialize();
|
|
|
|
/* No specific activation needed for a ST25TB */
|
|
|
|
/* Set NFCID */
|
|
gNfcDev.devList[devIt].nfcid = gNfcDev.devList[devIt].dev.st25tb.UID;
|
|
gNfcDev.devList[devIt].nfcidLen = RFAL_ST25TB_UID_LEN;
|
|
|
|
gNfcDev.devList[devIt].rfInterface = RFAL_NFC_INTERFACE_RF; /* ST25TB device activated */
|
|
break;
|
|
#endif /* RFAL_FEATURE_ST25TB */
|
|
|
|
/*******************************************************************************/
|
|
default:
|
|
return ERR_WRONG_STATE;
|
|
}
|
|
|
|
gNfcDev.activeDev = &gNfcDev.devList[devIt]; /* Assign active device to be used further on */
|
|
return ERR_NONE;
|
|
}
|
|
|
|
/*!
|
|
******************************************************************************
|
|
* \brief Listener Activation
|
|
*
|
|
* This method handles the listen mode Activation according to the different
|
|
* protocols the Reader/Initiator performs
|
|
*
|
|
* \return ERR_NONE : Operation completed with no error
|
|
* \return ERR_BUSY : Operation ongoing
|
|
* \return ERR_PROTO : Unexpected frame received
|
|
* \return ERR_XXXX : Error occurred
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
static ReturnCode rfalNfcListenActivation(void) {
|
|
bool isDataRcvd;
|
|
ReturnCode ret;
|
|
rfalLmState lmSt;
|
|
rfalBitRate bitRate;
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
uint8_t hdrLen;
|
|
|
|
/* Set the header length in NFC-A */
|
|
hdrLen = (RFAL_NFCDEP_SB_LEN + RFAL_NFCDEP_LEN_LEN);
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
lmSt = rfalListenGetState(&isDataRcvd, &bitRate);
|
|
|
|
switch(lmSt) {
|
|
#if RFAL_FEATURE_NFCA
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_ACTIVE_A: /* NFC-A CE activation */
|
|
case RFAL_LM_STATE_ACTIVE_Ax:
|
|
|
|
if(isDataRcvd) /* Check if Reader/Initator has sent some data */
|
|
{
|
|
/* Check if received data is a Sleep request */
|
|
if(rfalNfcaListenerIsSleepReq(
|
|
gNfcDev.rxBuf.rfBuf,
|
|
rfalConvBitsToBytes(gNfcDev.rxLen))) /* Check if received data is a SLP_REQ */
|
|
{
|
|
/* Set the Listen Mode in Sleep state */
|
|
EXIT_ON_ERR(
|
|
ret,
|
|
rfalListenSleepStart(
|
|
RFAL_LM_STATE_SLEEP_A,
|
|
gNfcDev.rxBuf.rfBuf,
|
|
sizeof(gNfcDev.rxBuf.rfBuf),
|
|
&gNfcDev.rxLen));
|
|
}
|
|
|
|
else if(gNfcDev.disc.activate_after_sak) {
|
|
gNfcDev.devList->type = RFAL_NFC_POLL_TYPE_NFCA;
|
|
rfalListenSetState(RFAL_LM_STATE_ACTIVE_A);
|
|
return ERR_NONE;
|
|
}
|
|
#if RFAL_FEATURE_ISO_DEP && RFAL_FEATURE_ISO_DEP_LISTEN
|
|
/* Check if received data is a valid RATS */
|
|
else if(rfalIsoDepIsRats(
|
|
gNfcDev.rxBuf.rfBuf, (uint8_t)rfalConvBitsToBytes(gNfcDev.rxLen))) {
|
|
rfalIsoDepAtsParam atsParam;
|
|
rfalIsoDepListenActvParam rxParam;
|
|
|
|
/* Set ATS parameters */
|
|
atsParam.fsci = (uint8_t)RFAL_ISODEP_DEFAULT_FSCI;
|
|
atsParam.fwi = RFAL_ISODEP_DEFAULT_FWI;
|
|
atsParam.sfgi = RFAL_ISODEP_DEFAULT_SFGI;
|
|
atsParam.didSupport = false;
|
|
atsParam.ta = RFAL_ISODEP_ATS_TA_SAME_D;
|
|
atsParam.hb = NULL;
|
|
atsParam.hbLen = 0;
|
|
|
|
/* Set Rx parameters */
|
|
rxParam.rxBuf =
|
|
(rfalIsoDepBufFormat*)&gNfcDev.rxBuf
|
|
.isoDepBuf; /* PRQA S 0310 # MISRA 11.3 - Intentional safe cast to avoiding large buffer duplication */
|
|
rxParam.rxLen = &gNfcDev.rxLen;
|
|
rxParam.isoDepDev = &gNfcDev.devList->proto.isoDep;
|
|
rxParam.isRxChaining = &gNfcDev.isRxChaining;
|
|
|
|
rfalListenSetState(RFAL_LM_STATE_CARDEMU_4A); /* Set next state CE T4T */
|
|
rfalIsoDepInitialize(); /* Initialize ISO-DEP layer to handle ISO14443-a activation / RATS */
|
|
|
|
/* Set ISO-DEP layer to digest RATS and handle activation */
|
|
EXIT_ON_ERR(
|
|
ret,
|
|
rfalIsoDepListenStartActivation(
|
|
&atsParam, NULL, gNfcDev.rxBuf.rfBuf, gNfcDev.rxLen, rxParam));
|
|
}
|
|
#endif /* RFAL_FEATURE_ISO_DEP_LISTEN */
|
|
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
|
|
/* Check if received data is a valid ATR_REQ */
|
|
else if(rfalNfcDepIsAtrReq(
|
|
&gNfcDev.rxBuf.rfBuf[hdrLen],
|
|
(rfalConvBitsToBytes(gNfcDev.rxLen) - hdrLen),
|
|
gNfcDev.devList->nfcid)) {
|
|
gNfcDev.devList->type = RFAL_NFC_POLL_TYPE_NFCA;
|
|
EXIT_ON_ERR(
|
|
ret,
|
|
rfalNfcNfcDepActivate(
|
|
gNfcDev.devList,
|
|
RFAL_NFCDEP_COMM_PASSIVE,
|
|
&gNfcDev.rxBuf.rfBuf[hdrLen],
|
|
(rfalConvBitsToBytes(gNfcDev.rxLen) - hdrLen)));
|
|
}
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
else {
|
|
return ERR_PROTO;
|
|
}
|
|
}
|
|
return ERR_BUSY;
|
|
|
|
#endif /* RFAL_FEATURE_NFCA */
|
|
|
|
#if RFAL_FEATURE_ISO_DEP && RFAL_FEATURE_ISO_DEP_LISTEN
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_CARDEMU_4A: /* T4T ISO-DEP activation */
|
|
|
|
ret = rfalIsoDepListenGetActivationStatus();
|
|
if(ret == ERR_NONE) {
|
|
gNfcDev.devList->type = RFAL_NFC_POLL_TYPE_NFCA;
|
|
gNfcDev.devList->rfInterface = RFAL_NFC_INTERFACE_ISODEP;
|
|
gNfcDev.devList->nfcid = NULL;
|
|
gNfcDev.devList->nfcidLen = 0;
|
|
}
|
|
return ret;
|
|
#endif /* RFAL_FEATURE_ISO_DEP_LISTEN */
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_READY_F: /* NFC-F CE activation */
|
|
|
|
if(isDataRcvd) /* Wait for the first received data */
|
|
{
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
/* Set the header length in NFC-F */
|
|
hdrLen = RFAL_NFCDEP_LEN_LEN;
|
|
|
|
if(rfalNfcDepIsAtrReq(
|
|
&gNfcDev.rxBuf.rfBuf[hdrLen],
|
|
(rfalConvBitsToBytes(gNfcDev.rxLen) - hdrLen),
|
|
gNfcDev.devList->nfcid)) {
|
|
gNfcDev.devList->type = RFAL_NFC_POLL_TYPE_NFCF;
|
|
EXIT_ON_ERR(
|
|
ret,
|
|
rfalNfcNfcDepActivate(
|
|
gNfcDev.devList,
|
|
RFAL_NFCDEP_COMM_PASSIVE,
|
|
&gNfcDev.rxBuf.rfBuf[hdrLen],
|
|
(rfalConvBitsToBytes(gNfcDev.rxLen) - hdrLen)));
|
|
} else
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
{
|
|
rfalListenSetState(
|
|
RFAL_LM_STATE_CARDEMU_3); /* First data already received - set T3T CE */
|
|
}
|
|
}
|
|
return ERR_BUSY;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_CARDEMU_3: /* T3T activated */
|
|
|
|
gNfcDev.devList->type = RFAL_NFC_POLL_TYPE_NFCF;
|
|
gNfcDev.devList->rfInterface = RFAL_NFC_INTERFACE_RF;
|
|
gNfcDev.devList->nfcid = NULL;
|
|
gNfcDev.devList->nfcidLen = 0;
|
|
|
|
return ERR_NONE;
|
|
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_TARGET_A: /* NFC-DEP activation */
|
|
case RFAL_LM_STATE_TARGET_F:
|
|
|
|
ret = rfalNfcDepListenGetActivationStatus();
|
|
if(ret == ERR_NONE) {
|
|
gNfcDev.devList->rfInterface = RFAL_NFC_INTERFACE_NFCDEP;
|
|
gNfcDev.devList->nfcidLen = RFAL_NFCDEP_NFCID3_LEN;
|
|
}
|
|
return ret;
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_IDLE: /* AP2P activation */
|
|
if(isDataRcvd) /* Check if Reader/Initator has sent some data */
|
|
{
|
|
if((gNfcDev.lmMask & RFAL_LM_MASK_ACTIVE_P2P) != 0U) /* Check if AP2P is enabled */
|
|
{
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
/* Calculate the header length in NFC-A or NFC-F mode*/
|
|
hdrLen =
|
|
((bitRate == RFAL_BR_106) ? (RFAL_NFCDEP_SB_LEN + RFAL_NFCDEP_LEN_LEN) :
|
|
RFAL_NFCDEP_LEN_LEN);
|
|
|
|
if(rfalNfcDepIsAtrReq(
|
|
&gNfcDev.rxBuf.rfBuf[hdrLen],
|
|
(rfalConvBitsToBytes(gNfcDev.rxLen) - hdrLen),
|
|
NULL)) {
|
|
gNfcDev.devList->type = RFAL_NFC_POLL_TYPE_AP2P;
|
|
rfalSetMode((RFAL_MODE_LISTEN_ACTIVE_P2P), bitRate, bitRate);
|
|
EXIT_ON_ERR(
|
|
ret,
|
|
rfalNfcNfcDepActivate(
|
|
gNfcDev.devList,
|
|
RFAL_NFCDEP_COMM_ACTIVE,
|
|
&gNfcDev.rxBuf.rfBuf[hdrLen],
|
|
(rfalConvBitsToBytes(gNfcDev.rxLen) - hdrLen)));
|
|
} else
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
{
|
|
return ERR_PROTO;
|
|
}
|
|
}
|
|
}
|
|
return ERR_BUSY;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_READY_A:
|
|
case RFAL_LM_STATE_READY_Ax:
|
|
case RFAL_LM_STATE_SLEEP_A:
|
|
case RFAL_LM_STATE_SLEEP_AF:
|
|
return ERR_BUSY;
|
|
|
|
/*******************************************************************************/
|
|
case RFAL_LM_STATE_POWER_OFF:
|
|
return ERR_LINK_LOSS;
|
|
|
|
default: /* Wait for activation */
|
|
break;
|
|
}
|
|
|
|
return ERR_INTERNAL;
|
|
}
|
|
#endif /* RFAL_FEATURE_LISTEN_MODE */
|
|
|
|
/*!
|
|
******************************************************************************
|
|
* \brief Poller NFC DEP Activate
|
|
*
|
|
* This method performs NFC-DEP Activation
|
|
*
|
|
* \param[in] device : device info
|
|
* \param[in] commMode : communication mode (Passive/Active)
|
|
* \param[in] atrReq : received ATR_REQ
|
|
* \param[in] atrReqLen : received ATR_REQ size
|
|
*
|
|
* \return ERR_NONE : Operation completed with no error
|
|
* \return ERR_BUSY : Operation ongoing
|
|
* \return ERR_XXXX : Error occurred
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
static ReturnCode rfalNfcNfcDepActivate(
|
|
rfalNfcDevice* device,
|
|
rfalNfcDepCommMode commMode,
|
|
const uint8_t* atrReq,
|
|
uint16_t atrReqLen) {
|
|
rfalNfcDepAtrParam initParam;
|
|
|
|
/* Supress warnings if Listen mode is disabled */
|
|
NO_WARNING(atrReq);
|
|
NO_WARNING(atrReqLen);
|
|
|
|
/* If we are in Poll mode */
|
|
if(rfalNfcIsRemDevListener(device->type)) {
|
|
/*******************************************************************************/
|
|
/* If Passive F use the NFCID2 retrieved from SENSF */
|
|
if(device->type == RFAL_NFC_LISTEN_TYPE_NFCF) {
|
|
initParam.nfcid = device->dev.nfcf.sensfRes.NFCID2;
|
|
initParam.nfcidLen = RFAL_NFCF_NFCID2_LEN;
|
|
} else {
|
|
initParam.nfcid = gNfcDev.disc.nfcid3;
|
|
initParam.nfcidLen = RFAL_NFCDEP_NFCID3_LEN;
|
|
}
|
|
|
|
initParam.BS = RFAL_NFCDEP_Bx_NO_HIGH_BR;
|
|
initParam.BR = RFAL_NFCDEP_Bx_NO_HIGH_BR;
|
|
initParam.DID = RFAL_NFCDEP_DID_NO;
|
|
initParam.NAD = RFAL_NFCDEP_NAD_NO;
|
|
initParam.LR = RFAL_NFCDEP_LR_254;
|
|
initParam.GB = gNfcDev.disc.GB;
|
|
initParam.GBLen = gNfcDev.disc.GBLen;
|
|
initParam.commMode = commMode;
|
|
initParam.operParam =
|
|
(RFAL_NFCDEP_OPER_FULL_MI_EN | RFAL_NFCDEP_OPER_EMPTY_DEP_DIS |
|
|
RFAL_NFCDEP_OPER_ATN_EN | RFAL_NFCDEP_OPER_RTOX_REQ_EN);
|
|
|
|
rfalNfcDepInitialize();
|
|
/* Perform NFC-DEP (P2P) activation: ATR and PSL if supported */
|
|
return rfalNfcDepInitiatorHandleActivation(
|
|
&initParam, gNfcDev.disc.maxBR, &device->proto.nfcDep);
|
|
}
|
|
/* If we are in Listen mode */
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
else if(rfalNfcIsRemDevPoller(device->type)) {
|
|
rfalNfcDepListenActvParam actvParams;
|
|
rfalNfcDepTargetParam targetParam;
|
|
|
|
ST_MEMCPY(targetParam.nfcid3, (uint8_t*)gNfcDev.disc.nfcid3, RFAL_NFCDEP_NFCID3_LEN);
|
|
targetParam.bst = RFAL_NFCDEP_Bx_NO_HIGH_BR;
|
|
targetParam.brt = RFAL_NFCDEP_Bx_NO_HIGH_BR;
|
|
targetParam.to = RFAL_NFCDEP_WT_TRG_MAX_L13; /* [LLCP] 1.3 6.2.1 */
|
|
targetParam.ppt = rfalNfcDepLR2PP(RFAL_NFCDEP_LR_254);
|
|
if(gNfcDev.disc.GBLen >= RFAL_NFCDEP_GB_MAX_LEN) {
|
|
return ERR_PARAM;
|
|
}
|
|
targetParam.GBtLen = gNfcDev.disc.GBLen;
|
|
if(gNfcDev.disc.GBLen > 0U) {
|
|
ST_MEMCPY(targetParam.GBt, gNfcDev.disc.GB, gNfcDev.disc.GBLen);
|
|
}
|
|
targetParam.operParam =
|
|
(RFAL_NFCDEP_OPER_FULL_MI_EN | RFAL_NFCDEP_OPER_EMPTY_DEP_DIS |
|
|
RFAL_NFCDEP_OPER_ATN_EN | RFAL_NFCDEP_OPER_RTOX_REQ_EN);
|
|
targetParam.commMode = commMode;
|
|
|
|
/* Set activation buffer (including header) for NFC-DEP */
|
|
actvParams.rxBuf =
|
|
(rfalNfcDepBufFormat*)&gNfcDev.rxBuf
|
|
.nfcDepBuf; /* PRQA S 0310 # MISRA 11.3 - Intentional safe cast to avoiding large buffer duplication */
|
|
actvParams.rxLen = &gNfcDev.rxLen;
|
|
actvParams.isRxChaining = &gNfcDev.isRxChaining;
|
|
actvParams.nfcDepDev = &gNfcDev.devList->proto.nfcDep;
|
|
|
|
rfalListenSetState(
|
|
((device->type == RFAL_NFC_POLL_TYPE_NFCA) ? RFAL_LM_STATE_TARGET_A :
|
|
RFAL_LM_STATE_TARGET_F));
|
|
|
|
rfalNfcDepInitialize();
|
|
/* Perform NFC-DEP (P2P) activation: send ATR_RES and handle activation */
|
|
return rfalNfcDepListenStartActivation(&targetParam, atrReq, atrReqLen, actvParams);
|
|
}
|
|
#endif /* RFAL_FEATURE_LISTEN_MODE */
|
|
|
|
else {
|
|
return ERR_INTERNAL;
|
|
}
|
|
}
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
/*!
|
|
******************************************************************************
|
|
* \brief Poller NFC Deactivate
|
|
*
|
|
* This method Deactivates the device if a deactivation procedure exists
|
|
*
|
|
* \return ERR_NONE : Operation completed with no error
|
|
* \return ERR_BUSY : Operation ongoing
|
|
* \return ERR_XXXX : Error occurred
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
static ReturnCode rfalNfcDeactivation(void) {
|
|
/* Check if a device has been activated */
|
|
if(gNfcDev.activeDev != NULL) {
|
|
if(rfalNfcIsRemDevListener(
|
|
gNfcDev.activeDev->type)) /* Listen mode no additional deactivation to be performed*/
|
|
{
|
|
#ifndef RFAL_NFC_SKIP_DEACT
|
|
switch(gNfcDev.activeDev->rfInterface) {
|
|
/*******************************************************************************/
|
|
case RFAL_NFC_INTERFACE_RF:
|
|
break; /* No specific deactivation to be performed */
|
|
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_ISO_DEP && RFAL_FEATURE_ISO_DEP_POLL
|
|
case RFAL_NFC_INTERFACE_ISODEP:
|
|
rfalIsoDepDeselect(); /* Send a Deselect to device */
|
|
break;
|
|
#endif /* RFAL_FEATURE_ISO_DEP_POLL */
|
|
|
|
/*******************************************************************************/
|
|
#if RFAL_FEATURE_NFC_DEP
|
|
case RFAL_NFC_INTERFACE_NFCDEP:
|
|
switch(gNfcDev.activeDev->type) {
|
|
case RFAL_NFC_LISTEN_TYPE_AP2P:
|
|
rfalNfcDepRLS(); /* Send a Release to device */
|
|
break;
|
|
default:
|
|
rfalNfcDepDSL(); /* Send a Deselect to device */
|
|
break;
|
|
}
|
|
break;
|
|
#endif /* RFAL_FEATURE_NFC_DEP */
|
|
|
|
default:
|
|
return ERR_REQUEST;
|
|
}
|
|
#endif /* RFAL_NFC_SKIP_DEACT */
|
|
}
|
|
}
|
|
|
|
#if RFAL_FEATURE_WAKEUP_MODE
|
|
rfalWakeUpModeStop();
|
|
#endif /* RFAL_FEATURE_WAKEUP_MODE */
|
|
|
|
#if RFAL_FEATURE_LISTEN_MODE
|
|
rfalListenStop();
|
|
#else
|
|
rfalFieldOff();
|
|
#endif
|
|
|
|
gNfcDev.activeDev = NULL;
|
|
return ERR_NONE;
|
|
}
|