| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126 |
- /**
- ******************************************************************************
- *
- * 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 "../include/rfal_nfc.h"
- #include "../utils.h"
- #include "../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; /* Devices 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 operation 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 successfuly 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 rfalNfcDataExchangeCustomStart(
- 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:
- ctx.rxBuf = gNfcDev.rxBuf.rfBuf;
- ctx.rxBufLen = 8 * sizeof(gNfcDev.rxBuf.rfBuf);
- ctx.rxRcvdLen = &gNfcDev.rxLen;
- ctx.txBuf = txData;
- ctx.txBufLen = txDataLen;
- ctx.flags = flags;
- ctx.fwt = fwt;
- *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;
- uint16_t tx_bytes = txDataLen / 8;
- if(tx_bytes > sizeof(gNfcDev.txBuf.isoDepBuf.apdu)) {
- return ERR_NOMEM;
- }
- if(tx_bytes > 0U) {
- ST_MEMCPY((uint8_t*)gNfcDev.txBuf.isoDepBuf.apdu, txData, tx_bytes);
- }
- 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 = tx_bytes;
- 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 successfuly 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 successful 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;
- /* Suppress 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;
- /* Suppress 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;
- /* Suppress 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 performed (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;
- /* Suppress 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;
- }
|