2715 lines
		
	
	
		
			116 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			2715 lines
		
	
	
		
			116 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
| 
 | |
| /******************************************************************************
 | |
|   * \attention
 | |
|   *
 | |
|   * <h2><center>© COPYRIGHT 2020 STMicroelectronics</center></h2>
 | |
|   *
 | |
|   * Licensed under ST MYLIBERTY SOFTWARE LICENSE AGREEMENT (the "License");
 | |
|   * You may not use this file except in compliance with the License.
 | |
|   * You may obtain a copy of the License at:
 | |
|   *
 | |
|   *        www.st.com/myliberty
 | |
|   *
 | |
|   * Unless required by applicable law or agreed to in writing, software 
 | |
|   * distributed under the License is distributed on an "AS IS" BASIS, 
 | |
|   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
 | |
|   * AND SPECIFICALLY DISCLAIMING THE IMPLIED WARRANTIES OF MERCHANTABILITY,
 | |
|   * FITNESS FOR A PARTICULAR PURPOSE, AND NON-INFRINGEMENT.
 | |
|   * See the License for the specific language governing permissions and
 | |
|   * limitations under the License.
 | |
|   *
 | |
| ******************************************************************************/
 | |
| 
 | |
| 
 | |
| /*
 | |
|  *      PROJECT:   NFCC firmware
 | |
|  *      LANGUAGE:  ISO C99
 | |
|  */
 | |
| 
 | |
| /*! \file rfal_nfcDep.c
 | |
|  *
 | |
|  *  \author  Gustavo Patricio
 | |
|  *
 | |
|  *  \brief Implementation of NFC-DEP protocol
 | |
|  *  
 | |
|  *  NFC-DEP is also known as NFCIP - Near Field Communication 
 | |
|  *  Interface and Protocol
 | |
|  *  
 | |
|  *  This implementation was based on the following specs:
 | |
|  *    - NFC Forum Digital 1.1
 | |
|  *    - ECMA 340 3rd Edition 2013
 | |
|  *
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * INCLUDES
 | |
|  ******************************************************************************
 | |
|  */
 | |
| #include "rfal_nfcDep.h"
 | |
| #include "rfal_nfcf.h"
 | |
| #include "utils.h"
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * ENABLE SWITCH
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| #if RFAL_FEATURE_NFC_DEP
 | |
|     
 | |
| /* Check for valid Block/Payload length  Digital 2.0 Table 90*/
 | |
| #if( (RFAL_FEATURE_NFC_DEP_BLOCK_MAX_LEN != 64)  && (RFAL_FEATURE_NFC_DEP_BLOCK_MAX_LEN != 128) && (RFAL_FEATURE_NFC_DEP_BLOCK_MAX_LEN != 192) && (RFAL_FEATURE_NFC_DEP_BLOCK_MAX_LEN != 254) )
 | |
|     #error " RFAL: Invalid NFC-DEP Block Max length. Please change RFAL_FEATURE_NFC_DEP_BLOCK_MAX_LEN. "
 | |
| #endif
 | |
| 
 | |
| /* Check for valid PDU length */
 | |
| #if( (RFAL_FEATURE_NFC_DEP_PDU_MAX_LEN < RFAL_FEATURE_NFC_DEP_BLOCK_MAX_LEN) )
 | |
|     #error " RFAL: Invalid NFC-DEP PDU Max length. Please change RFAL_FEATURE_NFC_DEP_PDU_MAX_LEN. "
 | |
| #endif     
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * DEFINES
 | |
|  ******************************************************************************
 | |
|  */
 | |
| #define NFCIP_ATR_RETRY_MAX             2U                              /*!< Max consecutive retrys of an ATR REQ with transm error*/
 | |
| 
 | |
| #define NFCIP_PSLPAY_LEN                (2U)                            /*!< PSL Payload length (BRS + FSL)                        */
 | |
| #define NFCIP_PSLREQ_LEN                (3U + RFAL_NFCDEP_LEN_LEN)      /*!< PSL REQ length (incl LEN)                             */
 | |
| #define NFCIP_PSLRES_LEN                (3U + RFAL_NFCDEP_LEN_LEN)      /*!< PSL RES length (incl LEN)                             */
 | |
| 
 | |
| #define NFCIP_ATRREQ_BUF_LEN            (RFAL_NFCDEP_ATRREQ_MAX_LEN + RFAL_NFCDEP_LEN_LEN) /*!< ATR REQ max length (incl LEN)     */
 | |
| #define NFCIP_ATRRES_BUF_LEN            (RFAL_NFCDEP_ATRRES_MAX_LEN + RFAL_NFCDEP_LEN_LEN) /*!< ATR RES max length (incl LEN)     */
 | |
| 
 | |
| #define NFCIP_RLSREQ_LEN                (3U + RFAL_NFCDEP_LEN_LEN)      /*!< RLS REQ length (incl LEN)                             */
 | |
| #define NFCIP_RLSRES_LEN                (3U + RFAL_NFCDEP_LEN_LEN)      /*!< RSL RES length (incl LEN)                             */
 | |
| #define NFCIP_RLSRES_MIN                (2U + RFAL_NFCDEP_LEN_LEN)      /*!< Minimum length for a RLS RES (incl LEN)               */
 | |
| 
 | |
| #define NFCIP_DSLREQ_LEN                (3U + RFAL_NFCDEP_LEN_LEN)      /*!< DSL REQ length (incl LEN)                             */
 | |
| #define NFCIP_DSLRES_LEN                (3U + RFAL_NFCDEP_LEN_LEN)      /*!< DSL RES length (incl LEN)                             */
 | |
| #define NFCIP_DSLRES_MIN                (2U + RFAL_NFCDEP_LEN_LEN)      /*!< Minimum length for a DSL RES (incl LEN)               */
 | |
| 
 | |
| #define NFCIP_DSLRES_MAX_LEN            (3U + RFAL_NFCDEP_LEN_LEN)      /*!< Maximum length for a DSL RES (incl LEN)               */
 | |
| #define NFCIP_RLSRES_MAX_LEN            (3U + RFAL_NFCDEP_LEN_LEN)      /*!< Minimum length for a RLS RES (incl LEN)               */
 | |
| #define NFCIP_TARGET_RES_MAX            ( MAX( NFCIP_RLSRES_MAX_LEN, NFCIP_DSLRES_MAX_LEN) ) /*!< Max target control res length    */
 | |
| 
 | |
| 
 | |
| 
 | |
| #define NFCIP_NO_FWT                    RFAL_FWT_NONE                   /*!< No FWT value - Target Mode                            */
 | |
| #define NFCIP_INIT_MIN_RTOX             1U                              /*!< Minimum RTOX value  Digital 1.0  14.8.4.1             */
 | |
| #define NFCIP_INIT_MAX_RTOX             59U                             /*!< Maximum RTOX value  Digital 1.0  14.8.4.1             */
 | |
| 
 | |
| #define NFCIP_TARG_MIN_RTOX             1U                              /*!< Minimum target RTOX value  Digital 1.0  14.8.4.1      */
 | |
| #define NFCIP_TARG_MAX_RTOX             59U                             /*!< Maximum target RTOX value  Digital 1.0  14.8.4.1      */
 | |
| 
 | |
| #define NFCIP_TRECOV                    1280U                           /*!< Digital 1.0  A.10  Trecov                             */
 | |
|  
 | |
| #define NFCIP_TIMEOUT_ADJUSTMENT        3072U                           /*!< Timeout Adjustment to compensate timing from end of Tx to end of frame  */
 | |
| #define NFCIP_RWT_ACTIVATION            (0x1000001U + NFCIP_TIMEOUT_ADJUSTMENT) /*!< Digital 2.2  B.11  RWT ACTIVATION  2^24 + RWT Delta + Adjustment*/
 | |
| #define NFCIP_RWT_ACM_ACTIVATION        (0x200001U + NFCIP_TIMEOUT_ADJUSTMENT)  /*!< Digital 2.2  B.11  RWT ACTIVATION  2^21 + RWT Delta + Adjustment*/
 | |
| 
 | |
| #define RFAL_NFCDEP_HEADER_PAD          (RFAL_NFCDEP_DEPREQ_HEADER_LEN - RFAL_NFCDEP_LEN_MIN) /*!< Difference between expected rcvd header len and max foreseen */
 | |
| 
 | |
| #ifndef RFAL_NFCDEP_MAX_TX_RETRYS
 | |
| #define RFAL_NFCDEP_MAX_TX_RETRYS   (uint8_t)3U          /*!< Number of retransmit retyrs                           */
 | |
| #endif /* RFAL_NFCDEP_MAX_TX_RETRYS */
 | |
|                                                          
 | |
| #ifndef RFAL_NFCDEP_TO_RETRYS                            
 | |
| #define RFAL_NFCDEP_TO_RETRYS       (uint8_t)3U          /*!< Number of retrys for Timeout                          */
 | |
| #endif /* RFAL_NFCDEP_TO_RETRYS */
 | |
|                                                          
 | |
| #ifndef RFAL_NFCDEP_MAX_RTOX_RETRYS                      
 | |
| #define RFAL_NFCDEP_MAX_RTOX_RETRYS (uint8_t)10U         /*!< Number of retrys for RTOX    Digital 2.0 17.12.4.3    */
 | |
| #endif /* RFAL_NFCDEP_MAX_RTOX_RETRYS */
 | |
|                                                          
 | |
| #ifndef RFAL_NFCDEP_MAX_NACK_RETRYS                      
 | |
| #define RFAL_NFCDEP_MAX_NACK_RETRYS (uint8_t)3U          /*!< Number of retrys for NACK                             */
 | |
| #endif /* RFAL_NFCDEP_MAX_NACK_RETRYS */
 | |
|                                                          
 | |
| #ifndef RFAL_NFCDEP_MAX_ATN_RETRYS                       
 | |
| #define RFAL_NFCDEP_MAX_ATN_RETRYS  (uint8_t)3U          /*!< Number of retrys for ATN                              */
 | |
| #endif /* RFAL_NFCDEP_MAX_ATN_RETRYS */
 | |
| 
 | |
| #define NFCIP_MIN_TXERROR_LEN           4U               /*!< Minimum frame length with error to be ignored  Digital 1.0 14.12.5.4 */
 | |
| 
 | |
| #define NFCIP_REQ                       (uint8_t)0xD4U   /*!<NFCIP REQuest code                                     */
 | |
| #define NFCIP_RES                       (uint8_t)0xD5U   /*!<NFCIP RESponce code                                    */
 | |
| 
 | |
| #define NFCIP_BS_MASK                   0x0FU            /*!< Bit mask for BS value on a ATR REQ/RES                */
 | |
| #define NFCIP_BR_MASK                   NFCIP_BS_MASK    /*!< Bit mask for BR value on a ATR REQ/RES                */
 | |
| 
 | |
| #define NFCIP_PP_GB_MASK                0x02U            /*!< Bit mask for GB value in PP byte on a ATR REQ/RES     */
 | |
| #define NFCIP_PP_NAD_MASK               0x01U            /*!< Bit mask for NAD value in PP byte on a ATR REQ/RES    */
 | |
| 
 | |
| #define NFCIP_PFB_xPDU_MASK             0xE0U            /*!< Bit mask for PDU type                                 */
 | |
| #define NFCIP_PFB_IPDU                  0x00U            /*!< Bit mask indicating a Information PDU                 */
 | |
| #define NFCIP_PFB_RPDU                  0x40U            /*!< Bit mask indicating a Response PDU                    */
 | |
| #define NFCIP_PFB_SPDU                  0x80U            /*!< Bit mask indicating a Supervisory PDU                 */
 | |
| 
 | |
| #define NFCIP_PFB_MI_BIT                0x10U            /*!< Bit mask for the chaining bit (MI) of PFB             */
 | |
| #define NFCIP_PFB_DID_BIT               0x04U            /*!< Bit mask for the DID presence bit of PFB              */
 | |
| #define NFCIP_PFB_NAD_BIT               0x08U            /*!< Bit mask for the NAD presence bit of PFB              */
 | |
| #define NFCIP_PFB_PNI_MASK              0x03U            /*!< Bit mask for the Packet Number Information            */
 | |
| 
 | |
| #define NFCIP_PFB_Rx_MASK               0x10U            /*!< Bit mask for the R-PDU type                           */
 | |
| #define NFCIP_PFB_ACK                   0x00U            /*!< Bit mask for R-PDU indicating ACK                     */
 | |
| #define NFCIP_PFB_NACK                  0x10U            /*!< Bit mask for R-PDU indicating NAK                     */
 | |
| 
 | |
| #define NFCIP_PFB_Sx_MASK               0x10U            /*!< Bit mask for the R-PDU type                           */
 | |
| #define NFCIP_PFB_ATN                   0x00U            /*!< Bit mask for R-PDU indicating ACK                     */
 | |
| #define NFCIP_PFB_TO                    0x10U            /*!< Bit mask for R-PDU indicating NAK                     */
 | |
| 
 | |
| #define NFCIP_PFB_INVALID               0xFFU            /*!< Invalid PFB value                                     */
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * MACROS
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| #define nfcipIsTransmissionError(e)    ( ((e) == ERR_CRC) || ((e) == ERR_FRAMING) || ((e) == ERR_PAR) ) /*!< Checks if is a Trasmission error */
 | |
| 
 | |
| 
 | |
| #define nfcipConv1FcToMs( v )          (rfalConv1fcToMs((v)) + 1U)                                     /*!< Converts value v 1fc into milliseconds (fc=13.56)     */
 | |
| 
 | |
| #define nfcipCmdIsReq( cmd )           (((uint8_t)(cmd) % 2U) == 0U)                                    /*!< Checks if the nfcip cmd is a REQ                      */
 | |
| 
 | |
| #define nfcip_PFBhasDID( pfb )         ( ((pfb) & NFCIP_PFB_DID_BIT) == NFCIP_PFB_DID_BIT)              /*!< Checks if pfb is signalling DID                       */
 | |
| #define nfcip_PFBhasNAD( pfb )         ( ((pfb) & NFCIP_PFB_NAD_BIT) == NFCIP_PFB_NAD_BIT)              /*!< Checks if pfb is signalling NAD                       */
 | |
| 
 | |
| #define nfcip_PFBisIPDU( pfb )         ( ((pfb) & NFCIP_PFB_xPDU_MASK) == NFCIP_PFB_IPDU)               /*!< Checks if pfb is a Information PDU                    */
 | |
| #define nfcip_PFBisRPDU( pfb )         ( ((pfb) & NFCIP_PFB_xPDU_MASK) == NFCIP_PFB_RPDU)               /*!< Checks if pfb is Response PDU                         */
 | |
| #define nfcip_PFBisSPDU( pfb )         ( ((pfb) & NFCIP_PFB_xPDU_MASK) == NFCIP_PFB_SPDU)               /*!< Checks if pfb is a Supervisory PDU                    */
 | |
| 
 | |
| #define nfcip_PFBisIMI( pfb )          ( nfcip_PFBisIPDU( pfb ) && (((pfb) & NFCIP_PFB_MI_BIT) == NFCIP_PFB_MI_BIT))  /*!< Checks if pfb is a Information PDU indicating MI chaining */
 | |
| 
 | |
| #define nfcip_PFBisRNACK( pfb )        ( nfcip_PFBisRPDU( pfb ) && (((pfb) & NFCIP_PFB_Rx_MASK) == NFCIP_PFB_NACK)) /*!< Checks if pfb is a R-PDU indicating NACK  */
 | |
| #define nfcip_PFBisRACK( pfb )         ( nfcip_PFBisRPDU( pfb ) && (((pfb) & NFCIP_PFB_Rx_MASK) == NFCIP_PFB_ACK )) /*!< Checks if pfb is a R-PDU indicating ACK   */
 | |
| 
 | |
| #define nfcip_PFBisSATN( pfb )         ( nfcip_PFBisSPDU( pfb ) && (((pfb) & NFCIP_PFB_Sx_MASK) == NFCIP_PFB_ATN))  /*!< Checks if pfb is a R-PDU indicating ATN   */
 | |
| #define nfcip_PFBisSTO( pfb )          ( nfcip_PFBisSPDU( pfb ) && (((pfb) & NFCIP_PFB_Sx_MASK) == NFCIP_PFB_TO) )  /*!< Checks if pfb is a R-PDU indicating TO    */
 | |
| 
 | |
| 
 | |
| #define nfcip_PFBIPDU( pni )           ( (uint8_t)( 0x00U | NFCIP_PFB_IPDU | ((pni) & NFCIP_PFB_PNI_MASK) ))/*!< Returns a PFB I-PDU with the given packet number (pni)                   */
 | |
| #define nfcip_PFBIPDU_MI( pni )        ( (uint8_t)(isoDep_PCBIBlock(pni) | NFCIP_PFB_MI_BIT))               /*!< Returns a PFB I-PDU with the given packet number (pni) indicating chaing */
 | |
| 
 | |
| #define nfcip_PFBRPDU( pni )           ( (uint8_t)( 0x00U | NFCIP_PFB_RPDU | ((pni) & NFCIP_PFB_PNI_MASK) ))/*!< Returns a PFB R-PDU with the given packet number (pni)                   */
 | |
| #define nfcip_PFBRPDU_NACK( pni )      ( (uint8_t)(nfcip_PFBRPDU(pni) | NFCIP_PFB_NACK))                    /*!< Returns a PFB R-PDU with the given packet number (pni) indicating NACK   */
 | |
| #define nfcip_PFBRPDU_ACK( pni )       ( (uint8_t)(nfcip_PFBRPDU(pni) | NFCIP_PFB_ACK))                     /*!< Returns a PFB R-PDU with the given packet number (pni) indicating ACK    */
 | |
| 
 | |
