/* * Copyright (C) 2010 NXP Semiconductors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * 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. * See the License for the specific language governing permissions and * limitations under the License. */ /** * \file phFriNfc_LlcpMacNfcip.c * \brief NFC LLCP MAC Mappings For Different RF Technologies. * * Project: NFC-FRI * */ /*include files*/ #include <phFriNfc_LlcpMac.h> #include <phLibNfcStatus.h> #include <phLibNfc.h> #include <phLibNfc_Internal.h> #include <stdio.h> #include <string.h> static NFCSTATUS phFriNfc_LlcpMac_Nfcip_Send(phFriNfc_LlcpMac_t *LlcpMac, phNfc_sData_t *psData, phFriNfc_LlcpMac_Send_CB_t LlcpMacSend_Cb, void *pContext); static void phFriNfc_LlcpMac_Nfcip_TriggerRecvCb(phFriNfc_LlcpMac_t *LlcpMac, NFCSTATUS status) { phFriNfc_LlcpMac_Reveive_CB_t pfReceiveCB; void *pReceiveContext; if (LlcpMac->MacReceive_Cb != NULL) { /* Save callback params */ pfReceiveCB = LlcpMac->MacReceive_Cb; pReceiveContext = LlcpMac->MacReceive_Context; /* Reset the pointer to the Receive Callback and Context*/ LlcpMac->MacReceive_Cb = NULL; LlcpMac->MacReceive_Context = NULL; /* Call the receive callback */ pfReceiveCB(pReceiveContext, status, LlcpMac->psReceiveBuffer); } } static void phFriNfc_LlcpMac_Nfcip_TriggerSendCb(phFriNfc_LlcpMac_t *LlcpMac, NFCSTATUS status) { phFriNfc_LlcpMac_Send_CB_t pfSendCB; void *pSendContext; if (LlcpMac->MacSend_Cb != NULL) { /* Save context in local variables */ pfSendCB = LlcpMac->MacSend_Cb; pSendContext = LlcpMac->MacSend_Context; /* Reset the pointer to the Send Callback */ LlcpMac->MacSend_Cb = NULL; LlcpMac->MacSend_Context = NULL; /* Call Send callback */ pfSendCB(pSendContext, status); } } static NFCSTATUS phFriNfc_LlcpMac_Nfcip_Chk(phFriNfc_LlcpMac_t *LlcpMac, phFriNfc_LlcpMac_Chk_CB_t ChkLlcpMac_Cb, void *pContext) { NFCSTATUS status = NFCSTATUS_SUCCESS; uint8_t Llcp_Magic_Number[] = {0x46,0x66,0x6D}; if(NULL == LlcpMac || NULL == ChkLlcpMac_Cb || NULL == pContext) { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_PARAMETER); } else { status = (NFCSTATUS)memcmp(Llcp_Magic_Number,LlcpMac->psRemoteDevInfo->RemoteDevInfo.NfcIP_Info.ATRInfo,3); if(!status) { LlcpMac->sConfigParam.buffer = &LlcpMac->psRemoteDevInfo->RemoteDevInfo.NfcIP_Info.ATRInfo[3] ; LlcpMac->sConfigParam.length = (LlcpMac->psRemoteDevInfo->RemoteDevInfo.NfcIP_Info.ATRInfo_Length - 3); status = NFCSTATUS_SUCCESS; } else { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_FAILED); } ChkLlcpMac_Cb(pContext,status); } return status; } static NFCSTATUS phFriNfc_LlcpMac_Nfcip_Activate (phFriNfc_LlcpMac_t *LlcpMac) { NFCSTATUS status = NFCSTATUS_SUCCESS; if(LlcpMac == NULL) { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_PARAMETER); } else { LlcpMac->LinkState = phFriNfc_LlcpMac_eLinkActivated; LlcpMac->LinkStatus_Cb(LlcpMac->LinkStatus_Context, LlcpMac->LinkState, &LlcpMac->sConfigParam, LlcpMac->PeerRemoteDevType); } return status; } static NFCSTATUS phFriNfc_LlcpMac_Nfcip_Deactivate (phFriNfc_LlcpMac_t *LlcpMac) { NFCSTATUS status = NFCSTATUS_SUCCESS; if(NULL == LlcpMac) { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_PARAMETER); } else { /* Set the flag of LinkStatus to deactivate */ LlcpMac->LinkState = phFriNfc_LlcpMac_eLinkDeactivated; if (LlcpMac->SendPending) { /* Reset Flag */ LlcpMac->SendPending = FALSE; phFriNfc_LlcpMac_Nfcip_TriggerSendCb(LlcpMac, NFCSTATUS_FAILED); } if (LlcpMac->RecvPending) { /* Reset Flag */ LlcpMac->RecvPending = FALSE; phFriNfc_LlcpMac_Nfcip_TriggerRecvCb(LlcpMac, NFCSTATUS_FAILED); } LlcpMac->LinkStatus_Cb(LlcpMac->LinkStatus_Context, LlcpMac->LinkState, NULL, LlcpMac->PeerRemoteDevType); } return status; } static void phFriNfc_LlcpMac_Nfcip_Send_Cb(void *pContext, NFCSTATUS Status) { phFriNfc_LlcpMac_t *LlcpMac = (phFriNfc_LlcpMac_t *)pContext; #ifdef LLCP_CHANGES if(gpphLibContext->LibNfcState.next_state == eLibNfcHalStateShutdown) { phLibNfc_Pending_Shutdown(); Status = NFCSTATUS_SHUTDOWN; } #endif /* #ifdef LLCP_CHANGES */ /* Reset Send and Receive Flag */ LlcpMac->SendPending = FALSE; LlcpMac->RecvPending = FALSE; phFriNfc_LlcpMac_Nfcip_TriggerSendCb(LlcpMac, Status); } static void phFriNfc_LlcpMac_Nfcip_Receive_Cb(void *pContext, NFCSTATUS Status) { phFriNfc_LlcpMac_t *LlcpMac = (phFriNfc_LlcpMac_t *)pContext; #ifdef LLCP_CHANGES phFriNfc_LlcpMac_Send_CB_t pfSendCB; void *pSendContext; if(gpphLibContext->LibNfcState.next_state == eLibNfcHalStateShutdown) { phLibNfc_Pending_Shutdown(); Status = NFCSTATUS_SHUTDOWN; } if (NFCSTATUS_SHUTDOWN == Status) { /* Save context in local variables */ pfSendCB = LlcpMac->MacSend_Cb; pSendContext = LlcpMac->MacSend_Context; /* Reset the pointer to the Send Callback */ LlcpMac->MacSend_Cb = NULL; LlcpMac->MacSend_Context = NULL; /* Reset Send and Receive Flag */ LlcpMac->SendPending = FALSE; LlcpMac->RecvPending = FALSE; } #endif /* #ifdef LLCP_CHANGES */ phFriNfc_LlcpMac_Nfcip_TriggerRecvCb(LlcpMac, Status); #ifdef LLCP_CHANGES if (NFCSTATUS_SHUTDOWN == Status) { if ((LlcpMac->SendPending) && (NULL != pfSendCB)) { pfSendCB(pSendContext, Status); } } else #endif /* #ifdef LLCP_CHANGES */ { /* Test if a send is pending */ if(LlcpMac->SendPending) { Status = phFriNfc_LlcpMac_Nfcip_Send(LlcpMac,LlcpMac->psSendBuffer,LlcpMac->MacSend_Cb,LlcpMac->MacSend_Context); } } } static void phFriNfc_LlcpMac_Nfcip_Transceive_Cb(void *pContext, NFCSTATUS Status) { phFriNfc_LlcpMac_t *LlcpMac = (phFriNfc_LlcpMac_t *)pContext; #ifdef LLCP_CHANGES if(gpphLibContext->LibNfcState.next_state == eLibNfcHalStateShutdown) { phLibNfc_Pending_Shutdown(); Status = NFCSTATUS_SHUTDOWN; } #endif /* #ifdef LLCP_CHANGES */ /* Reset Send and Receive Flag */ LlcpMac->SendPending = FALSE; LlcpMac->RecvPending = FALSE; /* Call the callbacks */ phFriNfc_LlcpMac_Nfcip_TriggerSendCb(LlcpMac, Status); phFriNfc_LlcpMac_Nfcip_TriggerRecvCb(LlcpMac, Status); } static NFCSTATUS phFriNfc_LlcpMac_Nfcip_Send(phFriNfc_LlcpMac_t *LlcpMac, phNfc_sData_t *psData, phFriNfc_LlcpMac_Send_CB_t LlcpMacSend_Cb, void *pContext) { NFCSTATUS status = NFCSTATUS_SUCCESS; if(NULL == LlcpMac || NULL == psData || NULL == LlcpMacSend_Cb || NULL == pContext) { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_PARAMETER); } else if(LlcpMac->MacSend_Cb != NULL && LlcpMac->PeerRemoteDevType == phFriNfc_LlcpMac_ePeerTypeInitiator) { /*Previous callback is pending */ status = NFCSTATUS_REJECTED; } else { /* Save the LlcpMacSend_Cb */ LlcpMac->MacSend_Cb = LlcpMacSend_Cb; LlcpMac->MacSend_Context = pContext; switch(LlcpMac->PeerRemoteDevType) { case phFriNfc_LlcpMac_ePeerTypeInitiator: { if(LlcpMac->RecvPending) { /*set the completion routines for the LLCP Transceive function*/ LlcpMac->MacCompletionInfo.CompletionRoutine = phFriNfc_LlcpMac_Nfcip_Transceive_Cb; LlcpMac->MacCompletionInfo.Context = LlcpMac; /* set the command type*/ LlcpMac->Cmd.NfcIP1Cmd = phHal_eNfcIP1_Raw; /* set the Additional Info*/ LlcpMac->psDepAdditionalInfo.DepFlags.MetaChaining = 0; LlcpMac->psDepAdditionalInfo.DepFlags.NADPresent = 0; LlcpMac->SendPending = TRUE; status = phFriNfc_OvrHal_Transceive(LlcpMac->LowerDevice, &LlcpMac->MacCompletionInfo, LlcpMac->psRemoteDevInfo, LlcpMac->Cmd, &LlcpMac->psDepAdditionalInfo, psData->buffer, (uint16_t)psData->length, LlcpMac->psReceiveBuffer->buffer, (uint16_t*)&LlcpMac->psReceiveBuffer->length); } else { LlcpMac->SendPending = TRUE; LlcpMac->psSendBuffer = psData; return status = NFCSTATUS_PENDING; } }break; case phFriNfc_LlcpMac_ePeerTypeTarget: { if(!LlcpMac->RecvPending) { LlcpMac->SendPending = TRUE; LlcpMac->psSendBuffer = psData; return status = NFCSTATUS_PENDING; } else { /*set the completion routines for the LLCP Send function*/ LlcpMac->MacCompletionInfo.CompletionRoutine = phFriNfc_LlcpMac_Nfcip_Send_Cb; LlcpMac->MacCompletionInfo.Context = LlcpMac; status = phFriNfc_OvrHal_Send(LlcpMac->LowerDevice, &LlcpMac->MacCompletionInfo, LlcpMac->psRemoteDevInfo, psData->buffer, (uint16_t)psData->length); } }break; default: { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_DEVICE); }break; } } return status; } static NFCSTATUS phFriNfc_LlcpMac_Nfcip_Receive(phFriNfc_LlcpMac_t *LlcpMac, phNfc_sData_t *psData, phFriNfc_LlcpMac_Reveive_CB_t LlcpMacReceive_Cb, void *pContext) { NFCSTATUS status = NFCSTATUS_SUCCESS; if(NULL == LlcpMac || NULL==psData || NULL == LlcpMacReceive_Cb || NULL == pContext) { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_PARAMETER); } else if(LlcpMac->MacReceive_Cb != NULL) { /*Previous callback is pending */ status = NFCSTATUS_REJECTED; } else { /* Save the LlcpMacReceive_Cb */ LlcpMac->MacReceive_Cb = LlcpMacReceive_Cb; LlcpMac->MacReceive_Context = pContext; /* Save the pointer to the receive buffer */ LlcpMac->psReceiveBuffer= psData; switch(LlcpMac->PeerRemoteDevType) { case phFriNfc_LlcpMac_ePeerTypeInitiator: { if(LlcpMac->SendPending) { /*set the completion routines for the LLCP Transceive function*/ LlcpMac->MacCompletionInfo.CompletionRoutine = phFriNfc_LlcpMac_Nfcip_Transceive_Cb; LlcpMac->MacCompletionInfo.Context = LlcpMac; /* set the command type*/ LlcpMac->Cmd.NfcIP1Cmd = phHal_eNfcIP1_Raw; /* set the Additional Info*/ LlcpMac->psDepAdditionalInfo.DepFlags.MetaChaining = 0; LlcpMac->psDepAdditionalInfo.DepFlags.NADPresent = 0; LlcpMac->RecvPending = TRUE; status = phFriNfc_OvrHal_Transceive(LlcpMac->LowerDevice, &LlcpMac->MacCompletionInfo, LlcpMac->psRemoteDevInfo, LlcpMac->Cmd, &LlcpMac->psDepAdditionalInfo, LlcpMac->psSendBuffer->buffer, (uint16_t)LlcpMac->psSendBuffer->length, psData->buffer, (uint16_t*)&psData->length); } else { LlcpMac->RecvPending = TRUE; return status = NFCSTATUS_PENDING; } }break; case phFriNfc_LlcpMac_ePeerTypeTarget: { /*set the completion routines for the LLCP Receive function*/ LlcpMac->MacCompletionInfo.CompletionRoutine = phFriNfc_LlcpMac_Nfcip_Receive_Cb; /* save the context of LlcpMacNfcip */ LlcpMac->MacCompletionInfo.Context = LlcpMac; LlcpMac->RecvPending = TRUE; status = phFriNfc_OvrHal_Receive(LlcpMac->LowerDevice, &LlcpMac->MacCompletionInfo, LlcpMac->psRemoteDevInfo, LlcpMac->psReceiveBuffer->buffer, (uint16_t*)&LlcpMac->psReceiveBuffer->length); }break; default: { status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_INVALID_DEVICE); }break; } } return status; } NFCSTATUS phFriNfc_LlcpMac_Nfcip_Register (phFriNfc_LlcpMac_t *LlcpMac) { NFCSTATUS status = NFCSTATUS_SUCCESS; if(NULL != LlcpMac) { LlcpMac->LlcpMacInterface.chk = phFriNfc_LlcpMac_Nfcip_Chk; LlcpMac->LlcpMacInterface.activate = phFriNfc_LlcpMac_Nfcip_Activate; LlcpMac->LlcpMacInterface.deactivate = phFriNfc_LlcpMac_Nfcip_Deactivate; LlcpMac->LlcpMacInterface.send = phFriNfc_LlcpMac_Nfcip_Send; LlcpMac->LlcpMacInterface.receive = phFriNfc_LlcpMac_Nfcip_Receive; return NFCSTATUS_SUCCESS; } else { return status = PHNFCSTVAL(CID_FRI_NFC_LLCP_MAC, NFCSTATUS_FAILED); } }