| #define nfcip_PFBSPDU()                ( (uint8_t)( 0x00U | NFCIP_PFB_SPDU ))                           /*!< Returns a PFB S-PDU                                   */
 | |
| #define nfcip_PFBSPDU_ATN()            ( (uint8_t)(nfcip_PFBSPDU() | NFCIP_PFB_ATN))                    /*!< Returns a PFB S-PDU indicating ATN                    */
 | |
| #define nfcip_PFBSPDU_TO()             ( (uint8_t)(nfcip_PFBSPDU() | NFCIP_PFB_TO))                     /*!< Returns a PFB S-PDU indicating TO                     */
 | |
| 
 | |
| 
 | |
| #define nfcip_PNIInc( pni )            ( (uint8_t) (((pni)+1U) & NFCIP_PFB_PNI_MASK) )                  /*!< Returns a incremented PNI from the given (pni)        */
 | |
| #define nfcip_PNIDec( pni )            ( (uint8_t) (((pni)-1U) & NFCIP_PFB_PNI_MASK) )                  /*!< Returns a decremented PNI from the given (pni)        */
 | |
| 
 | |
| #define nfcip_PBF_PNI( pfb )           ( (uint8_t) ((pfb) & NFCIP_PFB_PNI_MASK ))                       /*!< Returns the Packet Number Information (pni)           */
 | |
| 
 | |
| #define nfcip_PPwGB( lr )              ( rfalNfcDepLR2PP( lr ) | NFCIP_PP_GB_MASK)                      /*!< Returns a PP byte containing the given PP value indicating GB                  */
 | |
| 
 | |
| #define nfcip_DIDMax( did )            ( MIN( (did), RFAL_NFCDEP_DID_MAX) )                             /*!< Ensures that the given did has proper value  Digital 14.6.2.3 DID [0 14]       */
 | |
| #define nfcip_RTOXTargMax( wt )        (uint8_t)( MIN( (RFAL_NFCDEP_RWT_TRG_MAX / rfalNfcDepWT2RWT(wt)), NFCIP_TARG_MAX_RTOX) )/*!< Calculates the Maximum RTOX value for the given wt as a Target */
 | |
| 
 | |
| #define nfcipIsInitiator( st )         ( ((st) >= NFCIP_ST_INIT_IDLE) && ((st) <= NFCIP_ST_INIT_RLS) )  /*!< Checks if module is set as Initiator                                           */
 | |
| #define nfcipIsTarget( st )            (!nfcipIsInitiator(st))                                          /*!< Checks if module is set as Target                                              */
 | |
| 
 | |
| #define nfcipIsBRAllowed( br, mBR )    (((1U<<(br)) & (mBR)) != 0U)                                     /*!< Checks bit rate is allowed by given mask                                       */
 | |
| 
 | |
| #define nfcipIsEmptyDEPEnabled( op )   (!nfcipIsEmptyDEPDisabled(op))                                   /*!< Checks if empty payload is allowed by operation config  NCI 1.0 Table 81       */
 | |
| #define nfcipIsEmptyDEPDisabled( op )  (((op) & RFAL_NFCDEP_OPER_EMPTY_DEP_DIS) != 0U)                  /*!< Checks if empty payload is not allowed by operation config  NCI 1.0 Table 81   */
 | |
| 
 | |
| #define nfcipIsRTOXReqEnabled( op )    (!nfcipIsRTOXReqDisabled(op))                                    /*!< Checks if send a RTOX_REQ is allowed by operation config  NCI 1.0 Table 81     */
 | |
| #define nfcipIsRTOXReqDisabled( op )   (((op) & RFAL_NFCDEP_OPER_RTOX_REQ_DIS) != 0U)                   /*!< Checks if send a RTOX_REQ is not allowed by operation config  NCI 1.0 Table 81 */
 | |
| 
 | |
| 
 | |
| /*! Checks if isDeactivating callback is set and calls it, otherwise returns false */
 | |
| #define nfcipIsDeactivationPending()   ( (gNfcip.isDeactivating == NULL) ? false : gNfcip.isDeactivating() )
 | |
|     
 | |
| /*! Returns the RWT Activation according to the current communication mode */
 | |
| #define nfcipRWTActivation()            ((gNfcip.cfg.commMode == RFAL_NFCDEP_COMM_ACTIVE) ? NFCIP_RWT_ACM_ACTIVATION : NFCIP_RWT_ACTIVATION)
 | |
| 
 | |
| 
 | |
| #define nfcipRTOXAdjust( v )           ((v) - ((v)>>3))                                                 /*!< Adjust RTOX timer value to a percentage of the total, current 88% */ 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| 
 | |
| // timerPollTimeoutValue is necessary after timerCalculateTimeout so that system will wake up upon timer timeout.
 | |
| #define nfcipTimerStart( timer, time_ms ) do{platformTimerDestroy( timer);  (timer) = platformTimerCreate((uint16_t)(time_ms));} while (0) /*!< Configures and starts the RTOX timer            */
 | |
| #define nfcipTimerisExpired( timer )      platformTimerIsExpired( timer )                               /*!< Checks RTOX timer has expired                   */
 | |
| #define nfcipTimerDestroy( timer )        platformTimerDestroy( timer )                                 /*!< Destroys RTOX timer                             */
 | |
| 
 | |
| #define nfcipLogE(...)                                                                                  /*!< Macro for the error log method                  */
 | |
| #define nfcipLogW(...)                                                                                  /*!< Macro for the warning log method                */
 | |
| #define nfcipLogI(...)                                                                                  /*!< Macro for the info log method                   */
 | |
| #define nfcipLogD(...)                                                                                  /*!< Macro for the debug log method                  */
 | |
| 
 | |
| 
 | |
| /*! Digital 1.1 - 16.12.5.2  The Target SHALL NOT attempt any error recovery and remains in Rx mode upon Transmission or a Protocol Error */
 | |
| #define nfcDepReEnableRx( rxB, rxBL, rxL )       rfalTransceiveBlockingTx( NULL, 0, (rxB), (rxBL), (rxL), ( RFAL_TXRX_FLAGS_DEFAULT | (uint32_t)RFAL_TXRX_FLAGS_NFCIP1_ON ), RFAL_FWT_NONE )
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * LOCAL DATA TYPES
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| /*! Struct that holds all DEP parameters/configs for the following communications */
 | |
| typedef struct{
 | |
|     uint8_t   did;           /*!< Device ID (DID) to be used                      */
 | |
|     
 | |
|     uint8_t*  txBuf;         /*!< Pointer to the Tx buffer to be sent             */
 | |
|     uint16_t  txBufLen;      /*!< Length of the data in the txBuf                 */
 | |
|     uint8_t   txBufPaylPos;  /*!< Position inside txBuf where data starts         */
 | |
|     bool      txChaining;    /*!< Flag indicating chaining on transmission        */
 | |
|     
 | |
|     uint8_t*  rxBuf;         /*!< Pointer to the Rx buffer for incoming data      */
 | |
|     uint16_t  rxBufLen;      /*!< Length of the data in the rxBuf                 */
 | |
|     uint8_t   rxBufPaylPos;  /*!< Position inside rxBuf where data is to be placed*/
 | |
|     
 | |
|     uint32_t  fwt;           /*!< Frame Waiting Time (FWT) to be used             */
 | |
|     uint32_t  dFwt;          /*!< Delta Frame Waiting Time (dFWT) to be used      */
 | |
|     uint16_t  fsc;           /*!< Frame Size (FSC) to be used                     */
 | |
|     
 | |
| } rfalNfcDepDEPParams;
 | |
| 
 | |
| /*! NFCIP module states */
 | |
| typedef enum
 | |
| {
 | |
|     NFCIP_ST_IDLE,
 | |
|     NFCIP_ST_INIT_IDLE,
 | |
|     NFCIP_ST_INIT_ATR,
 | |
|     NFCIP_ST_INIT_PSL,
 | |
|     NFCIP_ST_INIT_DEP_IDLE,
 | |
|     NFCIP_ST_INIT_DEP_TX,
 | |
|     NFCIP_ST_INIT_DEP_RX,
 | |
|     NFCIP_ST_INIT_DEP_ATN,
 | |
|     NFCIP_ST_INIT_DSL,
 | |
|     NFCIP_ST_INIT_RLS,
 | |
|         
 | |
|     NFCIP_ST_TARG_WAIT_ATR,
 | |
|     NFCIP_ST_TARG_WAIT_ACTV,
 | |
|     NFCIP_ST_TARG_DEP_IDLE,
 | |
|     NFCIP_ST_TARG_DEP_RX,
 | |
|     NFCIP_ST_TARG_DEP_RTOX,
 | |
|     NFCIP_ST_TARG_DEP_TX,
 | |
|     NFCIP_ST_TARG_DEP_SLEEP
 | |
| } rfalNfcDepState;
 | |
| 
 | |
| /*! NFCIP commands (Request, Response) */
 | |
| typedef enum{
 | |
|     NFCIP_CMD_ATR_REQ = 0x00,
 | |
|     NFCIP_CMD_ATR_RES = 0x01,
 | |
|     NFCIP_CMD_WUP_REQ = 0x02,
 | |
|     NFCIP_CMD_WUP_RES = 0x03,
 | |
|     NFCIP_CMD_PSL_REQ = 0x04,
 | |
|     NFCIP_CMD_PSL_RES = 0x05,
 | |
|     NFCIP_CMD_DEP_REQ = 0x06,
 | |
|     NFCIP_CMD_DEP_RES = 0x07,
 | |
|     NFCIP_CMD_DSL_REQ = 0x08,
 | |
|     NFCIP_CMD_DSL_RES = 0x09,
 | |
|     NFCIP_CMD_RLS_REQ = 0x0A,
 | |
|     NFCIP_CMD_RLS_RES = 0x0B
 | |
| } rfalNfcDepCmd;
 | |
| 
 | |
| 
 | |
| /*! Struct that holds all NFCIP data */
 | |
| typedef struct{  
 | |
|   rfalNfcDepConfigs       cfg;               /*!< Holds the current configuration to be used    */
 | |
|   
 | |
|   rfalNfcDepState         state;             /*!< Current state of the NFCIP module             */
 | |
|   uint8_t                 pni;               /*!< Packet Number Information (PNI) counter       */
 | |
|   
 | |
|   uint8_t                 lastCmd;           /*!< Last command sent                             */
 | |
|   uint8_t                 lastPFB;           /*!< Last PFB sent                                 */
 | |
|   uint8_t                 lastPFBnATN;       /*!< Last PFB sent (excluding  ATN)                */
 | |
|   uint8_t                 lastRTOX;          /*!< Last RTOX value sent                          */
 | |
|   
 | |
|   uint8_t                 cntTxRetrys;       /*!< Retransmissions counter                       */
 | |
|   uint8_t                 cntTORetrys;       /*!< Timeouts counter                              */
 | |
|   uint8_t                 cntRTOXRetrys;     /*!< RTOX counter                                  */
 | |
|   uint8_t                 cntNACKRetrys;     /*!< NACK counter                                  */
 | |
|   uint8_t                 cntATNRetrys;      /*!< Attention (ATN) counter                       */
 | |
|   
 | |
|   uint16_t                fsc;               /*!< Current Frame Size (FSC) to be used           */
 | |
|   bool                    isTxChaining;      /*!< Flag for chaining on Transmission             */
 | |
|   bool                    isRxChaining;      /*!< Flag for chaining on Reception                */
 | |
|   uint8_t*                txBuf;             /*!< Pointer to the Tx buffer to be sent           */
 | |
|   uint8_t*                rxBuf;             /*!< Pointer to the Rx buffer for incoming data    */
 | |
|   uint16_t                txBufLen;          /*!< Length of the data in the txBuf               */
 | |
|   uint16_t                rxBufLen;          /*!< Length of rxBuf buffer                        */
 | |
|   uint16_t*               rxRcvdLen;         /*!< Length of the data in the rxBuf               */
 | |
|   uint8_t                 txBufPaylPos;      /*!< Position in txBuf where data starts           */
 | |
|   uint8_t                 rxBufPaylPos;      /*!< Position in rxBuf where data is to be placed  */
 | |
|   bool                    *isChaining;       /*!< Flag for chaining on Reception                */
 | |
|   
 | |
|   rfalNfcDepDevice        *nfcDepDev;        /*!< Pointer to NFC-DEP device info                */
 | |
| 
 | |
|   uint32_t                RTOXTimer;         /*!< Timer used for RTOX                           */  
 | |
|   rfalNfcDepDeactCallback isDeactivating;    /*!< Deactivating flag check callback              */
 | |
|   
 | |
|   bool                    isReqPending;      /*!< Flag pending REQ from Target activation       */
 | |
|   bool                    isTxPending;       /*!< Flag pending DEP Block while waiting RTOX Ack */
 | |
|   bool                    isWait4RTOX;       /*!< Flag for waiting RTOX Ack                     */
 | |
|   
 | |
|   rfalNfcDepPduTxRxParam  PDUParam;          /*!< PDU TxRx params                               */
 | |
|   uint16_t                PDUTxPos;          /*!< PDU Tx position                               */
 | |
|   uint16_t                PDURxPos;          /*!< PDU Rx position                               */
 | |
|   bool                    isPDURxChaining;   /*!< PDU Transceive chaining flag                  */
 | |
| }rfalNfcDep;
 | |
| 
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * LOCAL VARIABLES
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| static rfalNfcDep gNfcip;                    /*!< NFCIP module instance                         */
 | |
| 
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * LOCAL FUNCTION PROTOTYPES
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| static ReturnCode nfcipTxRx( rfalNfcDepCmd cmd, uint8_t* txBuf, uint32_t fwt, uint8_t* paylBuf, uint8_t paylBufLen, uint8_t* rxBuf, uint16_t rxBufLen, uint16_t *rxActLen );
 | |
| static ReturnCode nfcipTx( rfalNfcDepCmd cmd, uint8_t* txBuf, uint8_t *paylBuf, uint16_t paylLen, uint8_t pfbData, uint32_t fwt );
 | |
| static ReturnCode nfcipDEPControlMsg( uint8_t pfb, uint8_t RTOX  );
 | |
| static ReturnCode nfcipInitiatorHandleDEP( ReturnCode rxRes, uint16_t rxLen, uint16_t *outActRxLen, bool *outIsChaining );
 | |
| static ReturnCode nfcipTargetHandleRX( ReturnCode rxRes, uint16_t *outActRxLen, bool *outIsChaining );
 | |
| static ReturnCode nfcipTargetHandleActivation( rfalNfcDepDevice *nfcDepDev, uint8_t *outBRS );
 | |
| 
 | |
| 
 | |
| /*!
 | |
|  ******************************************************************************
 | |
|  * \brief NFCIP Configure
 | |
|  * 
 | |
|  * Configures the nfcip layer with the given configurations 
 | |
|  * 
 | |
|  * \param[in] cfg   : nfcip configuration for following communication
 | |
|  ******************************************************************************
 | |
|  */
 | |
| static void nfcipConfig( const rfalNfcDepConfigs * cfg );
 | |
| 
 | |
| 
 | |
| /*!
 | |
|  ******************************************************************************
 | |
|  * \brief Set DEP parameters
 | |
|  * 
 | |
|  * This method sets the parameters/configs for following Data Exchange
 | |
|  * Sets the nfcip module state according to the role it is configured
 | |
|  * 
 | |
|  * 
 | |
|  * \warning To be used only after proper Initiator/Target activation: 
 | |
|  *           nfcipTargetHandleActivation() or nfcipInitiatorActivate() has
 | |
|  *           returned success
 | |
|  *           
 | |
|  *  This must be called before  nfcipRun() in case of Target to pass 
 | |
|  *  rxBuffer
 | |
|  *  
 | |
|  *  Everytime some data needs to be transmitted call this to set it and 
 | |
|  *  call nfcipRun() until done or error
 | |
|  * 
 | |
|  * \param[in]  DEPParams  : the parameters to be used during Data Exchange 
 | |
|  ******************************************************************************
 | |
|  */
 | |
| static void nfcipSetDEPParams( const rfalNfcDepDEPParams *DEPParams );
 | |
| 
 | |
| 
 | |
| /*!
 | |
|  ******************************************************************************
 | |
|  * \brief NFCIP run protocol
 | |
|  * 
 | |
|  * This method handles all the nfcip protocol during Data Exchange (DEP
 | |
|  *  requests and responses).
 | |
|  *  
 | |
|  *  A data exchange cycle is considered a DEP REQ and a DEP RES.
 | |
|  *  
 | |
|  *  In case of Tx chaining(MI) must signal it with  nfcipSetDEPParams()
 | |
|  *  In case of Rx chaining(MI) outIsChaining will be set to true and the 
 | |
|  *  current data returned
 | |
|  *  
 | |
|  * \param[out] outActRxLen   : data received length
 | |
|  * \param[out] outIsChaining : true if other peer is performing chaining(MI)
 | |
|  *  
 | |
|  * \return ERR_NONE    : Data exchange cycle completed successfully
 | |
|  * \return ERR_TIMEOUT : Timeout occurred
 | |
|  * \return ERR_PROTO   : Protocol error occurred
 | |
|  * \return ERR_AGAIN   : Other peer is doing chaining(MI), current block 
 | |
|  *                       was received successfully call again until complete
 | |
|  * 
 | |
|  ******************************************************************************
 | |
|  */
 | |
| static ReturnCode nfcipRun( uint16_t *outActRxLen, bool *outIsChaining  );
 | |
| 
 | |
| 
 | |
| /*!
 | |
|  ******************************************************************************
 | |
|  * \brief Transmission method
 | |
|  * 
 | |
|  * This method checks if the current communication is Active or Passive
 | |
|  * and performs the necessary procedures for each communication type
 | |
|  * 
 | |
|  * Transmits the data hold in txBuf 
 | |
|  * 
 | |
|  * \param[in]  txBuf    : buffer to transmit
 | |
|  * \param[in]  txBufLen : txBuffer capacity 
 | |
|  * \param[in]  fwt      : fwt for current Tx
 | |
|  * 
 | |
|  * \return ERR_NONE       : No error
 | |
|  ******************************************************************************
 | |
|  */
 | |
| static ReturnCode nfcipDataTx( uint8_t* txBuf, uint16_t txBufLen, uint32_t fwt );
 | |
| 
 | |
| 
 | |
| /*!
 | |
|  ******************************************************************************
 | |
|  * \brief Reception method
 | |
|  * 
 | |
|  * This method checks if the current communication is Active or Passive
 | |
|  * and calls the appropriate reception method
 | |
|  * 
 | |
|  * Copies incoming data to rxBuf
 | |
|  * 
 | |
|  * \param[in] blocking    : reception is to be done blocking or non-blocking
 | |
|  * 
 | |
|  * \return ERR_BUSY       : Busy
 | |
|  * \return ERR_NONE       : No error
 | |
|  ******************************************************************************
 | |
|  */
 | |
| static ReturnCode nfcipDataRx( bool blocking );
 | |
| 
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * LOCAL FUNCTIONS
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| /*******************************************************************************/
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static bool nfcipDxIsSupported( uint8_t Dx, uint8_t BRx, uint8_t BSx )
 | |
| {
 | |
| 	uint8_t Bx;
 | |
| 	
 | |
| 	/* Take the min of the possible bit rates, we'll use one for both directions */
 | |
| 	Bx = MIN(BRx, BSx);
 | |
| 	
 | |
| 	/* Lower bit rates must be supported for P2P */
 | |
| 	if( (Dx <= (uint8_t)RFAL_NFCDEP_Dx_04_424) )
 | |
| 	{
 | |
| 		return true;
 | |
| 	}
 | |
| 	
 | |
| 	if( (Dx == (uint8_t)RFAL_NFCDEP_Dx_08_848) && (Bx >= (uint8_t)RFAL_NFCDEP_Bx_08_848) )
 | |
| 	{
 | |
| 		return true;
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipTxRx( rfalNfcDepCmd cmd, uint8_t* txBuf, uint32_t fwt, uint8_t* paylBuf, uint8_t paylBufLen, uint8_t* rxBuf, uint16_t rxBufLen, uint16_t *rxActLen )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     
 | |
|     if( (cmd == NFCIP_CMD_DEP_REQ) || (cmd == NFCIP_CMD_DEP_RES) ) /* this method cannot be used for DEPs */
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     /* Assign the global params for this TxRx */
 | |
|     gNfcip.rxBuf       = rxBuf;
 | |
|     gNfcip.rxBufLen    = rxBufLen;
 | |
|     gNfcip.rxRcvdLen   = rxActLen;
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
| 	/* Transmission                                                                */
 | |
| 	/*******************************************************************************/
 | |
|     if(txBuf != NULL)                                              /* if nothing to Tx, just do Rx */               
 | |
|     {
 | |
|         EXIT_ON_ERR( ret, nfcipTx( cmd, txBuf, paylBuf, paylBufLen, 0, fwt ) ); 
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
| 	/* Reception                                                                   */
 | |
| 	/*******************************************************************************/
 | |
|     ret = nfcipDataRx( true );
 | |
|     if( ret != ERR_NONE )
 | |
|     {
 | |
|         return ret;
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/    
 | |
|     *rxActLen = *rxBuf;                                      		/* Use LEN byte instead due to with/without CRC modes */
 | |
|     return ERR_NONE;                                         		/* Tx and Rx completed successfully                   */
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipDEPControlMsg( uint8_t pfb, uint8_t RTOX  )
 | |
| {
 | |
|     uint8_t        ctrlMsg[20];
 | |
|     rfalNfcDepCmd  depCmd;
 | |
|     uint32_t       fwt;
 | |
| 
 | |
| 
 | |
|     /*******************************************************************************/
 | |
|     /* Calculate Cmd and fwt to be used                                            */
 | |
|     /*******************************************************************************/
 | |
|     depCmd = ((gNfcip.cfg.role == RFAL_NFCDEP_ROLE_TARGET) ? NFCIP_CMD_DEP_RES : NFCIP_CMD_DEP_REQ);        
 | |
|     fwt    = ((gNfcip.cfg.role == RFAL_NFCDEP_ROLE_TARGET) ? NFCIP_NO_FWT : (nfcip_PFBisSTO( pfb ) ? ( (RTOX*gNfcip.cfg.fwt) + gNfcip.cfg.dFwt) : (gNfcip.cfg.fwt + gNfcip.cfg.dFwt) ) );
 | |
|     
 | |
|     if( nfcip_PFBisSTO( pfb ) )
 | |
|     {
 | |
|         ctrlMsg[RFAL_NFCDEP_DEPREQ_HEADER_LEN] = RTOX;
 | |
|         return nfcipTx( depCmd, ctrlMsg, &ctrlMsg[RFAL_NFCDEP_DEPREQ_HEADER_LEN], sizeof(uint8_t), pfb, fwt );
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         return nfcipTx( depCmd, ctrlMsg, NULL, 0, pfb, fwt );
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static void nfcipClearCounters( void )
 | |
| {
 | |
|     gNfcip.cntATNRetrys  = 0;
 | |
|     gNfcip.cntNACKRetrys = 0;
 | |
|     gNfcip.cntTORetrys   = 0;
 | |
|     gNfcip.cntTxRetrys   = 0;
 | |
|     gNfcip.cntRTOXRetrys = 0;
 | |
| }
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipInitiatorHandleDEP( ReturnCode rxRes, uint16_t rxLen, uint16_t *outActRxLen, bool *outIsChaining )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     uint8_t    nfcDepLen;
 | |
|     uint8_t    rxMsgIt;
 | |
|     uint8_t    rxPFB;
 | |
|     uint8_t    rxRTOX;
 | |
|     uint8_t    optHdrLen;
 | |
|     
 | |
|     ret        = ERR_INTERNAL;
 | |
|     rxMsgIt    = 0;
 | |
|     optHdrLen  = 0;
 | |
|     
 | |
|     *outActRxLen    = 0;
 | |
|     *outIsChaining  = false;
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Handle reception errors                                                     */
 | |
|     /*******************************************************************************/
 | |
|     switch( rxRes )
 | |
|     {
 | |
|         /*******************************************************************************/
 | |
|         /* Timeout ->  Digital 1.0 14.15.5.6 */
 | |
|         case ERR_TIMEOUT:
 | |
|             
 | |
|             nfcipLogI( " NFCIP(I) TIMEOUT  TORetrys:%d \r\n", gNfcip.cntTORetrys );
 | |
|             
 | |
|             /* Digital 1.0 14.15.5.6 - If nTO >= Max raise protocol error */
 | |
|             if( gNfcip.cntTORetrys++ >= RFAL_NFCDEP_TO_RETRYS )
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
| 
 | |
|             /*******************************************************************************/
 | |
|             /* Upon Timeout error, if Deactivation is pending, no more error recovery 
 | |
|              * will be done #54. 
 | |
|              * This is used to address the issue some devices that havea big TO. 
 | |
|              * Normally LLCP layer has timeout already, and NFCIP layer is still
 | |
|              * running error handling, retrying ATN/NACKs                                  */
 | |
|             /*******************************************************************************/
 | |
|             if( nfcipIsDeactivationPending() )
 | |
|             {
 | |
|                 nfcipLogI( " skipping error recovery due deactivation pending \r\n");
 | |
|                 return ERR_TIMEOUT;
 | |
|             }
 | |
|             
 | |
|             /* Digital 1.0 14.15.5.6 1)  If last PDU was NACK */
 | |
|             if( nfcip_PFBisRNACK(gNfcip.lastPFB) )
 | |
|             {
 | |
|                 /* Digital 1.0 14.15.5.6 2)  if NACKs failed raise protocol error  */
 | |
|                 if( gNfcip.cntNACKRetrys++ >= RFAL_NFCDEP_MAX_NACK_RETRYS )
 | |
|                 {
 | |
|                     return ERR_PROTO;
 | |
|                 }
 | |
|                 
 | |
|                 /* Send NACK */
 | |
|                 nfcipLogI( " NFCIP(I) Sending NACK retry: %d \r\n", gNfcip.cntNACKRetrys );
 | |
|                 EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBRPDU_NACK(gNfcip.pni), 0 ) );
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|             
 | |
|             nfcipLogI( " NFCIP(I) Checking if to send ATN  ATNRetrys: %d \r\n", gNfcip.cntATNRetrys );
 | |
|             
 | |
|             /* Digital 1.0 14.15.5.6 3)  Otherwise send ATN */                            
 | |
|             if( gNfcip.cntATNRetrys++ >= RFAL_NFCDEP_MAX_NACK_RETRYS )
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|                             
 | |
|             /* Send ATN */
 | |
|             nfcipLogI( " NFCIP(I) Sending ATN \r\n" );
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBSPDU_ATN(), 0 ) );
 | |
|             return ERR_BUSY;
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         /* Data rcvd with error ->  Digital 1.0 14.12.5.4 */
 | |
|         case ERR_CRC:
 | |
|         case ERR_PAR:
 | |
|         case ERR_FRAMING:
 | |
|         case ERR_RF_COLLISION:
 | |
|             
 | |
|             nfcipLogI( " NFCIP(I) rx Error: %d \r\n", rxRes );
 | |
|            
 | |
|             /* Digital 1.0 14.12.5.4 Tx Error with data, ignore */
 | |
|             if( rxLen < NFCIP_MIN_TXERROR_LEN )
 | |
|             {
 | |
|                 nfcipLogI( " NFCIP(I) Transmission error w data  \r\n" );
 | |
| #if 0
 | |
|                 if(gNfcip.cfg.commMode == RFAL_NFCDEP_COMM_PASSIVE)
 | |
|                 {
 | |
|                     nfcipLogI( " NFCIP(I) Transmission error w data -> reEnabling Rx \r\n" );
 | |
|                     nfcipReEnableRxTout( NFCIP_TRECOV );
 | |
|                     return ERR_BUSY;
 | |
|                 }
 | |
| #endif /* 0 */
 | |
|             }
 | |
| 
 | |
|             /* Digital 1.1 16.12.5.4  if NACKs failed raise Transmission error  */
 | |
|             if( gNfcip.cntNACKRetrys++ >= RFAL_NFCDEP_MAX_NACK_RETRYS )
 | |
|             {
 | |
|                 return ERR_FRAMING;
 | |
|             }
 | |
|                             
 | |
|             /* Send NACK */
 | |
|             nfcipLogI( " NFCIP(I) Sending NACK  \r\n" );
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBRPDU_NACK(gNfcip.pni), 0 ) );
 | |
|             return ERR_BUSY;
 | |
|             
 | |
|         case ERR_NONE:
 | |
|             break;
 | |
|             
 | |
|         case ERR_BUSY:
 | |
|             return ERR_BUSY;  /* Debug purposes */
 | |
|             
 | |
|         default:
 | |
|             nfcipLogW( " NFCIP(I) Error: %d \r\n", rxRes );
 | |
|             return rxRes;
 | |
|     }    
 | |
|         
 | |
|     /*******************************************************************************/
 | |
|     /* Rx OK check if valid DEP PDU                                                */
 | |
|     /*******************************************************************************/
 | |
|     
 | |
|     /* Due to different modes on ST25R391x (with/without CRC) use NFC-DEP LEN instead of bytes retrieved */
 | |
|     nfcDepLen = gNfcip.rxBuf[rxMsgIt++];
 | |
|     
 | |
|     nfcipLogD( " NFCIP(I) rx OK: %d bytes \r\n", nfcDepLen );
 | |
|     
 | |
|     /* Digital 1.0 14.15.5.5 Protocol Error  */
 | |
|     if( gNfcip.rxBuf[rxMsgIt++] != NFCIP_RES )
 | |
|     {
 | |
|     	nfcipLogW( " NFCIP(I) error %02X instead of %02X \r\n", gNfcip.rxBuf[--rxMsgIt], NFCIP_RES );
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     /* Digital 1.0 14.15.5.5 Protocol Error  */
 | |
|     if( gNfcip.rxBuf[rxMsgIt++] != (uint8_t)NFCIP_CMD_DEP_RES )
 | |
|     {
 | |
|     	nfcipLogW( " NFCIP(I) error %02X instead of %02X \r\n", gNfcip.rxBuf[--rxMsgIt], NFCIP_CMD_DEP_RES );
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     rxPFB = gNfcip.rxBuf[rxMsgIt++];
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Check for valid PFB type                                                    */
 | |
|     if( !(nfcip_PFBisSPDU( rxPFB ) || nfcip_PFBisRPDU( rxPFB ) || nfcip_PFBisIPDU( rxPFB )) )
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|         
 | |
|     /*******************************************************************************/
 | |
|     /* Digital 1.0 14.8.2.1  check if DID is expected and match -> Protocol Error  */
 | |
|     if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO ) 
 | |
|     {
 | |
|         if( (gNfcip.rxBuf[rxMsgIt++] != gNfcip.cfg.did) || !nfcip_PFBhasDID( rxPFB ) )
 | |
|         {
 | |
|             return ERR_PROTO;
 | |
|         }
 | |
|         optHdrLen++;                                    /* Inc header optional field cnt*/
 | |
|     }
 | |
|     else if( nfcip_PFBhasDID( rxPFB ) )                 /* DID not expected but rcv */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         /* MISRA 15.7 - Empty else */
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Digital 1.0 14.6.2.8 & 14.6.3.11 NAD must not be used  */
 | |
|     if( gNfcip.cfg.nad != RFAL_NFCDEP_NAD_NO ) 
 | |
|     {
 | |
|         if( (gNfcip.rxBuf[rxMsgIt++] != gNfcip.cfg.nad) || !nfcip_PFBhasNAD( rxPFB ) )
 | |
|         {
 | |
|             return ERR_PROTO;
 | |
|         }        
 | |
|         optHdrLen++;                                    /* Inc header optional field cnt*/
 | |
|     }
 | |
|     else if( nfcip_PFBhasNAD( rxPFB ) )                 /* NAD not expected but rcv */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         /* MISRA 15.7 - Empty else */
 | |
|     }
 | |
|        
 | |
|     /*******************************************************************************/
 | |
|     /* Process R-PDU                                                               */
 | |
|     /*******************************************************************************/
 | |
|     if( nfcip_PFBisRPDU( rxPFB ) )
 | |
|     {
 | |
|         /*******************************************************************************/
 | |
|         /* R ACK                                                                       */
 | |
|         /*******************************************************************************/
 | |
|         if( nfcip_PFBisRACK( rxPFB ) )
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(I) Rcvd ACK  \r\n" );
 | |
|             if( gNfcip.pni == nfcip_PBF_PNI( rxPFB ) )
 | |
|             {
 | |
|                 /* 14.12.3.3 R-ACK with correct PNI -> Increment */
 | |
|                 gNfcip.pni = nfcip_PNIInc( gNfcip.pni );
 | |
|                                 
 | |
|                 /* R-ACK while not performing chaining -> Protocol error*/
 | |
|                 if( !gNfcip.isTxChaining )
 | |
|                 {
 | |
|                     return ERR_PROTO;
 | |
|                 }
 | |
|                 
 | |
|                 nfcipClearCounters();
 | |
|                 gNfcip.state = NFCIP_ST_INIT_DEP_IDLE;
 | |
|                 return ERR_NONE;                            /* This block has been transmitted */
 | |
|             }
 | |
|             else  /* Digital 1.0 14.12.4.5 ACK with wrong PNI Initiator may retransmit */
 | |
|             {
 | |
|                 if( gNfcip.cntTxRetrys++ >= RFAL_NFCDEP_MAX_TX_RETRYS )
 | |
|                 {
 | |
|                     return ERR_PROTO;
 | |
|                 }
 | |
|                 
 | |
|                 /* Extended the MAY in Digital 1.0 14.12.4.5 to only reTransmit if the ACK
 | |
|                  * is for the previous DEP, otherwise raise Protocol immediately 
 | |
|                  * If the PNI difference is more than 1 it is worthless to reTransmit 3x
 | |
|                  * and after raise the error                                              */
 | |
|                 
 | |
|                 if( nfcip_PNIDec( gNfcip.pni ) ==  nfcip_PBF_PNI( rxPFB ) )
 | |
|                 {
 | |
|                     /* ReTransmit */
 | |
|                     nfcipLogI( " NFCIP(I) Rcvd ACK prev PNI -> reTx \r\n" );
 | |
|                     gNfcip.state = NFCIP_ST_INIT_DEP_TX;
 | |
|                     return ERR_BUSY;
 | |
|                 }
 | |
|                 
 | |
|                 nfcipLogI( " NFCIP(I) Rcvd ACK unexpected far PNI -> Error \r\n" );
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|         }
 | |
|         else /* Digital 1.0 - 14.12.5.2 Target must never send NACK  */
 | |
|         {            
 | |
|             return ERR_PROTO;
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Process S-PDU                                                               */
 | |
|     /*******************************************************************************/
 | |
|     if( nfcip_PFBisSPDU( rxPFB ) )                                
 | |
|     {
 | |
|         nfcipLogI( " NFCIP(I) Rcvd S-PDU  \r\n" );
 | |
|         /*******************************************************************************/
 | |
|         /* S ATN                                                                       */
 | |
|         /*******************************************************************************/
 | |
|         if( nfcip_PFBisSATN( rxPFB ) )                         /* If is a S-ATN        */
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(I) Rcvd ATN  \r\n" );
 | |
|             if( nfcip_PFBisSATN( gNfcip.lastPFB ) )            /* Check if is expected */
 | |
|             {  
 | |
|                 gNfcip.cntATNRetrys = 0;                       /* Clear ATN counter    */
 | |
|                 
 | |
|                 /* Although spec is not clear NFC Forum Digital test is expecting to
 | |
|                  * retransmit upon receiving ATN_RES */
 | |
|                 if( nfcip_PFBisSTO( gNfcip.lastPFBnATN ) )
 | |
|                 {
 | |
|                     nfcipLogI( " NFCIP(I) Rcvd ATN  -> reTx RTOX_RES \r\n" );
 | |
|                     EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBSPDU_TO(), gNfcip.lastRTOX ) );
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     /* ReTransmit ? */
 | |
|                     if( gNfcip.cntTxRetrys++ >= RFAL_NFCDEP_MAX_TX_RETRYS )
 | |
|                     {
 | |
|                         return ERR_PROTO;
 | |
|                     }
 | |
|                     
 | |
|                     nfcipLogI( " NFCIP(I) Rcvd ATN  -> reTx  PNI: %d \r\n", gNfcip.pni );
 | |
|                     gNfcip.state = NFCIP_ST_INIT_DEP_TX;
 | |
|                 }
 | |
|                 
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|             else                                               /* Digital 1.0  14.12.4.4 & 14.12.4.8 */
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|         }
 | |
|         /*******************************************************************************/
 | |
|         /* S TO                                                                        */
 | |
|         /*******************************************************************************/
 | |
|         else if( nfcip_PFBisSTO( rxPFB ) )                     /* If is a S-TO (RTOX)  */
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(I) Rcvd TO  \r\n" );
 | |
|             
 | |
|             rxRTOX = gNfcip.rxBuf[rxMsgIt++];
 | |
|             
 | |
|             /* Digital 1.1 16.12.4.3 - Initiator MAY stop accepting subsequent RTOX Req   *
 | |
|              *                       - RTOX request to an ATN -> Protocol error           */
 | |
|             if( (gNfcip.cntRTOXRetrys++ > RFAL_NFCDEP_MAX_RTOX_RETRYS) || nfcip_PFBisSATN( gNfcip.lastPFB ) )
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|             
 | |
|             /* Digital 1.1 16.8.4.1 RTOX must be between [1,59] */
 | |
|             if( (rxRTOX < NFCIP_INIT_MIN_RTOX) || (rxRTOX > NFCIP_INIT_MAX_RTOX) )
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|             
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBSPDU_TO(), rxRTOX ) );
 | |
|             gNfcip.lastRTOX = rxRTOX;
 | |
|             
 | |
|             return ERR_BUSY;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             /* Unexpected S-PDU */
 | |
|             return ERR_PROTO;                       /*  PRQA S  2880 # MISRA 2.1 - Guard code to prevent unexpected behavior */
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Process I-PDU                                                               */
 | |
|     /*******************************************************************************/
 | |
|     if( nfcip_PFBisIPDU( rxPFB ) )
 | |
|     {
 | |
|         if( gNfcip.pni != nfcip_PBF_PNI( rxPFB ) )
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(I) Rcvd IPDU wrong PNI     curPNI: %d rxPNI: %d \r\n", gNfcip.pni  , nfcip_PBF_PNI( rxPFB ) );
 | |
|             return ERR_PROTO;
 | |
|         }
 | |
|         
 | |
|         nfcipLogD( " NFCIP(I) Rcvd IPDU OK    PNI: %d \r\n", gNfcip.pni );
 | |
|         
 | |
|         /* 14.12.3.3 I-PDU with correct PNI -> Increment */
 | |
|         gNfcip.pni = nfcip_PNIInc( gNfcip.pni );
 | |
|                 
 | |
|         
 | |
|         /* Successful data Exchange */
 | |
|         nfcipClearCounters();
 | |
|         *outActRxLen  = ((uint16_t)nfcDepLen - RFAL_NFCDEP_DEP_HEADER - (uint16_t)optHdrLen);
 | |
|         
 | |
|         if( (&gNfcip.rxBuf[gNfcip.rxBufPaylPos] != &gNfcip.rxBuf[RFAL_NFCDEP_DEP_HEADER + optHdrLen]) && (*outActRxLen > 0U) )
 | |
|         {
 | |
|             ST_MEMMOVE( &gNfcip.rxBuf[gNfcip.rxBufPaylPos], &gNfcip.rxBuf[RFAL_NFCDEP_DEP_HEADER + optHdrLen], *outActRxLen );
 | |
|         }
 | |
| 
 | |
|         /*******************************************************************************/
 | |
|         /* Check if target is indicating chaining MI                                   */
 | |
|         /*******************************************************************************/
 | |
|         if( nfcip_PFBisIMI( rxPFB ) )
 | |
|         {
 | |
|             gNfcip.isRxChaining = true;
 | |
|             *outIsChaining      = true;
 | |
|             
 | |
|             nfcipLogD( " NFCIP(I) Rcvd IPDU OK w MI -> ACK \r\n" );
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBRPDU_ACK( gNfcip.pni ), gNfcip.rxBuf[rxMsgIt++] ) );
 | |
|             
 | |
|             return ERR_AGAIN;  /* Send Again signalling to run again, but some chaining data has arrived*/
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             gNfcip.isRxChaining = false;
 | |
|             gNfcip.state        = NFCIP_ST_INIT_DEP_IDLE;
 | |
|             
 | |
|             ret = ERR_NONE;    /* Data exchange done */
 | |
|         }
 | |
|     }
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipTargetHandleRX( ReturnCode rxRes, uint16_t *outActRxLen, bool *outIsChaining )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     uint8_t    nfcDepLen;
 | |
|     uint8_t    rxMsgIt;
 | |
|     uint8_t    rxPFB;
 | |
|     uint8_t    optHdrLen;
 | |
|     uint8_t    resBuf[RFAL_NFCDEP_HEADER_PAD + NFCIP_TARGET_RES_MAX];
 | |
|         
 | |
|     
 | |
|     ret        = ERR_INTERNAL;
 | |
|     rxMsgIt    = 0;
 | |
|     optHdrLen  = 0;
 | |
|     
 | |
|     *outActRxLen    = 0;
 | |
|     *outIsChaining  = false;
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Handle reception errors                                                     */
 | |
|     /*******************************************************************************/
 | |
|     switch( rxRes )
 | |
|     {
 | |
|         /*******************************************************************************/        
 | |
|         case ERR_NONE:
 | |
|             break;
 | |
|             
 | |
|         case ERR_LINK_LOSS:
 | |
|             nfcipLogW( " NFCIP(T) Error: %d \r\n", rxRes );
 | |
|             return rxRes;
 | |
|             
 | |
|         case ERR_BUSY:
 | |
|             return ERR_BUSY;  /* Debug purposes */
 | |
|             
 | |
|         case ERR_TIMEOUT:
 | |
|         case ERR_CRC:
 | |
|         case ERR_PAR:
 | |
|         case ERR_FRAMING:
 | |
|         case ERR_PROTO:
 | |
|         default:
 | |
|             /* Digital 1.1  16.12.5.2 The Target MUST NOT attempt any error recovery.      *
 | |
|              * The Target MUST always stay in receive mode when a                          *
 | |
|              * Transmission Error or a Protocol Error occurs.                              *
 | |
|              *                                                                             *
 | |
|              * Do not push Transmission/Protocol Errors to upper layer in Listen Mode #766 */
 | |
|             
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY;
 | |
|     }    
 | |
|         
 | |
|     /*******************************************************************************/
 | |
|     /* Rx OK check if valid DEP PDU                                                */
 | |
|     /*******************************************************************************/
 | |
|     
 | |
|     /* Due to different modes on ST25R391x (with/without CRC) use NFC-DEP LEN instead of bytes retrieved */
 | |
|     nfcDepLen = gNfcip.rxBuf[rxMsgIt++];
 | |
|         
 | |
|     nfcipLogD( " NFCIP(T) rx OK: %d bytes \r\n", nfcDepLen );
 | |
|     
 | |
|     if( gNfcip.rxBuf[rxMsgIt++] != NFCIP_REQ )
 | |
|     {
 | |
|         nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|         return ERR_BUSY; /* ERR_PROTO - Ignore bad request */
 | |
|     }
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Check whether target rcvd a normal DEP or deactivation request              */
 | |
|     /*******************************************************************************/
 | |
|     switch( gNfcip.rxBuf[rxMsgIt++] )
 | |
|     {               
 | |
|         /*******************************************************************************/
 | |
|         case (uint8_t)NFCIP_CMD_DEP_REQ:
 | |
|             break;                                /* Continue to normal DEP processing */
 | |
|     
 | |
|         /*******************************************************************************/
 | |
|         case (uint8_t)NFCIP_CMD_DSL_REQ:
 | |
|             
 | |
|             nfcipLogI( " NFCIP(T) rx DSL \r\n" );
 | |
|             
 | |
|             /* Digital 1.0  14.9.1.2 If DID is used and incorrect ignore it */
 | |
|             /* [Digital 1.0, 16.9.1.2]: If DID == 0, Target SHALL ignore DSL_REQ with DID */
 | |
|             if (   (((gNfcip.rxBuf[rxMsgIt++] != gNfcip.cfg.did) || (nfcDepLen != RFAL_NFCDEP_DSL_RLS_LEN_DID)) && (gNfcip.cfg.did != RFAL_NFCDEP_DID_NO) )
 | |
|                 || ((gNfcip.cfg.did == RFAL_NFCDEP_DID_NO) && (nfcDepLen != RFAL_NFCDEP_DSL_RLS_LEN_NO_DID))
 | |
|                )
 | |
|             {
 | |
|                 nfcipLogI( " NFCIP(T) DSL wrong DID, ignoring \r\n" );
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|             
 | |
|             nfcipTx( NFCIP_CMD_DSL_RES, resBuf, NULL, 0, 0, NFCIP_NO_FWT );
 | |
|             
 | |
|             gNfcip.state = NFCIP_ST_TARG_DEP_SLEEP;
 | |
|             return ERR_SLEEP_REQ;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case (uint8_t)NFCIP_CMD_RLS_REQ:
 | |
|             
 | |
|             nfcipLogI( " NFCIP(T) rx RLS \r\n" );
 | |
|             
 | |
|             /* Digital 1.0  14.10.1.2 If DID is used and incorrect ignore it */
 | |
|             /* [Digital 1.0, 16.10.2.2]: If DID == 0, Target SHALL ignore DSL_REQ with DID */
 | |
|             if (   (((gNfcip.rxBuf[rxMsgIt++] != gNfcip.cfg.did) || (nfcDepLen != RFAL_NFCDEP_DSL_RLS_LEN_DID)) && (gNfcip.cfg.did != RFAL_NFCDEP_DID_NO) )
 | |
|                 || ((gNfcip.cfg.did == RFAL_NFCDEP_DID_NO) && (nfcDepLen > RFAL_NFCDEP_DSL_RLS_LEN_NO_DID))
 | |
|                )
 | |
|             {
 | |
|                 nfcipLogI( " NFCIP(T) RLS wrong DID, ignoring \r\n" );
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|                 
 | |
|             nfcipTx( NFCIP_CMD_RLS_RES, resBuf, NULL, 0, 0, NFCIP_NO_FWT );
 | |
|             
 | |
|             gNfcip.state = NFCIP_ST_TARG_DEP_IDLE;
 | |
|             return ERR_RELEASE_REQ;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         /*case NFCIP_CMD_PSL_REQ:              PSL must be handled in Activation only */
 | |
|         /*case NFCIP_CMD_WUP_REQ:              WUP not in NFC Forum Digital 1.0       */
 | |
|         default:
 | |
|             
 | |
|             /* Don't go to NFCIP_ST_TARG_DEP_IDLE state as it needs to ignore this    *
 | |
|              * invalid frame, and keep waiting for more frames                        */
 | |
|             
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY; /* ERR_PROTO - Ignore bad frame */
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     
 | |
|     rxPFB = gNfcip.rxBuf[rxMsgIt++];                    /* Store rcvd PFB  */
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Check for valid PFB type                                                    */
 | |
|     if( !(nfcip_PFBisSPDU( rxPFB ) || nfcip_PFBisRPDU( rxPFB ) || nfcip_PFBisIPDU( rxPFB )) )
 | |
|     {
 | |
|         nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|         return ERR_BUSY; /* ERR_PROTO - Ignore invalid PFB  */
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/   
 | |
|     if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO ) 
 | |
|     {
 | |
|         if( !nfcip_PFBhasDID( rxPFB ) )
 | |
|         {
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY; /* ERR_PROTO - Ignore bad/missing DID  */
 | |
|         }
 | |
|         if( gNfcip.rxBuf[rxMsgIt++] != gNfcip.cfg.did ) /* MISRA 13.5 */
 | |
|         {
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY; /* ERR_PROTO - Ignore bad/missing DID  */
 | |
|         }
 | |
|         optHdrLen++;                                    /* Inc header optional field cnt*/
 | |
|     }
 | |
|     else if( nfcip_PFBhasDID( rxPFB ) )                 /* DID not expected but rcv     */
 | |
|     {
 | |
|         nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|         return ERR_BUSY; /* ERR_PROTO - Ignore unexpected DID  */
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         /* MISRA 15.7 - Empty else */
 | |
|     }
 | |
|                                   
 | |
|         
 | |
|     /*******************************************************************************/
 | |
|     if( gNfcip.cfg.nad != RFAL_NFCDEP_NAD_NO ) 
 | |
|     {
 | |
|         if( (gNfcip.rxBuf[rxMsgIt++] != gNfcip.cfg.did) || !nfcip_PFBhasDID( rxPFB ) )
 | |
|         {
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY;                            /* ERR_PROTO - Ignore bad/missing DID  */
 | |
|         }
 | |
|         optHdrLen++;                                    /* Inc header optional field cnt*/
 | |
|     }
 | |
|     else if( nfcip_PFBhasNAD( rxPFB ) )                 /* NAD not expected but rcv */
 | |
|     {
 | |
|         nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|         return ERR_BUSY;                                /* ERR_PROTO - Ignore unexpected NAD  */
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         /* MISRA 15.7 - Empty else */
 | |
|     }
 | |
|     
 | |
|        
 | |
|     /*******************************************************************************/
 | |
|     /* Process R-PDU                                                               */
 | |
|     /*******************************************************************************/
 | |
|     if( nfcip_PFBisRPDU( rxPFB ) )
 | |
|     {
 | |
|         nfcipLogD( " NFCIP(T) Rcvd R-PDU  \r\n" );
 | |
|         /*******************************************************************************/
 | |
|         /* R ACK                                                                       */
 | |
|         /*******************************************************************************/
 | |
|         if( nfcip_PFBisRACK( rxPFB ) )
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(T) Rcvd ACK  \r\n" );
 | |
|             if( gNfcip.pni == nfcip_PBF_PNI( rxPFB ) )
 | |
|             {
 | |
|                 /* R-ACK while not performing chaining -> Protocol error */
 | |
|                 if( !gNfcip.isTxChaining )
 | |
|                 {
 | |
|                     nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|                     return ERR_BUSY;                    /* ERR_PROTO - Ignore unexpected ACK  */
 | |
|                 }
 | |
|                 
 | |
|                 /* This block has been transmitted and acknowledged, perform RTOX until next data is provided  */
 | |
|                 
 | |
|                 /* Digital 1.1  16.12.4.7 - If ACK rcvd continue with chaining or an RTOX */
 | |
|                 nfcipTimerStart( gNfcip.RTOXTimer, nfcipRTOXAdjust( nfcipConv1FcToMs( rfalNfcDepWT2RWT( gNfcip.cfg.to ) )) );
 | |
|                 gNfcip.state = NFCIP_ST_TARG_DEP_RTOX;
 | |
|                 
 | |
|                 return ERR_NONE;                        /* This block has been transmitted */
 | |
|             }
 | |
|             
 | |
|             /* Digital 1.0 14.12.3.4 - If last send was ATN and rx PNI is minus 1 */
 | |
|             else if( nfcip_PFBisSATN( gNfcip.lastPFB ) &&  (nfcip_PNIDec(gNfcip.pni) == nfcip_PBF_PNI( rxPFB )) )    
 | |
|             {   
 | |
|                 nfcipLogI( " NFCIP(T) wrong PNI, last was ATN reTx  \r\n" );
 | |
|                 /* Spec says to leave current PNI as is, but will be Inc after Tx, remaining the same */
 | |
|                 gNfcip.pni = nfcip_PNIDec( gNfcip.pni );
 | |
|                 
 | |
|                 gNfcip.state = NFCIP_ST_TARG_DEP_TX;
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 /* MISRA 15.7 - Empty else */
 | |
|             }
 | |
|         }
 | |
|         /*******************************************************************************/
 | |
|         /* R NACK                                                                      */
 | |
|         /*******************************************************************************/
 | |
|         /* ISO 18092 12.6.1.3.3 When rcv NACK if PNI = prev PNI sent ->  reTx          */
 | |
|         else if( nfcip_PFBisRNACK( rxPFB ) && (nfcip_PNIDec(gNfcip.pni) == nfcip_PBF_PNI( rxPFB ) ) )
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(T) Rcvd NACK  \r\n" );
 | |
|             
 | |
|             gNfcip.pni = nfcip_PNIDec( gNfcip.pni );   /* Dec so that has the prev PNI */
 | |
|             
 | |
|             gNfcip.state = NFCIP_ST_TARG_DEP_TX;
 | |
|             return ERR_BUSY;
 | |
|         }
 | |
|         else
 | |
|         {        
 | |
|             nfcipLogI( " NFCIP(T) Unexpected R-PDU \r\n" );
 | |
|             
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY; /* ERR_PROTO - Ignore unexpected R-PDU  */
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Process S-PDU                                                               */
 | |
|     /*******************************************************************************/
 | |
|     if( nfcip_PFBisSPDU( rxPFB ) )
 | |
|     {
 | |
|         nfcipLogD( " NFCIP(T) Rcvd S-PDU  \r\n" );
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         /* S ATN                                                                       */
 | |
|         /*******************************************************************************/
 | |
|         /* ISO 18092 12.6.3 Attention                                                  */
 | |
|         if( nfcip_PFBisSATN( rxPFB ) )                         /*    If is a S-ATN     */
 | |
|         {            
 | |
|             nfcipLogI( " NFCIP(T) Rcvd ATN  curPNI: %d \r\n", gNfcip.pni );
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBSPDU_ATN(), 0 ) );
 | |
|             return ERR_BUSY;
 | |
|         }
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         /* S TO                                                                        */
 | |
|         /*******************************************************************************/
 | |
|         else if( nfcip_PFBisSTO( rxPFB ) )                     /* If is a S-TO (RTOX)  */
 | |
|         {
 | |
|             if( nfcip_PFBisSTO( gNfcip.lastPFBnATN ) )
 | |
|             {
 | |
|                 nfcipLogI( " NFCIP(T) Rcvd TO  \r\n" );
 | |
|                 
 | |
|                 /* Digital 1.1  16.8.4.6  RTOX value in RES different that in REQ -> Protocol Error */
 | |
|                 if( gNfcip.lastRTOX != gNfcip.rxBuf[rxMsgIt++] )
 | |
|                 {
 | |
|                     nfcipLogI( " NFCIP(T) Mismatched RTOX value \r\n" );
 | |
|                     
 | |
|                     nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|                     return ERR_BUSY; /* ERR_PROTO - Ignore unexpected RTOX value  */
 | |
|                 }
 | |
|                 
 | |
|                 /* Clear waiting for RTOX Ack Flag */
 | |
|                 gNfcip.isWait4RTOX = false;
 | |
|                 
 | |
|                 /* Check if a Tx is already pending */
 | |
|                 if( gNfcip.isTxPending )
 | |
|                 {
 | |
|                     nfcipLogW( " NFCIP(T) Tx pending, go immediately to TX \r\n" );
 | |
|                     
 | |
|                     gNfcip.state = NFCIP_ST_TARG_DEP_TX;
 | |
|                     return ERR_BUSY;
 | |
|                 }
 | |
|                 
 | |
|                 /* Start RTOX timer and change to check state  */
 | |
|                 nfcipTimerStart( gNfcip.RTOXTimer, nfcipRTOXAdjust( nfcipConv1FcToMs( gNfcip.lastRTOX * rfalNfcDepWT2RWT(gNfcip.cfg.to ) ) ) );
 | |
|                 gNfcip.state = NFCIP_ST_TARG_DEP_RTOX;
 | |
|                 
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             /* Unexpected S-PDU */
 | |
|             nfcipLogI( " NFCIP(T) Unexpected S-PDU \r\n" );         /*  PRQA S  2880 # MISRA 2.1 - Guard code to prevent unexpected behavior */
 | |
|             
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY; /* ERR_PROTO - Ignore unexpected S-PDU  */
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Process I-PDU                                                               */
 | |
|     /*******************************************************************************/
 | |
|     if( nfcip_PFBisIPDU( rxPFB ) )
 | |
|     {
 | |
|         if( gNfcip.pni != nfcip_PBF_PNI( rxPFB ) )
 | |
|         {
 | |
|             nfcipLogI( " NFCIP(T) Rcvd IPDU wrong PNI     curPNI: %d rxPNI: %d \r\n", gNfcip.pni, nfcip_PBF_PNI( rxPFB ) );
 | |
|             
 | |
|             /* Digital 1.1 16.12.3.4 - If last send was ATN and rx PNI is minus 1 */
 | |
|             if( nfcip_PFBisSATN(gNfcip.lastPFB ) &&  (nfcip_PNIDec(gNfcip.pni) == nfcip_PBF_PNI( rxPFB )) ) 
 | |
|             {
 | |
|                 /* Spec says to leave current PNI as is, but will be Inc after Data Tx, remaining the same */
 | |
|                 gNfcip.pni = nfcip_PNIDec(gNfcip.pni);
 | |
|                 
 | |
|                 if( nfcip_PFBisIMI( rxPFB ) )
 | |
|                 {
 | |
|                     nfcipLogI( " NFCIP(T) PNI = prevPNI && ATN before && chaining -> send ACK  \r\n" );
 | |
|                     EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBRPDU_ACK( gNfcip.pni ), gNfcip.rxBuf[rxMsgIt++] ) );
 | |
|                     
 | |
|                     /* Digital 1.1 16.12.3.4 (...) leave the current PNI unchanged afterwards */
 | |
|                     gNfcip.pni = nfcip_PNIInc( gNfcip.pni );
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     nfcipLogI( " NFCIP(T) PNI = prevPNI && ATN before -> reTx last I-PDU  \r\n" );
 | |
|                     gNfcip.state = NFCIP_ST_TARG_DEP_TX;
 | |
|                 }
 | |
|                 
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|                         
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             return ERR_BUSY;            /* ERR_PROTO - Ignore bad PNI value  */
 | |
|         }
 | |
|         
 | |
|         nfcipLogD( " NFCIP(T) Rcvd IPDU OK PNI: %d  \r\n", gNfcip.pni );
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         /* Successful data exchange                                                    */
 | |
|         /*******************************************************************************/
 | |
|         *outActRxLen  = ((uint16_t)nfcDepLen - RFAL_NFCDEP_DEP_HEADER - (uint16_t)optHdrLen);
 | |
|         
 | |
|         nfcipClearCounters();
 | |
| 
 | |
|         if( (&gNfcip.rxBuf[gNfcip.rxBufPaylPos] != &gNfcip.rxBuf[RFAL_NFCDEP_DEP_HEADER + optHdrLen]) && (*outActRxLen > 0U) )
 | |
|         {
 | |
|             ST_MEMMOVE( &gNfcip.rxBuf[gNfcip.rxBufPaylPos], &gNfcip.rxBuf[RFAL_NFCDEP_DEP_HEADER + optHdrLen], *outActRxLen );
 | |
|         }
 | |
|         
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         /* Check if Initiator is indicating chaining MI                                */
 | |
|         /*******************************************************************************/
 | |
|         if( nfcip_PFBisIMI( rxPFB ) )
 | |
|         {
 | |
|             gNfcip.isRxChaining = true;
 | |
|             *outIsChaining      = true;
 | |
|             
 | |
|             nfcipLogD( " NFCIP(T) Rcvd IPDU OK w MI -> ACK \r\n" );
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBRPDU_ACK( gNfcip.pni ), gNfcip.rxBuf[rxMsgIt++] ) );
 | |
|             
 | |
|             gNfcip.pni = nfcip_PNIInc( gNfcip.pni );
 | |
|             
 | |
|             return ERR_AGAIN;  /* Send Again signalling to run again, but some chaining data has arrived*/
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if(gNfcip.isRxChaining)
 | |
|             {
 | |
|                 nfcipLogI( " NFCIP(T) Rcvd last IPDU chaining finished \r\n" );
 | |
|             }
 | |
|             
 | |
|             /*******************************************************************************/
 | |
|             /* Reception done, send to DH and start RTOX timer                             */
 | |
|             /*******************************************************************************/
 | |
|             nfcipTimerStart( gNfcip.RTOXTimer, nfcipRTOXAdjust( nfcipConv1FcToMs( rfalNfcDepWT2RWT( gNfcip.cfg.to ) )) );
 | |
|             gNfcip.state = NFCIP_ST_TARG_DEP_RTOX;
 | |
|             
 | |
|             gNfcip.isRxChaining = false;
 | |
|             ret = ERR_NONE;                            /* Data exchange done */
 | |
|         }
 | |
|     }
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipTx( rfalNfcDepCmd cmd, uint8_t* txBuf, uint8_t *paylBuf, uint16_t paylLen, uint8_t pfbData, uint32_t fwt )
 | |
| {
 | |
|     uint16_t txBufIt;
 | |
|     uint8_t *txBlock;
 | |
|     uint8_t *payloadBuf;
 | |
|     uint8_t  pfb;
 | |
|     
 | |
|    
 | |
|     if( txBuf == NULL )
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
| 
 | |
|     payloadBuf = paylBuf;                                               /* MISRA 17.8: Use intermediate variable */
 | |
|     
 | |
|     if( (paylLen == 0U) || (payloadBuf == NULL) )
 | |
|     {
 | |
|         payloadBuf = (uint8_t*) &txBuf[RFAL_NFCDEP_DEPREQ_HEADER_LEN];  /* If not a DEP (no Data) ensure enough space for header */
 | |
|     }
 | |
|     
 | |
|     
 | |
|     txBufIt  = 0;
 | |
|     pfb      = pfbData;                                                 /* MISRA 17.8: Use intermediate variable */
 | |
|     
 | |
|     txBlock  = payloadBuf;                                              /* Point to beginning of the Data, and go backwards     */    
 | |
|         
 | |
|     
 | |
|     gNfcip.lastCmd = (uint8_t)cmd;                                      /* Store last cmd sent    */
 | |
|     gNfcip.lastPFB = NFCIP_PFB_INVALID;                                 /* Reset last pfb sent    */
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Compute outgoing NFCIP message                                              */
 | |
|     /*******************************************************************************/
 | |
|     switch( cmd )
 | |
|     {
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_CMD_ATR_RES:
 | |
|         case NFCIP_CMD_ATR_REQ:
 | |
|             
 | |
|             rfalNfcDepSetNFCID( payloadBuf, gNfcip.cfg.nfcid, gNfcip.cfg.nfcidLen );    /* NFCID */
 | |
|             txBufIt += RFAL_NFCDEP_NFCID3_LEN;
 | |
|             
 | |
|             payloadBuf[txBufIt++] = gNfcip.cfg.did;                                     /* DID   */
 | |
|             payloadBuf[txBufIt++] = gNfcip.cfg.bs;                                      /* BS    */
 | |
|             payloadBuf[txBufIt++] = gNfcip.cfg.br;                                      /* BR    */
 | |
|             
 | |
|             if( cmd == NFCIP_CMD_ATR_RES )
 | |
|             {
 | |
|                 payloadBuf[txBufIt++] = gNfcip.cfg.to;                                  /* ATR_RES[ TO ] */
 | |
|             }
 | |
|                                     
 | |
|             if( gNfcip.cfg.gbLen > 0U)
 | |
|             {
 | |
|                 payloadBuf[txBufIt++] = nfcip_PPwGB( gNfcip.cfg.lr );                   /* PP signalling GB  */
 | |
|                 ST_MEMCPY( &payloadBuf[txBufIt], gNfcip.cfg.gb, gNfcip.cfg.gbLen );     /* set General Bytes */
 | |
|                 txBufIt += gNfcip.cfg.gbLen;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 payloadBuf[txBufIt++] = rfalNfcDepLR2PP( gNfcip.cfg.lr );               /* PP without GB     */
 | |
|             }
 | |
|             
 | |
|             if( (txBufIt + RFAL_NFCDEP_CMDTYPE_LEN + RFAL_NFCDEP_CMD_LEN) > RFAL_NFCDEP_ATRREQ_MAX_LEN )   /* Check max ATR length (ATR_REQ = ATR_RES)*/
 | |
|             {
 | |
|                 return ERR_PARAM;
 | |
|             }
 | |
|             break;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_CMD_WUP_REQ:                               /* ISO 18092 - 12.5.2.1 */
 | |
|             
 | |
|             rfalNfcDepSetNFCID( (payloadBuf), gNfcip.cfg.nfcid, gNfcip.cfg.nfcidLen );   /* NFCID */
 | |
|             txBufIt += RFAL_NFCDEP_NFCID3_LEN;
 | |
|             
 | |
|             *(--txBlock) = gNfcip.cfg.did;                                               /* DID   */
 | |
|             break;
 | |
|                     
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_CMD_WUP_RES:                               /* ISO 18092 - 12.5.2.2 */
 | |
|         case NFCIP_CMD_PSL_REQ:
 | |
|         case NFCIP_CMD_PSL_RES:
 | |
|             
 | |
|             *(--txBlock) = gNfcip.cfg.did;                                               /* DID   */
 | |
|             break;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_CMD_RLS_REQ:
 | |
|         case NFCIP_CMD_RLS_RES:
 | |
|         case NFCIP_CMD_DSL_REQ:
 | |
|         case NFCIP_CMD_DSL_RES:
 | |
|             
 | |
|             /* Digital 1.0 - 14.8.1.1 & 14.9.1.1 & 14.10.1.1 Only add DID if not 0 */
 | |
|             if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO )
 | |
|             {
 | |
|                 *(--txBlock) = gNfcip.cfg.did;                                           /* DID   */
 | |
|             }
 | |
|             break;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_CMD_DEP_REQ:
 | |
|         case NFCIP_CMD_DEP_RES:
 | |
|             
 | |
|             /* Compute optional PFB bits */
 | |
|             if (gNfcip.cfg.did != RFAL_NFCDEP_DID_NO)                {   pfb |= NFCIP_PFB_DID_BIT;       }
 | |
|             if (gNfcip.cfg.nad != RFAL_NFCDEP_NAD_NO)                {   pfb |= NFCIP_PFB_NAD_BIT;       }
 | |
|             if ((gNfcip.isTxChaining) && (nfcip_PFBisIPDU(pfb)) )    {   pfb |= NFCIP_PFB_MI_BIT;        }
 | |
|             
 | |
|             /* Store PFB for future handling */
 | |
|             gNfcip.lastPFB       = pfb;                                                  /* store PFB sent */
 | |
|             
 | |
|             if( !nfcip_PFBisSATN(pfb) )
 | |
|             {
 | |
|                 gNfcip.lastPFBnATN   = pfb;                                              /* store last PFB different then ATN */
 | |
|             }
 | |
|             
 | |
|             
 | |
|             /* Add NAD if it is to be supported */
 | |
|             if( gNfcip.cfg.nad != RFAL_NFCDEP_NAD_NO )      
 | |
|             {
 | |
|                 *(--txBlock) = gNfcip.cfg.nad;                                           /* NAD   */
 | |
|             }
 | |
|             
 | |
|             /* Digital 1.0 - 14.8.1.1 & 14.8.1.1 Only add DID if not 0 */
 | |
|             if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO )
 | |
|             {
 | |
|                 *(--txBlock) = gNfcip.cfg.did;                                           /* DID   */
 | |
|             }
 | |
|             
 | |
|             *(--txBlock) = pfb;                                                          /* PFB */
 | |
|                         
 | |
|             
 | |
|             /* NCI 1.0 - Check if Empty frames are allowed */
 | |
|             if( (paylLen == 0U) && nfcipIsEmptyDEPDisabled(gNfcip.cfg.oper) && nfcip_PFBisIPDU(pfb) )
 | |
|             {
 | |
|                 return ERR_PARAM;
 | |
|             }
 | |
|             break;
 | |
| 
 | |
|         /*******************************************************************************/
 | |
|         default:
 | |
|             return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Prepend Header                                                              */
 | |
|     /*******************************************************************************/    
 | |
|     *(--txBlock) = (uint8_t)cmd;                                                         /* CMD     */
 | |
|     *(--txBlock) = (uint8_t)( nfcipCmdIsReq(cmd) ? NFCIP_REQ : NFCIP_RES );              /* CMDType */
 | |
|         
 | |
|     
 | |
|     txBufIt += paylLen + (uint16_t)((uint32_t)payloadBuf - (uint32_t)txBlock);           /* Calculate overall buffer size */
 | |
|     
 | |
|     
 | |
|     if( txBufIt > gNfcip.fsc )                                                           /* Check if msg length violates the maximum payload size FSC */
 | |
|     {
 | |
|         return ERR_NOTSUPP;
 | |
|     }
 | |
|         
 | |
|     /*******************************************************************************/
 | |
|     return nfcipDataTx( txBlock, txBufIt, fwt );
 | |
| }
 | |
| 
 | |
| /*
 | |
|  ******************************************************************************
 | |
|  * GLOBAL FUNCTIONS
 | |
|  ******************************************************************************
 | |
|  */
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static void nfcipConfig( const rfalNfcDepConfigs * cfg )
 | |
| {
 | |
|     if (cfg == NULL) 
 | |
|     {
 | |
|         return;
 | |
|     }
 | |
|     
 | |
|     ST_MEMCPY(&gNfcip.cfg, cfg, sizeof(rfalNfcDepConfigs));          /* Copy given config to local       */
 | |
|     
 | |
|     gNfcip.cfg.to   = MIN( RFAL_NFCDEP_WT_TRG_MAX, gNfcip.cfg.to);   /* Ensure proper WT value           */
 | |
|     gNfcip.cfg.did  = nfcip_DIDMax( gNfcip.cfg.did );                /* Ensure proper DID value          */
 | |
|     gNfcip.fsc      = rfalNfcDepLR2FS( gNfcip.cfg.lr );              /* Calculate FSC based on given LR  */
 | |
|     
 | |
|     gNfcip.state = ( ( gNfcip.cfg.role ==  RFAL_NFCDEP_ROLE_TARGET) ? NFCIP_ST_TARG_WAIT_ATR : NFCIP_ST_INIT_IDLE );
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipRun( uint16_t *outActRxLen, bool *outIsChaining  )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     
 | |
|     ret = ERR_SYNTAX;
 | |
|     
 | |
|     nfcipLogD( " NFCIP Run() state: %d \r\n", gNfcip.state );
 | |
|     
 | |
|     switch( gNfcip.state )
 | |
|     {
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_ST_IDLE:
 | |
|         case NFCIP_ST_INIT_DEP_IDLE:
 | |
|         case NFCIP_ST_TARG_DEP_IDLE:
 | |
|         case NFCIP_ST_TARG_DEP_SLEEP:
 | |
|             return ERR_NONE;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_ST_INIT_DEP_TX:
 | |
|             
 | |
|             nfcipLogD( " NFCIP(I) Tx PNI: %d txLen: %d \r\n", gNfcip.pni, gNfcip.txBufLen );
 | |
|             ret = nfcipTx( NFCIP_CMD_DEP_REQ, gNfcip.txBuf, &gNfcip.txBuf[gNfcip.txBufPaylPos], gNfcip.txBufLen, nfcip_PFBIPDU( gNfcip.pni ), (gNfcip.cfg.fwt + gNfcip.cfg.dFwt) );
 | |
|                         
 | |
|             switch( ret )
 | |
|             {
 | |
|                 case ERR_NONE:
 | |
|                     gNfcip.state = NFCIP_ST_INIT_DEP_RX;
 | |
|                     break;
 | |
|                 
 | |
|                 case ERR_PARAM:
 | |
|                 default:
 | |
|                     gNfcip.state = NFCIP_ST_INIT_DEP_IDLE;
 | |
|                     return ret;
 | |
|             }
 | |
|             /* fall through */
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_ST_INIT_DEP_RX:          /*  PRQA S 2003 # MISRA 16.3 - Intentional fall through */
 | |
| 
 | |
|             ret = nfcipDataRx( false );
 | |
|             
 | |
|             if( ret != ERR_BUSY )
 | |
|             {
 | |
|                 ret = nfcipInitiatorHandleDEP( ret, *gNfcip.rxRcvdLen, outActRxLen, outIsChaining );
 | |
|             }
 | |
|             
 | |
|             break;
 | |
|             
 | |
|         /*******************************************************************************/    
 | |
|         case NFCIP_ST_TARG_DEP_RTOX:
 | |
|             
 | |
|             if( !nfcipTimerisExpired( gNfcip.RTOXTimer ) )                    /* Do nothing until RTOX timer has expired */
 | |
|             {
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|             
 | |
|             /* If we cannot send a RTOX raise a Timeout error so that we do not   
 | |
|              * hold the field On forever in AP2P                                  */
 | |
|             if( nfcipIsRTOXReqDisabled(gNfcip.cfg.oper) )
 | |
|             {
 | |
|                 /* We should reEnable Rx, and measure time between our field Off to 
 | |
|                  * either report link loss or recover               #287          */
 | |
|                 nfcipLogI( " NFCIP(T) RTOX not sent due to config, NOT reenabling Rx \r\n" );
 | |
|                 return ERR_TIMEOUT;
 | |
|             } 
 | |
| 
 | |
|             if( gNfcip.cntRTOXRetrys++ > RFAL_NFCDEP_MAX_RTOX_RETRYS )              /* Check maximum consecutive RTOX requests */
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|             
 | |
|             nfcipLogI( " NFCIP(T) RTOX sent \r\n" );
 | |
|             
 | |
|             gNfcip.lastRTOX = nfcip_RTOXTargMax(gNfcip.cfg.to);               /* Calculate requested RTOX value, and send it */                        
 | |
|             EXIT_ON_ERR( ret, nfcipDEPControlMsg( nfcip_PFBSPDU_TO(), gNfcip.lastRTOX ) );
 | |
|             
 | |
|             /* Set waiting for RTOX Ack Flag */
 | |
|             gNfcip.isWait4RTOX = true;
 | |
|             
 | |
|             gNfcip.state = NFCIP_ST_TARG_DEP_RX;                              /* Go back to Rx to process RTOX ack       */
 | |
|             return ERR_BUSY;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_ST_TARG_DEP_TX:
 | |
|             
 | |
|             nfcipLogD( " NFCIP(T) Tx PNI: %d txLen: %d \r\n", gNfcip.pni, gNfcip.txBufLen );
 | |
|             ret = nfcipTx( NFCIP_CMD_DEP_RES, gNfcip.txBuf, &gNfcip.txBuf[gNfcip.txBufPaylPos], gNfcip.txBufLen, nfcip_PFBIPDU( gNfcip.pni ), NFCIP_NO_FWT );
 | |
|             
 | |
|             /* Clear flags */
 | |
|             gNfcip.isTxPending = false;
 | |
|             gNfcip.isWait4RTOX = false;
 | |
|             
 | |
|             /* Digital 1.0 14.12.3.4 Increment the current PNI after Tx */
 | |
|             gNfcip.pni = nfcip_PNIInc( gNfcip.pni );
 | |
|             
 | |
|             switch( ret )
 | |
|             {
 | |
|                 case ERR_NONE:
 | |
|                     gNfcip.state = NFCIP_ST_TARG_DEP_RX;                        /* All OK, goto Rx state          */
 | |
|                     break;
 | |
|                 
 | |
|                 case ERR_PARAM:
 | |
|                 default:
 | |
|                     gNfcip.state = NFCIP_ST_TARG_DEP_IDLE;                      /* Upon Tx error, goto IDLE state */
 | |
|                     return ret;
 | |
|             }
 | |
|             /* fall through */
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         case NFCIP_ST_TARG_DEP_RX:   /*  PRQA S 2003 # MISRA 16.3 - Intentional fall through */
 | |
|             
 | |
|             if( gNfcip.isReqPending )    /* if already has Data should be from a DEP from nfcipTargetHandleActivation()  */
 | |
|             {
 | |
|                 nfcipLogD( " NFCIP(T) Skipping Rx Using DEP from Activation \r\n" );
 | |
|                 
 | |
|                 gNfcip.isReqPending = false;
 | |
|                 ret = ERR_NONE;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 ret = nfcipDataRx( false );
 | |
|             }
 | |
|             
 | |
|             if( ret != ERR_BUSY )
 | |
|             {
 | |
|                 ret = nfcipTargetHandleRX( ret, outActRxLen, outIsChaining );
 | |
|             }
 | |
|             
 | |
|             break;
 | |
|             
 | |
|         /*******************************************************************************/
 | |
|         default:
 | |
|             /* MISRA 16.4: no empty default statement (a comment being enough) */
 | |
|             break;
 | |
|     }
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| void rfalNfcDepSetDeactivatingCallback( rfalNfcDepDeactCallback pFunc )
 | |
| {
 | |
|     gNfcip.isDeactivating = pFunc;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| void rfalNfcDepInitialize( void )
 | |
| {
 | |
|     nfcipLogD( " NFCIP Ini() \r\n" );
 | |
| 	
 | |
|     gNfcip.state          = NFCIP_ST_IDLE;
 | |
|     gNfcip.isDeactivating = NULL;
 | |
|     
 | |
|     gNfcip.isTxPending    = false;
 | |
|     gNfcip.isWait4RTOX    = false;
 | |
|     gNfcip.isReqPending   = false;
 | |
|     
 | |
|             
 | |
|     gNfcip.cfg.oper  = (RFAL_NFCDEP_OPER_FULL_MI_DIS | RFAL_NFCDEP_OPER_EMPTY_DEP_EN | RFAL_NFCDEP_OPER_ATN_EN | RFAL_NFCDEP_OPER_RTOX_REQ_EN);
 | |
|     
 | |
|     gNfcip.cfg.did   = RFAL_NFCDEP_DID_NO;
 | |
|     gNfcip.cfg.nad   = RFAL_NFCDEP_NAD_NO;
 | |
|     
 | |
|     gNfcip.cfg.br    = RFAL_NFCDEP_Bx_NO_HIGH_BR;
 | |
|     gNfcip.cfg.bs    = RFAL_NFCDEP_Bx_NO_HIGH_BR;
 | |
|     
 | |
|     gNfcip.cfg.lr    = RFAL_NFCDEP_LR_254;
 | |
|     gNfcip.fsc       = rfalNfcDepLR2FS( gNfcip.cfg.lr );
 | |
|     
 | |
|     gNfcip.cfg.gbLen = 0;    
 | |
|     
 | |
|     gNfcip.cfg.fwt   = RFAL_NFCDEP_MAX_FWT;
 | |
|     gNfcip.cfg.dFwt  = RFAL_NFCDEP_MAX_FWT;
 | |
|     
 | |
|     gNfcip.pni       = 0;    
 | |
|     
 | |
|     /* Destroy any ongoing RTOX timer*/
 | |
|     nfcipTimerDestroy( gNfcip.RTOXTimer );
 | |
|     gNfcip.RTOXTimer = 0U;
 | |
|     
 | |
|     gNfcip.PDUTxPos = 0;
 | |
|     gNfcip.PDURxPos = 0;
 | |
|     gNfcip.PDUParam.rxLen = NULL;
 | |
|     gNfcip.PDUParam.rxBuf = NULL;
 | |
|     gNfcip.PDUParam.txBuf = NULL;
 | |
|     
 | |
| 
 | |
|     nfcipClearCounters();
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static void nfcipSetDEPParams( const rfalNfcDepDEPParams *DEPParams )
 | |
| {
 | |
|     nfcipLogD( " NFCIP SetDEP() txLen: %d \r\n", DEPParams->txBufLen );
 | |
| 	
 | |
|     gNfcip.isTxChaining = DEPParams->txChaining;
 | |
|     gNfcip.txBuf        = DEPParams->txBuf;
 | |
|     gNfcip.rxBuf        = DEPParams->rxBuf;
 | |
|     gNfcip.txBufLen     = DEPParams->txBufLen;
 | |
|     gNfcip.rxBufLen     = DEPParams->rxBufLen;
 | |
|     gNfcip.txBufPaylPos = DEPParams->txBufPaylPos;
 | |
|     gNfcip.rxBufPaylPos = DEPParams->rxBufPaylPos;
 | |
|     
 | |
|     if( DEPParams->did != RFAL_NFCDEP_DID_KEEP )
 | |
|     {
 | |
|         gNfcip.cfg.did  = nfcip_DIDMax( DEPParams->did );
 | |
|     }
 | |
|     
 | |
|     gNfcip.cfg.fwt      = DEPParams->fwt;
 | |
|     gNfcip.cfg.dFwt     = DEPParams->dFwt;
 | |
|     gNfcip.fsc          = DEPParams->fsc;
 | |
|     
 | |
|     
 | |
|     
 | |
|     if(gNfcip.cfg.role == RFAL_NFCDEP_ROLE_TARGET)
 | |
|     {
 | |
|         /* If there's any data to be sent go for Tx */
 | |
|         if(DEPParams->txBufLen > 0U)
 | |
|         {
 | |
|             /* Ensure that an RTOX Ack is not being expected at moment */
 | |
|             if( !gNfcip.isWait4RTOX )
 | |
|             {
 | |
|                 gNfcip.state = NFCIP_ST_TARG_DEP_TX;
 | |
|                 return;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 /* If RTOX Ack is expected, signal a pending Tx to be transmitted right after */
 | |
|                 gNfcip.isTxPending = true;
 | |
|                 nfcipLogW( " NFCIP(T) Waiting RTOX, queueing outgoing DEP Block \r\n" );                
 | |
|             }
 | |
|         }    
 | |
|     
 | |
|         /*Digital 1.0  14.12.4.1 In target mode the first PDU MUST be sent by the Initiator */
 | |
|         gNfcip.state = NFCIP_ST_TARG_DEP_RX;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     /* New data TxRx request clear previous error counters for consecutive TxRx without reseting communication/protocol layer*/
 | |
|     nfcipClearCounters();
 | |
|     
 | |
|     gNfcip.state = NFCIP_ST_INIT_DEP_TX;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| bool rfalNfcDepTargetRcvdATR( void )
 | |
| {
 | |
|     return ( (gNfcip.cfg.role == RFAL_NFCDEP_ROLE_TARGET) && nfcipIsTarget(gNfcip.state) && (gNfcip.state > NFCIP_ST_TARG_WAIT_ATR) );
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| bool rfalNfcDepIsAtrReq( const uint8_t* buf, uint16_t bufLen, uint8_t* nfcid3 )
 | |
| {
 | |
|     uint8_t msgIt;
 | |
|     
 | |
|     msgIt = 0;
 | |
|     
 | |
|     if ( (bufLen < RFAL_NFCDEP_ATRREQ_MIN_LEN) || (bufLen > RFAL_NFCDEP_ATRREQ_MAX_LEN) )
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     
 | |
|     if ( buf[msgIt++] != NFCIP_REQ )
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     
 | |
|     if( buf[msgIt++] != (uint8_t)NFCIP_CMD_ATR_REQ )
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     
 | |
|     /* Output NFID3 if requested */
 | |
|     if( nfcid3 != NULL )
 | |
|     {
 | |
|         ST_MEMCPY( nfcid3, &buf[RFAL_NFCDEP_ATR_REQ_NFCID3_POS], RFAL_NFCDEP_NFCID3_LEN );
 | |
|     }
 | |
|      
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipTargetHandleActivation( rfalNfcDepDevice *nfcDepDev, uint8_t *outBRS )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     uint8_t    msgIt;
 | |
|     uint8_t    txBuf[RFAL_NFCDEP_HEADER_PAD + NFCIP_PSLRES_LEN];
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /*  Check if we are in correct state                                           */
 | |
|     /*******************************************************************************/
 | |
|     if( gNfcip.state != NFCIP_ST_TARG_WAIT_ACTV )
 | |
|     {
 | |
|         return ERR_WRONG_STATE;
 | |
|     }
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /*  Check required parameters                                                  */
 | |
|     /*******************************************************************************/
 | |
|     if( outBRS == NULL )
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /*  Wait and process incoming cmd (PSL / DEP)                                  */
 | |
|     /*******************************************************************************/        
 | |
|     ret = nfcipDataRx( false );
 | |
| 
 | |
|     if( ret != ERR_NONE )
 | |
|     {
 | |
|         return ret;
 | |
|     }
 | |
|     
 | |
|     msgIt   = 0;
 | |
|     *outBRS = RFAL_NFCDEP_BRS_MAINTAIN;                   /* set out BRS to be maintained */
 | |
|  
 | |
|     msgIt++;                                              /* Skip LEN byte                */
 | |
|     
 | |
|     if ( gNfcip.rxBuf[msgIt++] != NFCIP_REQ )
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( gNfcip.rxBuf[msgIt] == (uint8_t)NFCIP_CMD_PSL_REQ )
 | |
|     {
 | |
|         msgIt++;
 | |
|         
 | |
|         if( gNfcip.rxBuf[msgIt++] != gNfcip.cfg.did )     /* Checking DID                 */
 | |
|         {
 | |
|             return ERR_PROTO;
 | |
|         }
 | |
|         
 | |
|         nfcipLogI( " NFCIP(T) PSL REQ rcvd \r\n" );
 | |
|         
 | |
|         *outBRS = gNfcip.rxBuf[msgIt++];                  /* assign output BRS value      */
 | |
|         
 | |
|         /* Store FSL(LR) and update current config */
 | |
|         gNfcip.cfg.lr = (gNfcip.rxBuf[msgIt++] & RFAL_NFCDEP_LR_VAL_MASK);
 | |
|         gNfcip.fsc    = rfalNfcDepLR2FS( gNfcip.cfg.lr );
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         /* Update NFC-DDE Device info */
 | |
|         if( nfcDepDev != NULL )
 | |
|         {
 | |
|             /* Update Bitrate info */
 | |
|             /* PRQA S 4342 2 # MISRA 10.5 - Layout of enum rfalBitRate and definition of rfalNfcDepBRS2DSI guarantee no invalid enum values to be created */
 | |
|             nfcDepDev->info.DSI = (rfalBitRate)rfalNfcDepBRS2DSI( *outBRS );   /* DSI codes the bit rate from Initiator to Target */
 | |
|             nfcDepDev->info.DRI = (rfalBitRate)rfalNfcDepBRS2DRI( *outBRS );   /* DRI codes the bit rate from Target to Initiator */
 | |
|             
 | |
|             /* Update Length Reduction and Frame Size */
 | |
|             nfcDepDev->info.LR = gNfcip.cfg.lr;
 | |
|             nfcDepDev->info.FS = gNfcip.fsc;
 | |
|             
 | |
|             /* Update PPi byte */
 | |
|             nfcDepDev->activation.Initiator.ATR_REQ.PPi &= ~RFAL_NFCDEP_PP_LR_MASK;
 | |
|             nfcDepDev->activation.Initiator.ATR_REQ.PPi |= rfalNfcDepLR2PP( gNfcip.cfg.lr );
 | |
|         }
 | |
|         
 | |
|         rfalSetBitRate( RFAL_BR_KEEP, gNfcip.nfcDepDev->info.DSI );
 | |
|         
 | |
|         EXIT_ON_ERR( ret, nfcipTx( NFCIP_CMD_PSL_RES, txBuf, NULL, 0, 0, NFCIP_NO_FWT ) );
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         if( gNfcip.rxBuf[msgIt] == (uint8_t)NFCIP_CMD_DEP_REQ )
 | |
|         {
 | |
|             msgIt++;
 | |
|                     
 | |
|             /*******************************************************************************/
 | |
|             /* Digital 1.0 14.12.3.1 PNI must be initialized to 0 */
 | |
|             if( nfcip_PBF_PNI( gNfcip.rxBuf[msgIt] ) != 0U )
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|             
 | |
|             /*******************************************************************************/
 | |
|             /* Digital 1.0 14.8.2.1  check if DID is expected and match -> Protocol Error  */
 | |
|             if( nfcip_PFBhasDID( gNfcip.rxBuf[ msgIt] ) ) 
 | |
|             {
 | |
|                 if( gNfcip.rxBuf[++msgIt] != gNfcip.cfg.did )
 | |
|                 {
 | |
|                     return ERR_PROTO;
 | |
|                 }
 | |
|             }
 | |
|             else if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO )          /* DID expected but not rcv */
 | |
|             {
 | |
|                 return ERR_PROTO;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 /* MISRA 15.7 - Empty else */
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         /* Signal Request pending to be digested on normal Handling (DEP_REQ, DSL_REQ, RLS_REQ) */
 | |
|         gNfcip.isReqPending = true;
 | |
|     }
 | |
|     
 | |
|     gNfcip.state = NFCIP_ST_TARG_DEP_RX;
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepATR( const rfalNfcDepAtrParam* param, rfalNfcDepAtrRes *atrRes, uint8_t* atrResLen )
 | |
| {
 | |
|     ReturnCode        ret;
 | |
|     rfalNfcDepConfigs cfg;
 | |
|     uint16_t          rxLen;
 | |
|     uint8_t           msgIt;
 | |
|     uint8_t           txBuf[RFAL_NFCDEP_ATRREQ_MAX_LEN];
 | |
|     uint8_t           rxBuf[NFCIP_ATRRES_BUF_LEN];
 | |
|     
 | |
|     
 | |
|     if( (param == NULL) || (atrRes == NULL) || (atrResLen == NULL) )
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Configure NFC-DEP layer                                                     */
 | |
|     /*******************************************************************************/
 | |
|     
 | |
|     cfg.did  = param->DID;
 | |
|     cfg.nad  = param->NAD;
 | |
|     cfg.fwt  = RFAL_NFCDEP_MAX_FWT;
 | |
|     cfg.dFwt = RFAL_NFCDEP_MAX_FWT;
 | |
|     cfg.br   = param->BR;
 | |
|     cfg.bs   = param->BS;
 | |
|     cfg.lr   = param->LR;
 | |
|     cfg.to   = RFAL_NFCDEP_WT_TRG_MAX;            /* Not used in Initiator mode */
 | |
|     
 | |
|     
 | |
|     cfg.gbLen = param->GBLen;
 | |
|     if( cfg.gbLen > 0U )                          /* MISRA 21.18 */
 | |
|     {
 | |
|         ST_MEMCPY( cfg.gb, param->GB, cfg.gbLen );
 | |
|     }
 | |
|     
 | |
|     cfg.nfcidLen = param->nfcidLen;
 | |
|     if( cfg.nfcidLen > 0U )                       /* MISRA 21.18 */
 | |
|     {
 | |
|         ST_MEMCPY( cfg.nfcid, param->nfcid, cfg.nfcidLen );
 | |
|     }
 | |
|     
 | |
|     cfg.role     = RFAL_NFCDEP_ROLE_INITIATOR;
 | |
|     cfg.oper     = param->operParam;
 | |
|     cfg.commMode = param->commMode;
 | |
| 
 | |
|     rfalNfcDepInitialize();
 | |
|     nfcipConfig( &cfg );
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Send ATR_REQ                                                                */
 | |
|     /*******************************************************************************/
 | |
|     
 | |
|     EXIT_ON_ERR( ret, nfcipTxRx(NFCIP_CMD_ATR_REQ, txBuf, nfcipRWTActivation(), NULL, 0, rxBuf, NFCIP_ATRRES_BUF_LEN, &rxLen ) );
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* ATR sent, check response                                                    */
 | |
|     /*******************************************************************************/
 | |
|     msgIt = 0;
 | |
|     rxLen = ((uint16_t)rxBuf[msgIt++] - RFAL_NFCDEP_LEN_LEN);                           /* use LEN byte             */
 | |
|     
 | |
|     if( (rxLen < RFAL_NFCDEP_ATRRES_MIN_LEN) || (rxLen > RFAL_NFCDEP_ATRRES_MAX_LEN) )  /* Checking length: ATR_RES */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[msgIt++] != NFCIP_RES )                                                   /* Checking if is a response*/
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[msgIt++] != (uint8_t)NFCIP_CMD_ATR_RES )                                  /* Checking if is a ATR RES */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     ST_MEMCPY( (uint8_t*)atrRes, (rxBuf + RFAL_NFCDEP_LEN_LEN), rxLen );
 | |
|     *atrResLen = (uint8_t)rxLen;
 | |
|     
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepPSL( uint8_t BRS, uint8_t FSL )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     uint16_t   rxLen;
 | |
|     uint8_t    msgIt;
 | |
|     uint8_t    txBuf[NFCIP_PSLREQ_LEN + NFCIP_PSLPAY_LEN];
 | |
|     uint8_t    rxBuf[NFCIP_PSLRES_LEN];
 | |
|     
 | |
|     msgIt = NFCIP_PSLREQ_LEN;
 | |
|     
 | |
|     txBuf[msgIt++] = BRS;
 | |
|     txBuf[msgIt++] = FSL;
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Send PSL REQ and wait for response                                          */
 | |
|     /*******************************************************************************/
 | |
|     EXIT_ON_ERR( ret, nfcipTxRx( NFCIP_CMD_PSL_REQ, txBuf, nfcipRWTActivation(), &txBuf[NFCIP_PSLREQ_LEN], (msgIt - NFCIP_PSLREQ_LEN), rxBuf, NFCIP_PSLRES_LEN, &rxLen ) );
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* PSL sent, check response                                                    */
 | |
|     /*******************************************************************************/
 | |
|     msgIt = 0;
 | |
|     rxLen = (uint16_t)(rxBuf[msgIt++]);                /* use LEN byte                   */
 | |
|         
 | |
|     if( rxLen < NFCIP_PSLRES_LEN )                     /* Checking length: LEN + RLS_RES */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[msgIt++] != NFCIP_RES )                  /* Checking if is a response      */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[msgIt++] != (uint8_t)NFCIP_CMD_PSL_RES ) /* Checking if is a PSL RES       */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[msgIt++] != gNfcip.cfg.did )             /* Checking DID                   */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepDSL( void )
 | |
| {   
 | |
|     ReturnCode ret;
 | |
|     uint8_t  txBuf[ RFAL_NFCDEP_HEADER_PAD + NFCIP_DSLREQ_LEN];
 | |
|     uint8_t  rxBuf[NFCIP_DSLRES_LEN];
 | |
|     uint8_t  rxMsgIt;
 | |
|     uint16_t rxLen = 0;
 | |
|         
 | |
|     if( gNfcip.cfg.role == RFAL_NFCDEP_ROLE_TARGET )
 | |
|     {
 | |
|         return ERR_NONE;                                  /* Target has no deselect procedure */
 | |
|     }
 | |
|     
 | |
|     /* Repeating a DSL REQ is optional, not doing it */
 | |
|     EXIT_ON_ERR( ret, nfcipTxRx( NFCIP_CMD_DSL_REQ, txBuf, nfcipRWTActivation(), NULL, 0, rxBuf, (uint16_t)sizeof(rxBuf), &rxLen  ) );
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     rxMsgIt = 0;
 | |
|     
 | |
|     if( rxBuf[rxMsgIt++] < NFCIP_DSLRES_MIN )             /* Checking length: LEN + DSL_RES */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[rxMsgIt++] != NFCIP_RES )                   /* Checking if is a response      */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[rxMsgIt++] != (uint8_t)NFCIP_CMD_DSL_RES )  /* Checking if is DSL RES          */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO ) 
 | |
|     {
 | |
|         if ( rxBuf[rxMsgIt++] != gNfcip.cfg.did ) 
 | |
|         {
 | |
|             return ERR_PROTO;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepRLS( void )
 | |
| {   
 | |
|     ReturnCode ret;
 | |
|     uint8_t    txBuf[RFAL_NFCDEP_HEADER_PAD + NFCIP_RLSREQ_LEN];
 | |
|     uint8_t    rxBuf[NFCIP_RLSRES_LEN];    
 | |
|     uint8_t    rxMsgIt;
 | |
|     uint16_t   rxLen = 0;
 | |
|     
 | |
|     if ( gNfcip.cfg.role == RFAL_NFCDEP_ROLE_TARGET )  /* Target has no release procedure */
 | |
|     {
 | |
|         return ERR_NONE;
 | |
|     }
 | |
|         
 | |
|     /* Repeating a RLS REQ is optional, not doing it */
 | |
|     EXIT_ON_ERR( ret, nfcipTxRx( NFCIP_CMD_RLS_REQ, txBuf, nfcipRWTActivation(), NULL, 0, rxBuf, (uint16_t)sizeof(rxBuf), &rxLen  ) );
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     rxMsgIt = 0;
 | |
|        
 | |
|     if( rxBuf[rxMsgIt++] < NFCIP_RLSRES_MIN )             /* Checking length: LEN + RLS_RES */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|         
 | |
|     if( rxBuf[rxMsgIt++] != NFCIP_RES )                   /* Checking if is a response      */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|     
 | |
|     if( rxBuf[rxMsgIt++] != (uint8_t)NFCIP_CMD_RLS_RES )  /* Checking if is RLS RES         */
 | |
|     {
 | |
|         return ERR_PROTO;
 | |
|     }
 | |
|      
 | |
|     if( gNfcip.cfg.did != RFAL_NFCDEP_DID_NO ) 
 | |
|     {
 | |
|         if ( rxBuf[rxMsgIt++] != gNfcip.cfg.did ) 
 | |
|         {
 | |
|             return ERR_PROTO;
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepInitiatorHandleActivation( rfalNfcDepAtrParam* param, rfalBitRate desiredBR, rfalNfcDepDevice* nfcDepDev )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     uint8_t    maxRetyrs;
 | |
|     uint8_t    PSL_BRS;
 | |
|     uint8_t    PSL_FSL;
 | |
|     bool       sendPSL;
 | |
|     
 | |
|     if( (param == NULL) || (nfcDepDev == NULL) )
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     param->NAD = RFAL_NFCDEP_NAD_NO;          /* Digital 1.1  16.6.2.9  Initiator SHALL NOT use NAD */
 | |
|     maxRetyrs  = NFCIP_ATR_RETRY_MAX;
 | |
|         
 | |
|     /*******************************************************************************/
 | |
|     /* Send ATR REQ and wait for response                                          */
 | |
|     /*******************************************************************************/
 | |
|     do{  /* Upon transmission error ATR REQ should be retried */
 | |
|         
 | |
|          ret = rfalNfcDepATR( param, &nfcDepDev->activation.Target.ATR_RES, &nfcDepDev->activation.Target.ATR_RESLen );
 | |
|          
 | |
|          if( nfcipIsTransmissionError(ret) )
 | |
|          {
 | |
|              continue;
 | |
|          }
 | |
|          break;
 | |
|     }
 | |
|     while( (maxRetyrs--) != 0U );
 | |
|     
 | |
|     if( ret != ERR_NONE )
 | |
|     {
 | |
|         return ret;
 | |
|     }
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Compute NFC-DEP device with ATR_RES                                         */
 | |
|     /*******************************************************************************/
 | |
|     nfcDepDev->info.GBLen = (nfcDepDev->activation.Target.ATR_RESLen - RFAL_NFCDEP_ATRRES_MIN_LEN);
 | |
|     nfcDepDev->info.DID   = nfcDepDev->activation.Target.ATR_RES.DID;
 | |
|     nfcDepDev->info.NAD   = RFAL_NFCDEP_NAD_NO;                                      /* Digital 1.1  16.6.3.11 Initiator SHALL ignore b1 of PPt */
 | |
|     nfcDepDev->info.LR    = rfalNfcDepPP2LR( nfcDepDev->activation.Target.ATR_RES.PPt );
 | |
|     nfcDepDev->info.FS    = rfalNfcDepLR2FS( nfcDepDev->info.LR );
 | |
|     nfcDepDev->info.WT    = (nfcDepDev->activation.Target.ATR_RES.TO & RFAL_NFCDEP_WT_MASK);
 | |
|     nfcDepDev->info.FWT   = rfalNfcDepCalculateRWT( nfcDepDev->info.WT );
 | |
|     nfcDepDev->info.dFWT  = RFAL_NFCDEP_WT_DELTA;
 | |
|     
 | |
|     rfalGetBitRate( &nfcDepDev->info.DSI, &nfcDepDev->info.DRI );
 | |
|     
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Check if a PSL needs to be sent                                                */
 | |
|     /*******************************************************************************/
 | |
|     sendPSL = false;
 | |
|     PSL_BRS = rfalNfcDepDx2BRS( nfcDepDev->info.DSI );  /* Set current bit rate divisor on both directions  */
 | |
|     PSL_FSL = nfcDepDev->info.LR;                       /* Set current Frame Size                           */
 | |
|     
 | |
|     
 | |
| 
 | |
|     /* Activity 1.0  9.4.4.15 & 9.4.6.3   NFC-DEP Activation PSL
 | |
|     *  Activity 2.0  9.4.4.17 & 9.4.6.6   NFC-DEP Activation PSL
 | |
|     *     
 | |
|     *  PSL_REQ shall only be sent if desired bit rate is different from current (Activity 1.0)
 | |
|     *  PSL_REQ shall be sent to update LR or bit rate  (Activity 2.0)
 | |
|     * */
 | |
| 
 | |
| #if 0 /* PSL due to LR is disabled, can be enabled if desired*/
 | |
|     /*******************************************************************************/
 | |
|     /* Check Frame Size                                                            */
 | |
|     /*******************************************************************************/
 | |
|     if( gNfcip.cfg.lr < nfcDepDev->info.LR )  /* If our Length reduction is smaller */   
 | |
|     {
 | |
|         sendPSL = true;
 | |
|         
 | |
|         nfcDepDev->info.LR   = MIN( nfcDepDev->info.LR, gNfcip.cfg.lr );
 | |
|         
 | |
|         gNfcip.cfg.lr = nfcDepDev->info.LR;                /* Update nfcip LR  to be used */
 | |
|         gNfcip.fsc    = rfalNfcDepLR2FS( gNfcip.cfg.lr );  /* Update nfcip FSC to be used */     
 | |
|         
 | |
|         PSL_FSL       = gNfcip.cfg.lr;                     /* Set LR to be sent           */
 | |
|         
 | |
|         nfcipLogI( " NFCIP(I) Frame Size differ, PSL new fsc: %d \r\n", gNfcip.fsc );
 | |
|     }
 | |
| #endif
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /* Check Baud rates                                                            */
 | |
|     /*******************************************************************************/
 | |
|     if( (nfcDepDev->info.DSI != desiredBR) && (desiredBR != RFAL_BR_KEEP) )    /* if desired BR is different    */
 | |
|     {
 | |
|         if( nfcipDxIsSupported( (uint8_t)desiredBR, nfcDepDev->activation.Target.ATR_RES.BRt, nfcDepDev->activation.Target.ATR_RES.BSt ) )  /* if desired BR is supported     */    /* MISRA 13.5 */
 | |
|         {
 | |
|             sendPSL = true;
 | |
|             PSL_BRS = rfalNfcDepDx2BRS( desiredBR );
 | |
|         
 | |
|             nfcipLogI( " NFCIP(I) BR differ, PSL BR: 0x%02X \r\n", PSL_BRS );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     if( sendPSL )
 | |
|     {
 | |
|         /*******************************************************************************/
 | |
|         /* Send PSL REQ and wait for response                                          */
 | |
|         /*******************************************************************************/
 | |
|         EXIT_ON_ERR( ret, rfalNfcDepPSL(PSL_BRS, PSL_FSL) );
 | |
|         
 | |
|         /* Check if bit rate has been changed */
 | |
|         if( nfcDepDev->info.DSI != desiredBR )
 | |
|         {
 | |
|             /* Check if device was in Passive NFC-A and went to higher bit rates, use NFC-F */
 | |
|             if( (nfcDepDev->info.DSI == RFAL_BR_106) && (gNfcip.cfg.commMode == RFAL_NFCDEP_COMM_PASSIVE) )
 | |
|             {
 | |
|                 
 | |
|             #if RFAL_FEATURE_NFCF 
 | |
|                 /* If Passive initialize NFC-F module */
 | |
|                 rfalNfcfPollerInitialize( desiredBR );
 | |
|             #else /* RFAL_FEATURE_NFCF */
 | |
|                 return ERR_NOTSUPP;
 | |
|             #endif /* RFAL_FEATURE_NFCF */
 | |
|                 
 | |
|             }
 | |
|             
 | |
|             nfcDepDev->info.DRI  = desiredBR;  /* DSI Bit Rate coding from Initiator  to Target  */
 | |
|             nfcDepDev->info.DSI  = desiredBR;  /* DRI Bit Rate coding from Target to Initiator   */
 | |
|             
 | |
|             rfalSetBitRate( nfcDepDev->info.DSI, nfcDepDev->info.DRI );
 | |
|         }
 | |
|         
 | |
|         
 | |
|         return ERR_NONE;   /* PSL has been sent    */
 | |
|     }
 | |
|     
 | |
|     return ERR_NONE;       /* No PSL has been sent */
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| uint32_t rfalNfcDepCalculateRWT( uint8_t wt )
 | |
| {
 | |
|     /* Digital 1.0  14.6.3.8  &  Digital 1.1  16.6.3.9     */
 | |
|     /* Digital 1.1  16.6.3.9 treat all RFU values as WT=14 */
 | |
|     uint8_t responseWaitTime = MIN( RFAL_NFCDEP_WT_INI_MAX, wt );
 | |
|         
 | |
|     return (uint32_t)rfalNfcDepWT2RWT(responseWaitTime);
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipDataTx( uint8_t* txBuf, uint16_t txBufLen, uint32_t fwt )
 | |
| {
 | |
|    return rfalTransceiveBlockingTx( txBuf, txBufLen, gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen, (RFAL_TXRX_FLAGS_DEFAULT | (uint32_t)RFAL_TXRX_FLAGS_NFCIP1_ON), ((fwt == NFCIP_NO_FWT) ? RFAL_FWT_NONE : fwt) );
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| static ReturnCode nfcipDataRx( bool blocking )
 | |
| {
 | |
|     ReturnCode ret;
 | |
|     
 | |
|     /* Perform Rx either blocking or non-blocking */
 | |
|     if( blocking )
 | |
|     {
 | |
|         ret = rfalTransceiveBlockingRx();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ret = rfalGetTransceiveStatus();
 | |
|     }
 | |
|     
 | |
|     if( ret != ERR_BUSY )
 | |
|     {
 | |
|         if( gNfcip.rxRcvdLen != NULL )
 | |
|         {
 | |
|             (*gNfcip.rxRcvdLen) = rfalConvBitsToBytes( *gNfcip.rxRcvdLen );
 | |
|                         
 | |
|             if( (ret == ERR_NONE) && (gNfcip.rxBuf != NULL) )
 | |
|             {
 | |
|                 /* Digital 1.1  16.4.1.3 - Length byte LEN SHALL have a value between 3 and 255 -> otherwise treat as Transmission Error *
 | |
|                  *                       - Ensure that actual received and frame length do match, otherwise treat as Transmission error  */
 | |
|                 if( (*gNfcip.rxRcvdLen != (uint16_t)*gNfcip.rxBuf) || (*gNfcip.rxRcvdLen < RFAL_NFCDEP_LEN_MIN) || (*gNfcip.rxRcvdLen > RFAL_NFCDEP_LEN_MAX) )
 | |
|                 {
 | |
|                     return ERR_FRAMING;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepListenStartActivation( const rfalNfcDepTargetParam *param, const uint8_t *atrReq, uint16_t atrReqLength, rfalNfcDepListenActvParam rxParam )
 | |
| {
 | |
|     ReturnCode        ret;
 | |
|     rfalNfcDepConfigs cfg;
 | |
|     
 | |
|     
 | |
|     if( (param == NULL) || (atrReq == NULL) || (rxParam.rxLen == NULL) )
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /*  Check whether is a valid ATR_REQ Compute NFC-DEP device                    */
 | |
|     if( !rfalNfcDepIsAtrReq( atrReq, atrReqLength, NULL ) )
 | |
|     {
 | |
|         return ERR_PARAM;
 | |
|     }
 | |
|     
 | |
|     rxParam.nfcDepDev->activation.Initiator.ATR_REQLen = (uint8_t)atrReqLength;                   /* nfcipIsAtrReq() is already checking Min and Max buffer lengths */ 
 | |
|     if( atrReqLength > 0U )                                                                       /* MISRA 21.18 */
 | |
|     {
 | |
|         ST_MEMCPY( (uint8_t*)&rxParam.nfcDepDev->activation.Initiator.ATR_REQ, atrReq, atrReqLength );
 | |
|     }        
 | |
|     
 | |
|     rxParam.nfcDepDev->info.GBLen = (uint8_t)(atrReqLength - RFAL_NFCDEP_ATRREQ_MIN_LEN);
 | |
|     rxParam.nfcDepDev->info.DID   = rxParam.nfcDepDev->activation.Initiator.ATR_REQ.DID;
 | |
|     rxParam.nfcDepDev->info.NAD   = RFAL_NFCDEP_NAD_NO;                        /* Digital 1.1  16.6.2.9  Initiator SHALL NOT use NAD */
 | |
|     rxParam.nfcDepDev->info.LR    = rfalNfcDepPP2LR( rxParam.nfcDepDev->activation.Initiator.ATR_REQ.PPi );
 | |
|     rxParam.nfcDepDev->info.FS    = rfalNfcDepLR2FS( rxParam.nfcDepDev->info.LR );
 | |
|     rxParam.nfcDepDev->info.WT    = 0;
 | |
|     rxParam.nfcDepDev->info.FWT   = NFCIP_NO_FWT;
 | |
|     rxParam.nfcDepDev->info.dFWT  = NFCIP_NO_FWT;
 | |
|     
 | |
|     rfalGetBitRate( &rxParam.nfcDepDev->info.DSI, &rxParam.nfcDepDev->info.DRI );
 | |
|     
 | |
|     
 | |
|     /* Store Device Info location, updated upon a PSL  */
 | |
|     gNfcip.nfcDepDev = rxParam.nfcDepDev;
 | |
|     
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     cfg.did = rxParam.nfcDepDev->activation.Initiator.ATR_REQ.DID;
 | |
|     cfg.nad = RFAL_NFCDEP_NAD_NO;
 | |
|     
 | |
|     cfg.fwt   = RFAL_NFCDEP_MAX_FWT;
 | |
|     cfg.dFwt  = RFAL_NFCDEP_MAX_FWT;
 | |
| 
 | |
|     cfg.br = param->brt;
 | |
|     cfg.bs = param->bst;
 | |
|     
 | |
|     cfg.lr = rfalNfcDepPP2LR(param->ppt);
 | |
|     
 | |
|     cfg.gbLen = param->GBtLen;
 | |
|     if( cfg.gbLen > 0U )          /* MISRA 21.18 */
 | |
|     {
 | |
|         ST_MEMCPY(cfg.gb, param->GBt, cfg.gbLen);
 | |
|     }
 | |
|     
 | |
|     cfg.nfcidLen = RFAL_NFCDEP_NFCID3_LEN;
 | |
|     ST_MEMCPY(cfg.nfcid, param->nfcid3, RFAL_NFCDEP_NFCID3_LEN);
 | |
| 
 | |
|     cfg.to = param->to;
 | |
|     
 | |
|     cfg.role     = RFAL_NFCDEP_ROLE_TARGET;
 | |
|     cfg.oper     = param->operParam;
 | |
|     cfg.commMode = param->commMode;
 | |
| 
 | |
|     rfalNfcDepInitialize();
 | |
|     nfcipConfig( &cfg );
 | |
|         
 | |
|     
 | |
|     /*******************************************************************************/
 | |
|     /*  Reply with ATR RES to Initiator                                            */
 | |
|     /*******************************************************************************/
 | |
|     gNfcip.rxBuf        = (uint8_t*)rxParam.rxBuf;
 | |
|     gNfcip.rxBufLen     = sizeof(rfalNfcDepBufFormat);
 | |
|     gNfcip.rxRcvdLen    = rxParam.rxLen;
 | |
|     gNfcip.rxBufPaylPos = RFAL_NFCDEP_DEPREQ_HEADER_LEN;
 | |
|     gNfcip.isChaining   = rxParam.isRxChaining;
 | |
|     gNfcip.txBufPaylPos = RFAL_NFCDEP_DEPREQ_HEADER_LEN;
 | |
|    
 | |
|     EXIT_ON_ERR( ret, nfcipTx( NFCIP_CMD_ATR_RES, (uint8_t*) gNfcip.rxBuf, NULL, 0, 0, NFCIP_NO_FWT ) );
 | |
|     
 | |
|     gNfcip.state = NFCIP_ST_TARG_WAIT_ACTV;
 | |
|     
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepListenGetActivationStatus( void )
 | |
| {
 | |
|     ReturnCode err;
 | |
|     uint8_t    BRS;
 | |
|     
 | |
|     BRS = RFAL_NFCDEP_BRS_MAINTAIN;
 | |
|     
 | |
|     err = nfcipTargetHandleActivation( gNfcip.nfcDepDev, &BRS );
 | |
|     
 | |
|     switch (err)
 | |
|     {
 | |
|         case ERR_NONE:
 | |
|             
 | |
|             if( BRS != RFAL_NFCDEP_BRS_MAINTAIN )
 | |
|             {
 | |
|                 /* DSI codes the bit rate from Initiator to Target */
 | |
|                 /* DRI codes the bit rate from Target to Initiator */
 | |
|                 
 | |
|                 if( gNfcip.cfg.commMode == RFAL_NFCDEP_COMM_ACTIVE )
 | |
|                 {
 | |
|                     EXIT_ON_ERR( err, rfalSetMode( RFAL_MODE_LISTEN_ACTIVE_P2P, gNfcip.nfcDepDev->info.DRI, gNfcip.nfcDepDev->info.DSI ));
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     EXIT_ON_ERR( err, rfalSetMode( ((RFAL_BR_106 == gNfcip.nfcDepDev->info.DRI) ? RFAL_MODE_LISTEN_NFCA : RFAL_MODE_LISTEN_NFCF), gNfcip.nfcDepDev->info.DRI, gNfcip.nfcDepDev->info.DSI ));
 | |
|                 }
 | |
|             }
 | |
|             break;
 | |
|         
 | |
|         case ERR_BUSY:
 | |
|             // do nothing
 | |
|             break;
 | |
|             
 | |
|         case ERR_PROTO:
 | |
|         default:
 | |
|             // re-enable receiving of data
 | |
|             nfcDepReEnableRx( gNfcip.rxBuf, gNfcip.rxBufLen, gNfcip.rxRcvdLen );
 | |
|             break;
 | |
|     }
 | |
|     
 | |
|     return err;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepStartTransceive( const rfalNfcDepTxRxParam *param )
 | |
| {
 | |
|     rfalNfcDepDEPParams nfcDepParams;
 | |
|     
 | |
|     nfcDepParams.txBuf        = (uint8_t *)param->txBuf;
 | |
|     nfcDepParams.txBufLen     = param->txBufLen;
 | |
|     nfcDepParams.txChaining   = param->isTxChaining;
 | |
|     nfcDepParams.txBufPaylPos = RFAL_NFCDEP_DEPREQ_HEADER_LEN;  /* position in txBuf where actual outgoing data is located */
 | |
|     nfcDepParams.did          = RFAL_NFCDEP_DID_KEEP;
 | |
|     nfcDepParams.rxBufPaylPos = RFAL_NFCDEP_DEPREQ_HEADER_LEN;
 | |
|     nfcDepParams.rxBuf        = (uint8_t *)param->rxBuf;
 | |
|     nfcDepParams.rxBufLen     = sizeof(rfalNfcDepBufFormat);
 | |
|     nfcDepParams.fsc          = param->FSx;
 | |
|     nfcDepParams.fwt          = param->FWT;
 | |
|     nfcDepParams.dFwt         = param->dFWT;
 | |
| 
 | |
|     gNfcip.rxRcvdLen          = param->rxLen;
 | |
|     gNfcip.isChaining         = param->isRxChaining;
 | |
| 
 | |
|     nfcipSetDEPParams(&nfcDepParams);
 | |
|     
 | |
|     return ERR_NONE;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepGetTransceiveStatus( void )
 | |
| {
 | |
|     return nfcipRun( gNfcip.rxRcvdLen, gNfcip.isChaining );
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|  /*******************************************************************************/
 | |
|  static void rfalNfcDepPdu2BLockParam( rfalNfcDepPduTxRxParam pduParam, rfalNfcDepTxRxParam *blockParam, uint16_t txPos, uint16_t rxPos )
 | |
| {
 | |
|     uint16_t maxInfLen;
 | |
|     
 | |
|     NO_WARNING(rxPos); /* Keep this param for future use */
 | |
| 
 | |
|     blockParam->DID    = pduParam.DID;
 | |
|     blockParam->FSx    = pduParam.FSx;
 | |
|     blockParam->FWT    = pduParam.FWT;
 | |
|     blockParam->dFWT   = pduParam.dFWT;
 | |
| 
 | |
|     /* Calculate max INF/Payload to be sent to other device */
 | |
|     maxInfLen  = (blockParam->FSx - (RFAL_NFCDEP_HEADER + RFAL_NFCDEP_DEP_PFB_LEN));
 | |
|     maxInfLen += ((blockParam->DID != RFAL_NFCDEP_DID_NO) ? RFAL_NFCDEP_DID_LEN : 0U);
 | |
| 
 | |
| 
 | |
|     if( (pduParam.txBufLen - txPos) > maxInfLen )
 | |
|     {
 | |
|         blockParam->isTxChaining = true;
 | |
|         blockParam->txBufLen     = maxInfLen;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         blockParam->isTxChaining = false;
 | |
|         blockParam->txBufLen     = (pduParam.txBufLen - txPos);
 | |
|     }
 | |
| 
 | |
|     /* TxBuf is moved to the beginning for every Block */
 | |
|     blockParam->txBuf        = (rfalNfcDepBufFormat*)pduParam.txBuf;   /*  PRQA S 0310 # MISRA 11.3 - Intentional safe cast to avoiding large buffer duplication */
 | |
|     blockParam->rxBuf        = pduParam.tmpBuf;                        /* Simply using the pdu buffer is not possible because of current ACK handling */
 | |
|     blockParam->isRxChaining = &gNfcip.isPDURxChaining;
 | |
|     blockParam->rxLen        = pduParam.rxLen;
 | |
| }
 | |
|  
 | |
|  
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepStartPduTransceive( rfalNfcDepPduTxRxParam param )
 | |
| {
 | |
|     rfalNfcDepTxRxParam txRxParam;
 | |
|     
 | |
|     /* Initialize and store APDU context */
 | |
|     gNfcip.PDUParam = param;
 | |
|     gNfcip.PDUTxPos = 0;
 | |
|     gNfcip.PDURxPos = 0;
 | |
|     
 | |
|     /* Convert PDU TxRxParams to Block TxRxParams */
 | |
|     rfalNfcDepPdu2BLockParam( gNfcip.PDUParam, &txRxParam, gNfcip.PDUTxPos, gNfcip.PDURxPos );
 | |
|     
 | |
|     return rfalNfcDepStartTransceive( &txRxParam );
 | |
| }
 | |
|  
 | |
|  
 | |
| /*******************************************************************************/
 | |
| ReturnCode rfalNfcDepGetPduTransceiveStatus( void )
 | |
| {
 | |
|     ReturnCode          ret;
 | |
|     rfalNfcDepTxRxParam txRxParam;
 | |
|     
 | |
|     ret = rfalNfcDepGetTransceiveStatus();
 | |
|     switch( ret )
 | |
|     {
 | |
|         /*******************************************************************************/
 | |
|         case ERR_NONE:
 | |
|             
 | |
|             /* Check if we are still doing chaining on Tx */
 | |
|             if( gNfcip.isTxChaining )
 | |
|             {
 | |
|                 /* Add already Tx bytes */
 | |
|                 gNfcip.PDUTxPos += gNfcip.txBufLen;
 | |
|                 
 | |
|                 /* Convert APDU TxRxParams to I-Block TxRxParams */
 | |
|                 rfalNfcDepPdu2BLockParam( gNfcip.PDUParam, &txRxParam, gNfcip.PDUTxPos, gNfcip.PDURxPos );
 | |
|                 
 | |
|                 if( txRxParam.txBufLen > 0U )      /* MISRA 21.18 */
 | |
|                 {
 | |
|                     /* Move next Block to beginning of APDU Tx buffer */
 | |
|                     ST_MEMCPY( gNfcip.PDUParam.txBuf->pdu, &gNfcip.PDUParam.txBuf->pdu[gNfcip.PDUTxPos], txRxParam.txBufLen );
 | |
|                 }
 | |
|                 
 | |
|                 EXIT_ON_ERR( ret, rfalNfcDepStartTransceive( &txRxParam ) );
 | |
|                 return ERR_BUSY;
 | |
|             }
 | |
|             
 | |
|             /* PDU TxRx is done */
 | |
|             /* fall through */
 | |
|          
 | |
|         /*******************************************************************************/
 | |
|         case ERR_AGAIN:       /*  PRQA S 2003 # MISRA 16.3 - Intentional fall through */
 | |
|         
 | |
|         
 | |
|             /* Check if no PDU transceive has been started before (data from rfalNfcDepListenStartActivation) */
 | |
|             if( gNfcip.PDUParam.rxLen == NULL )
 | |
|             {
 | |
|                 /* In Listen mode first chained packet cannot be retrieved via APDU interface */
 | |
|                 if( ret == ERR_AGAIN  )
 | |
|                 {
 | |
|                     return ERR_NOTSUPP;
 | |
|                 }
 | |
|                 
 | |
|                 /* TxRx is complete and full data is already available */
 | |
|                 return ERR_NONE;
 | |
|             }
 | |
|             
 | |
|             
 | |
|             if( (*gNfcip.PDUParam.rxLen) > 0U )    /* MISRA 21.18 */
 | |
|             {
 | |
|                 /* Ensure that data in tmpBuf still fits into PDU buffer */
 | |
|                 if( (uint16_t)((uint16_t)gNfcip.PDURxPos + (*gNfcip.PDUParam.rxLen)) > RFAL_FEATURE_NFC_DEP_PDU_MAX_LEN )
 | |
|                 {
 | |
|                     return ERR_NOMEM;
 | |
|                 }
 | |
|                 
 | |
|                 /* Copy chained packet from tmp buffer to PDU buffer */
 | |
|                 ST_MEMCPY( &gNfcip.PDUParam.rxBuf->pdu[gNfcip.PDURxPos], gNfcip.PDUParam.tmpBuf->inf, *gNfcip.PDUParam.rxLen );
 | |
|                 gNfcip.PDURxPos += *gNfcip.PDUParam.rxLen;
 | |
|             }
 | |
|             
 | |
|             /* Update output param rxLen */
 | |
|             *gNfcip.PDUParam.rxLen = gNfcip.PDURxPos;
 | |
|             
 | |
|             /* Wait for following Block or PDU TxRx is done */
 | |
|             return ((ret == ERR_AGAIN) ? ERR_BUSY : ERR_NONE);
 | |
|         
 | |
|         /*******************************************************************************/
 | |
|         default:
 | |
|             /* MISRA 16.4: no empty default statement (a comment being enough) */
 | |
|             break;
 | |
|     }
 | |
|     
 | |
|     return ret;
 | |
|  }
 | |
| 
 | |
| 
 | |
| #endif /* RFAL_FEATURE_NFC_DEP */
 | 
