/*
 * 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_MifULFormat.c
 * \brief NFC Ndef Formatting For Mifare ultralight card.
 *
 * Project: NFC-FRI
 *
 * $Date: Mon Dec 13 14:14:12 2010 $
 * $Author: ing02260 $
 * $Revision: 1.9 $
 * $Aliases:  $
 *
 */

#include <phFriNfc_MifULFormat.h>
#include <phFriNfc_OvrHal.h>

/*! \ingroup grp_file_attributes
 *  \name NDEF Mapping
 *
 * File: \ref phFriNfc_MifULFormat.c
 *
 */
/*@{*/
#define PHFRINFCMIFULFORMAT_FILEREVISION "$Revision: 1.9 $"
#define PHFRINFCMIFULFORMAT_FILEALIASES  "$Aliases:  $"
/*@}*/

#ifdef FRINFC_READONLY_NDEF
    /* Mifare UL OTP block number is 3 */
    #define RD_LOCK_OTP_BLOCK_NUMBER            0x02U
    #define OTP_BLOCK_NUMBER                    0x03U
    /* READ ONLY value that shall be written in the OTP to make the card read only */
    #define READ_ONLY_VALUE_IN_OTP              0x0FU
    /* Mifare UL OTP block number is 3 */
    #define MIFARE_UL_READ_MAX_SIZE             16U
    /* 1st Lock byte value */
    #define MIFARE_UL_LOCK_BYTE1_VALUE          0xF8U
    /* 2nd Lock byte value */
    #define MIFARE_UL_LOCK_BYTE2_VALUE          0xFFU
    /* Mifare ULC dynamic lock byte address */
    #define MIFARE_ULC_DYNAMIC_LOCK_BYTES_ADDR  0x28U
    /* Type 2 STATIC CARD memory value in the OTP */
    #define TYPE_2_STATIC_MEM_SIZE_VALUE        0x06U
    /* Type 2 DYNAMIC CARD memory value in the OTP */
    #define TYPE_2_DYNAMIC_MEM_SIZE_VALUE       0x12U
    /* Lock byte 3 value to be ORed with the existing value */
    #define MIFARE_UL_LOCK_BYTE3_VALUE          0xEEU
    /* Posiiton of the memory information in the stored OTP bytes */
    #define TYPE_2_MEM_SIZE_POSITION            0x02U
    /* 3rd Lock byte position after reading the block number 0x28 */
    #define TYPE_2_LOCK_BYTE3_POS_RD_BLK28      0x00U

#ifdef PH_NDEF_MIFARE_ULC

    /* Lock control TLVs, TYPE identifier */
    #define LOCK_CTRL_TYPE_IN_TLV               0x01U
    /* Lock control TLVs, Length expected */
    #define LOCK_CTRL_LEN_IN_TLV                0x03U

    /* NDEF message TLVs, TYPE identifier */
    #define NDEF_TYPE_IN_TLV                    0x03U

    #define MFUL_NULL_TLV                       0x00U
    #define THREE_BYTE_LENGTH_FIELD             0xFFU
    #define TERMINATOR_TLV                      0xFEU
    #define MIFARE_ULC_SIZE                     0xC0U
    #define MFUL_NIBBLE_SIZE                    0x04U
    #define MFUL_NIBBLE_MASK                    0x0FU
    #define MFUL_BYTE_SIZE_IN_BITS              0x08U
    #define MFUL_BLOCK_SIZE_IN_BYTES            0x04U
    /* Initial (0 to 3 blocks) 4 blocks are ignored, i.e., 16 bytes */
    #define MFUL_INITIAL_BYTES_IGNORED          0x10U

    #define MFUL_CONVERT_BITS_TO_BYTES(bits_to_bytes) \
            (((bits_to_bytes % MFUL_BYTE_SIZE_IN_BITS) > 0) ? \
            ((bits_to_bytes / MFUL_BYTE_SIZE_IN_BITS) + 1) : \
            (bits_to_bytes / MFUL_BYTE_SIZE_IN_BITS))

    typedef enum phFriNfc_MfUL_Parse
    {
        LOCK_TLV_T, 
        LOCK_TLV_L,
        LOCK_TLV_V, 
        NDEF_TLV_T, 
        NDEF_TLV_L,
        NDEF_TLV_V
    }phFriNfc_MfUL_Parse_t;

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

#endif /* #ifdef FRINFC_READONLY_NDEF */
/*!
* \brief \copydoc page_ovr Helper function for Mifare UL. This function calls the   
* transceive function
*/
static NFCSTATUS phFriNfc_MfUL_H_Transceive(phFriNfc_sNdefSmtCrdFmt_t    *NdefSmtCrdFmt);

/*!
* \brief \copydoc page_ovr Helper function for Mifare UL. This function calls the   
* read or write operation
*/
static NFCSTATUS phFriNfc_MfUL_H_WrRd(phFriNfc_sNdefSmtCrdFmt_t *NdefSmtCrdFmt);

/*!
* \brief \copydoc page_ovr Helper function for Mifare UL. This function fills the  
* send buffer for transceive function
*/
static void phFriNfc_MfUL_H_fillSendBuf(phFriNfc_sNdefSmtCrdFmt_t   *NdefSmtCrdFmt,
                                        uint8_t                     BlockNo);

/*!
* \brief \copydoc page_ovr Helper function for Mifare UL. This function shall process
* the read bytes
*/
static NFCSTATUS phFriNfc_MfUL_H_ProRd16Bytes(phFriNfc_sNdefSmtCrdFmt_t *NdefSmtCrdFmt);

/*!
* \brief \copydoc page_ovr Helper function for Mifare UL. This function shall process the
* OTP bytes written
*/
static NFCSTATUS phFriNfc_MfUL_H_ProWrOTPBytes(phFriNfc_sNdefSmtCrdFmt_t    *NdefSmtCrdFmt);

#ifdef FRINFC_READONLY_NDEF

#ifdef PH_NDEF_MIFARE_ULC

static 
NFCSTATUS 
phFriNfc_MfUL_ParseTLVs (
    phFriNfc_sNdefSmtCrdFmt_t       *NdefSmtCrdFmt, 
    uint8_t                         *data_to_parse, 
    uint8_t                         size_to_parse);

static 
NFCSTATUS 
phFriNfc_MfUL_GetLockBytesInfo (
    phFriNfc_sNdefSmtCrdFmt_t       *NdefSmtCrdFmt);

static 
NFCSTATUS 
phFriNfc_MfUL_GetDefaultLockBytesInfo (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt);

static 
uint8_t 
phFriNfc_MfUL_GetSkipSize (
    phFriNfc_sNdefSmtCrdFmt_t       *NdefSmtCrdFmt, 
    uint8_t                         block_number, 
    uint8_t                         byte_number);

static 
NFCSTATUS 
phFriNfc_MfUL_ReadWriteLockBytes (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt);

static 
NFCSTATUS
phFriNfc_MfUL_UpdateAndWriteLockBits (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt);

static 
uint8_t
phFriNfc_MfUL_CalcRemainingLockBits (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt);

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

#endif /* #ifdef FRINFC_READONLY_NDEF */

static int MemCompare1 ( void *s1, void *s2, unsigned int n );
/*The function does a comparision of two strings and returns a non zero value 
if two strings are unequal*/
static int MemCompare1 ( void *s1, void *s2, unsigned int n )
{
    int8_t   diff = 0;
    int8_t *char_1  =(int8_t *)s1;
    int8_t *char_2  =(int8_t *)s2;
    if(NULL == s1 || NULL == s2)
    {
        PHDBG_CRITICAL_ERROR("NULL pointer passed to memcompare");
    }
    else
    {
        for(;((n>0)&&(diff==0));n--,char_1++,char_2++)
        {
            diff = *char_1 - *char_2;
        }
    }
    return (int)diff;
}

void phFriNfc_MfUL_Reset(phFriNfc_sNdefSmtCrdFmt_t    *NdefSmtCrdFmt)
{
    uint8_t OTPByte[] = PH_FRINFC_MFUL_FMT_OTP_BYTES; 

    NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = PH_FRINFC_MFUL_FMT_VAL_0;
    (void)memcpy(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                OTPByte,
                sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes));
#ifdef FRINFC_READONLY_NDEF
    NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[0] = 0;
    NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[1] = 0;
    NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[2] = 0;
    NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[3] = 0;
#endif /* #ifdef FRINFC_READONLY_NDEF */
}

NFCSTATUS phFriNfc_MfUL_Format(phFriNfc_sNdefSmtCrdFmt_t    *NdefSmtCrdFmt)
{
    NFCSTATUS               Result = NFCSTATUS_SUCCESS;
    uint8_t                 OTPByte[] = PH_FRINFC_MFUL_FMT_OTP_BYTES;
    
    NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = PH_FRINFC_MFUL_FMT_VAL_0;
    (void)memcpy(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                OTPByte,
                sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes));

    /* Set the state */
    NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RD_16BYTES;
    /* Initialise current block to the lock bits block */
    NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = PH_FRINFC_MFUL_FMT_VAL_2;
    
    /* Start authentication */
    Result = phFriNfc_MfUL_H_WrRd(NdefSmtCrdFmt);
    return Result;
}

#ifdef FRINFC_READONLY_NDEF

NFCSTATUS
phFriNfc_MfUL_ConvertToReadOnly (
    phFriNfc_sNdefSmtCrdFmt_t    *NdefSmtCrdFmt)
{
    NFCSTATUS               result = NFCSTATUS_SUCCESS;

    NdefSmtCrdFmt->AddInfo.Type2Info.DefaultLockBytesFlag = TRUE;
    NdefSmtCrdFmt->AddInfo.Type2Info.ReadDataIndex = 0;

    NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_RD_16BYTES;

    result = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);

    return result;
}

#endif /* #ifdef FRINFC_READONLY_NDEF */

void phFriNfc_MfUL_Process(void             *Context,
                           NFCSTATUS        Status)
{
    phFriNfc_sNdefSmtCrdFmt_t  *NdefSmtCrdFmt = (phFriNfc_sNdefSmtCrdFmt_t *)Context;
    
    if(Status == NFCSTATUS_SUCCESS)
    {
        switch(NdefSmtCrdFmt->State)
        {
        case PH_FRINFC_MFUL_FMT_RD_16BYTES:
            Status = phFriNfc_MfUL_H_ProRd16Bytes(NdefSmtCrdFmt);
            break;

        case PH_FRINFC_MFUL_FMT_WR_OTPBYTES:
            Status = phFriNfc_MfUL_H_ProWrOTPBytes(NdefSmtCrdFmt);
            break;

        case PH_FRINFC_MFUL_FMT_WR_TLV:
#ifdef PH_NDEF_MIFARE_ULC               
            if (NdefSmtCrdFmt->CardType == PH_FRINFC_NDEFMAP_MIFARE_ULC_CARD)
            {
                /* Write NDEF TLV in block number 5 */
                NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 
                                                        PH_FRINFC_MFUL_FMT_VAL_5;
                /* Card already have the OTP bytes so write TLV */
                NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_WR_TLV1;
                
                Status = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
            }
#endif /* #ifdef PH_NDEF_MIFARE_ULC */

            break;

#ifdef FRINFC_READONLY_NDEF

        case PH_FRINFC_MFUL_FMT_RO_RD_16BYTES:
        {
            if (MIFARE_UL_READ_MAX_SIZE == *NdefSmtCrdFmt->SendRecvLength)
            {
                uint8_t         otp_lock_page_size = 0;
                uint8_t         i = 0;

                otp_lock_page_size = sizeof (NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes);
                (void)memcpy ((void *)NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes,
                            (void *)NdefSmtCrdFmt->SendRecvBuf,
                            sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes));

                NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[2] = (uint8_t)
                                    (NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[2] 
                                    | MIFARE_UL_LOCK_BYTE1_VALUE);
                NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes[3] = MIFARE_UL_LOCK_BYTE2_VALUE;
                i = (uint8_t)(i + otp_lock_page_size);

                otp_lock_page_size = sizeof (NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes);                                
                
                (void)memcpy ((void *)NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                            (void *)(NdefSmtCrdFmt->SendRecvBuf + i),
                            sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes));

                NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes[(otp_lock_page_size - 1)] =
                                                        READ_ONLY_VALUE_IN_OTP;

                switch (NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes[TYPE_2_MEM_SIZE_POSITION])
                {
                    case TYPE_2_STATIC_MEM_SIZE_VALUE:
                    {
                        NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_WR_OTP_BYTES;
                        Status = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
                        break;
                    }

#ifdef PH_NDEF_MIFARE_ULC
                    case TYPE_2_DYNAMIC_MEM_SIZE_VALUE:
                    {
                        NdefSmtCrdFmt->State = 
                                PH_FRINFC_MFUL_FMT_RO_NDEF_PARSE_RD_BYTES;

                        /* Start reading from block 4 */
                        NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 4;
                        Status = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
                        break;
                    }
#endif /* #ifdef PH_NDEF_MIFARE_ULC */

                    default:
                    {
                        Status = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                            NFCSTATUS_INVALID_DEVICE_REQUEST);
                        break;
                    }
                }
            }
            else
            {
                Status = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                    NFCSTATUS_INVALID_RECEIVE_LENGTH);
            }
            break;
        }

        case PH_FRINFC_MFUL_FMT_RO_WR_OTP_BYTES:
        {
            switch (NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes[TYPE_2_MEM_SIZE_POSITION])
            {
                case TYPE_2_STATIC_MEM_SIZE_VALUE:
#ifdef PH_NDEF_MIFARE_ULC
                case TYPE_2_DYNAMIC_MEM_SIZE_VALUE:
#endif /* #ifdef PH_NDEF_MIFARE_ULC */
                {
                    NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_WR_LOCK_BYTES;
                    Status = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
                    break;
                }

                default:
                {
                    Status = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                        NFCSTATUS_INVALID_DEVICE_REQUEST);
                    break;
                }
            }
            break;
        }

#ifdef PH_NDEF_MIFARE_ULC

        case PH_FRINFC_MFUL_FMT_RO_NDEF_PARSE_RD_BYTES:
        {
            if (MIFARE_UL_READ_MAX_SIZE == *NdefSmtCrdFmt->SendRecvLength)
            {
                Status = phFriNfc_MfUL_ParseTLVs (NdefSmtCrdFmt, 
                                        NdefSmtCrdFmt->SendRecvBuf, 
                                        (uint8_t)*NdefSmtCrdFmt->SendRecvLength);

                if (!Status)
                {
                    NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 
                        NdefSmtCrdFmt->AddInfo.Type2Info.LockBlockNumber;
                    Status = phFriNfc_MfUL_ReadWriteLockBytes (NdefSmtCrdFmt);
                }
            }
            else
            {
                Status = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                    NFCSTATUS_INVALID_RECEIVE_LENGTH);
            }
            break;
        }

        case PH_FRINFC_MFUL_FMT_RO_RD_DYN_LOCK_BYTES:
        {
            if (MIFARE_UL_READ_MAX_SIZE == *NdefSmtCrdFmt->SendRecvLength)
            {
                (void)memcpy ((void *)NdefSmtCrdFmt->AddInfo.Type2Info.ReadData,
                            (void *)NdefSmtCrdFmt->SendRecvBuf,
                            sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.ReadData));

                NdefSmtCrdFmt->AddInfo.Type2Info.ReadDataIndex = 0;

                Status = phFriNfc_MfUL_UpdateAndWriteLockBits (NdefSmtCrdFmt);
                
            }
            else
            {
                Status = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                    NFCSTATUS_INVALID_RECEIVE_LENGTH);
            } 
            break;
        }

        case PH_FRINFC_MFUL_FMT_RO_WR_DYN_LOCK_BYTES:
        {
            NdefSmtCrdFmt->AddInfo.Type2Info.ReadDataIndex = (uint8_t)
                                    (NdefSmtCrdFmt->AddInfo.Type2Info.ReadDataIndex + 
                                    MFUL_BLOCK_SIZE_IN_BYTES);

            if (!phFriNfc_MfUL_CalcRemainingLockBits (NdefSmtCrdFmt))
            {
                /* There is no lock bits to write, then write OTP bytes */
                NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_WR_OTP_BYTES;
                Status = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
            }
            else if ((NdefSmtCrdFmt->AddInfo.Type2Info.ReadDataIndex < 
                MIFARE_UL_READ_MAX_SIZE) 
                && (phFriNfc_MfUL_CalcRemainingLockBits (NdefSmtCrdFmt)))
            {
                /* If remaining lock bits has to be written and the data is already read */
                Status = phFriNfc_MfUL_UpdateAndWriteLockBits (NdefSmtCrdFmt);
            }
            else
            {
                /* Increment current block by 4 because if a data is read then 16 
                    bytes will be given which is 4 blocks */
                NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = (uint8_t)
                            (NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock + 4);
                Status = phFriNfc_MfUL_ReadWriteLockBytes (NdefSmtCrdFmt);
            }
            break;
        }

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

        case PH_FRINFC_MFUL_FMT_RO_WR_LOCK_BYTES:
        {
            /* Do nothing */
            break;
        }

#endif /* #ifdef FRINFC_READONLY_NDEF */

#ifdef PH_NDEF_MIFARE_ULC   
        case PH_FRINFC_MFUL_FMT_WR_TLV1:
        
        break;
#endif /* #ifdef PH_NDEF_MIFARE_ULC */
        
        default:
            Status = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                NFCSTATUS_INVALID_DEVICE_REQUEST);
            break;
        }
    }
    /* Status is not success then call completion routine */
    if(Status != NFCSTATUS_PENDING)
    {
        phFriNfc_SmtCrdFmt_HCrHandler(NdefSmtCrdFmt, Status);
    }
}

#ifdef FRINFC_READONLY_NDEF

#ifdef PH_NDEF_MIFARE_ULC

static 
uint8_t 
phFriNfc_MfUL_GetSkipSize (
    phFriNfc_sNdefSmtCrdFmt_t       *NdefSmtCrdFmt, 
    uint8_t                         block_number, 
    uint8_t                         byte_number)
{
    uint8_t                     skip_size = 0;
    phFriNfc_Type2_AddInfo_t    *ps_type2_info = 
                                &NdefSmtCrdFmt->AddInfo.Type2Info;

    /* This check is added, because the default lock bits is always 
        present after the DATA AREA. 
        So, default lock bytes doesnt have any skip size */
    if (!ps_type2_info->DefaultLockBytesFlag)
    {
        /* Only check for the lock control TLV */
        if ((block_number == ps_type2_info->LockBlockNumber) 
            && (byte_number == ps_type2_info->LockByteNumber))
        {
            skip_size = MFUL_CONVERT_BITS_TO_BYTES(ps_type2_info->NoOfLockBits);
        }
    }
    
    return skip_size;
}

static 
NFCSTATUS 
phFriNfc_MfUL_GetLockBytesInfo (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt)
{
    NFCSTATUS                       result = NFCSTATUS_SUCCESS;
    phFriNfc_Type2_AddInfo_t        *ps_type2_info = 
                                    &(NdefSmtCrdFmt->AddInfo.Type2Info);
    uint8_t                         page_address = 0;
    uint8_t                         bytes_offset = 0;
    uint8_t                         lock_index = 0;


    page_address = (uint8_t)(ps_type2_info->DynLockBytes[lock_index] >> MFUL_NIBBLE_SIZE);
    bytes_offset = (uint8_t)(ps_type2_info->DynLockBytes[lock_index] & MFUL_NIBBLE_MASK);

    lock_index = (lock_index + 1);
    ps_type2_info->NoOfLockBits = ps_type2_info->DynLockBytes[lock_index];

    lock_index = (lock_index + 1);
    ps_type2_info->LockBytesPerPage = 
                            (ps_type2_info->DynLockBytes[lock_index] & MFUL_NIBBLE_MASK);
    ps_type2_info->BytesLockedPerLockBit = 
                            (ps_type2_info->DynLockBytes[lock_index] >> MFUL_NIBBLE_SIZE);

    /* Apply the formula to calculate byte address 
        ByteAddr = ((PageAddr * (2 ^ BytesPerPage)) + ByteOffset)
    */
    ps_type2_info->LockByteNumber = (uint8_t)((page_address 
                                * (1 << ps_type2_info->LockBytesPerPage))
                                + bytes_offset);

    ps_type2_info->LockBlockNumber = (uint8_t)(ps_type2_info->LockByteNumber / 
                                                MFUL_BLOCK_SIZE_IN_BYTES);
    ps_type2_info->LockByteNumber = (uint8_t)(ps_type2_info->LockByteNumber % 
                                                MFUL_BLOCK_SIZE_IN_BYTES);

#if 0
    if (
        /* Out of bound memory check */    
        ((ps_locktlv_info->ByteAddr + ps_locktlv_info->Size) > 
        (uint16_t)(psNdefMap->TopazContainer.CCByteBuf[2] * 
        TOPAZ_BYTES_PER_BLOCK)) || 

        /* Check the static lock and reserved areas memory blocks */
        ((ps_locktlv_info->ByteAddr >= TOPAZ_STATIC_LOCK_RES_START) && 
        (ps_locktlv_info->ByteAddr < TOPAZ_STATIC_LOCK_RES_END)) || 
        (((ps_locktlv_info->ByteAddr + ps_locktlv_info->Size - 1) >= 
        TOPAZ_STATIC_LOCK_RES_START) && 
        ((ps_locktlv_info->ByteAddr + ps_locktlv_info->Size - 1) < 
        TOPAZ_STATIC_LOCK_RES_END))
        )
    {
        ps_locktlv_info->ByteAddr = 0;
        result = PHNFCSTVAL(CID_FRI_NFC_NDEF_MAP,
                            NFCSTATUS_NO_NDEF_SUPPORT);
    }
    else
    {
        ps_locktlv_info->BlkNum = (ps_locktlv_info->ByteAddr / 
                                    TOPAZ_BYTES_PER_BLOCK);
        ps_locktlv_info->ByteNum = (ps_locktlv_info->ByteAddr % 
                                    TOPAZ_BYTES_PER_BLOCK);
    }
#endif /* #if 0 */

    return result;
}

static 
uint8_t
phFriNfc_MfUL_CalcRemainingLockBits (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt)
{
    uint8_t                         lock_bits_remaining = 0;
    phFriNfc_Type2_AddInfo_t        *ps_type2_info = 
                                    &(NdefSmtCrdFmt->AddInfo.Type2Info);

    lock_bits_remaining = (uint8_t)(ps_type2_info->NoOfLockBits - 
                                    ps_type2_info->LockBitsWritten);

    return lock_bits_remaining;
}

static 
NFCSTATUS
phFriNfc_MfUL_UpdateAndWriteLockBits (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt)
{
    NFCSTATUS                       result = NFCSTATUS_SUCCESS;
    phFriNfc_Type2_AddInfo_t        *ps_type2_info = 
                                    &(NdefSmtCrdFmt->AddInfo.Type2Info);
    uint8_t                         byte_index = 0;
    uint8_t                         no_of_bits_left_in_block = 0;
    uint8_t                         remaining_lock_bits = 0;
    uint8_t                         remaining_lock_bytes = 0;
    /* Array of 3 is used because the lock bits with 4 bytes in a block
        is handled in the function phFriNfc_MfUL_ReadWriteLockBytes 
        So use this function only if lock bytes doesnt use the entire block */
    uint8_t                         lock_bytes_value[MFUL_BLOCK_SIZE_IN_BYTES] = {0};
    uint8_t                         lock_byte_index = 0;

    (void)memcpy ((void *)lock_bytes_value, 
                (void*)(ps_type2_info->ReadData + ps_type2_info->ReadDataIndex), 
                sizeof (ps_type2_info->DynLockBytes));
    remaining_lock_bits = phFriNfc_MfUL_CalcRemainingLockBits (NdefSmtCrdFmt);
    
    if (ps_type2_info->CurrentBlock == ps_type2_info->LockBlockNumber)
    {
        /* 1st write to lock bits, so byte_index is updated */
        byte_index = ps_type2_info->LockByteNumber;
    }
    
    no_of_bits_left_in_block = (uint8_t)((MFUL_BLOCK_SIZE_IN_BYTES - byte_index) * 
                                MFUL_BYTE_SIZE_IN_BITS);

    if (no_of_bits_left_in_block >= remaining_lock_bits)
    {
        /* Entire lock bits can be written 
            if block size is more than number of lock bits. 
            so allocate the lock bits with value 1b and 
            dont change the remaining bits */
        if (remaining_lock_bits % MFUL_BYTE_SIZE_IN_BITS)
        {
            /* mod operation has resulted in a value, means lock bits ends in between a byte */
            uint8_t         mod_value = 0;

            remaining_lock_bytes = ((remaining_lock_bits /
                                    MFUL_BYTE_SIZE_IN_BITS) + 1);

            /* mod_value is used to fill the only partial bits and 
                remaining bits shall be untouched */
            mod_value = (uint8_t)(remaining_lock_bits % MFUL_BYTE_SIZE_IN_BITS);
            if (remaining_lock_bits > MFUL_BYTE_SIZE_IN_BITS)
            {
                /* lock bits to write is greater than 8 bits */
                while (lock_byte_index < (remaining_lock_bytes - 1))
                {
                    /* Set 1b to all bits left in the block */
                    lock_bytes_value[byte_index] = 0xFF;
                    lock_byte_index = (uint8_t)(lock_byte_index + 1);
                    byte_index = (uint8_t)(byte_index + 1);
                }
                /* Last byte of the lock bits shall be filled partially,
                    Set only the remaining lock bits and dont change 
                    the other bit value */
                lock_bytes_value[byte_index] = 0;
                lock_bytes_value[byte_index] = (uint8_t)
                        SET_BITS8 (lock_bytes_value[byte_index], 0, 
                                    mod_value, 1);
            }
            else
            {
                /* lock bits to write is less than 8 bits, so 
                    there is only one byte to write.
                    Set only the remaining lock bits and dont change 
                    the other bit value */
                lock_bytes_value[0] = (uint8_t)SET_BITS8 (lock_bytes_value[0], 0, 
                                                        mod_value, 1);
            }
        } /* if (remaining_lock_bits % MFUL_BYTE_SIZE_IN_BITS) */
        else
        {
            /* MOD operation is 00, that means entire byte value shall be 0xFF, means
            every bit shall be to 1 */
            remaining_lock_bytes = (remaining_lock_bits /
                                    MFUL_BYTE_SIZE_IN_BITS);

            while (lock_byte_index < remaining_lock_bytes)
            {
                /* Set 1b to all bits left in the block */
                lock_bytes_value[byte_index] = 0xFF;
                lock_byte_index = (uint8_t)(lock_byte_index + 1);
                byte_index = (uint8_t)(byte_index + 1);
            }
        } /* else of if (remaining_lock_bits % MFUL_BYTE_SIZE_IN_BITS) */
        ps_type2_info->LockBitsWritten = (uint8_t)(ps_type2_info->LockBitsWritten + 
                                            remaining_lock_bits);
    } /* if (no_of_bits_left_in_block >= remaining_lock_bits) */
    else
    {
        /* Update till the left bits in the block and then carry 
            out next operation after this write */
        while (lock_byte_index < (no_of_bits_left_in_block / MFUL_BYTE_SIZE_IN_BITS))
        {
            /* Set 1b to all bits left in the block */
            lock_bytes_value[byte_index] = 0xFF;
            lock_byte_index = (uint8_t)(lock_byte_index + 1);
            byte_index = (uint8_t)(byte_index + 1);
        }
        ps_type2_info->LockBitsWritten = (uint8_t)(ps_type2_info->LockBitsWritten + 
                                            no_of_bits_left_in_block);
    } /* else of if (no_of_bits_left_in_block >= remaining_lock_bits) */

    
    /* Copy the values back to the DynLockBytes structure member */ 
    (void)memcpy ((void*)ps_type2_info->DynLockBytes,
                (void *)lock_bytes_value,  
                sizeof (ps_type2_info->DynLockBytes));


    NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_WR_DYN_LOCK_BYTES;
    result = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);

    return result;
}

static 
NFCSTATUS 
phFriNfc_MfUL_ReadWriteLockBytes (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt)
{
    NFCSTATUS                       result = NFCSTATUS_SUCCESS;
    phFriNfc_Type2_AddInfo_t        *ps_type2_info = 
                                    &(NdefSmtCrdFmt->AddInfo.Type2Info);
    uint8_t                         write_flag = FALSE;

    if (/* Lock bytes starts from the beginning of the block */
        (0 == ps_type2_info->LockByteNumber)
        /* To make sure this is the first read */
        && (ps_type2_info->CurrentBlock == ps_type2_info->LockBlockNumber)
        /* Lock bytes are greater than or equal to the block size, i.e., 4 bytes */
        && (phFriNfc_MfUL_CalcRemainingLockBits (NdefSmtCrdFmt)
        >= (MFUL_BLOCK_SIZE_IN_BYTES * MFUL_BYTE_SIZE_IN_BITS)))
    {        
        /* Then directly write the lock bytes, dont waste time for read  */
        (void)memset ((void *)ps_type2_info->DynLockBytes, 0xFF, 
                        sizeof (ps_type2_info->DynLockBytes));
        write_flag = TRUE;
    }
    else if (ps_type2_info->CurrentBlock == ps_type2_info->LockBlockNumber)
    {
        /* Read is mandatory, First read and then update the block, 
            because chances are there that lock byte may start in between 
            the block */
    }
    else if (/* check if remaining bytes exceeds or same as the block size */
        (phFriNfc_MfUL_CalcRemainingLockBits (NdefSmtCrdFmt)
        >= (MFUL_BLOCK_SIZE_IN_BYTES * MFUL_BYTE_SIZE_IN_BITS)))
    {
        /* Then directly write the lock bytes, dont waste time for read */
        (void)memset ((void *)ps_type2_info->DynLockBytes, 0xFF, 
                        sizeof (ps_type2_info->DynLockBytes));
        write_flag = TRUE;
    }
    else
    {
        /* Read is mandatory, First read and then update the block */
    }

    if (write_flag)
    {
        NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_WR_DYN_LOCK_BYTES;
        result = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
    }
    else
    {
        NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_RD_DYN_LOCK_BYTES;
        result = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
    }

    return result;
}

static 
NFCSTATUS 
phFriNfc_MfUL_GetDefaultLockBytesInfo (
    phFriNfc_sNdefSmtCrdFmt_t          *NdefSmtCrdFmt)
{
    NFCSTATUS                       result = NFCSTATUS_SUCCESS;
    phFriNfc_Type2_AddInfo_t        *ps_type2_info = 
                                    &(NdefSmtCrdFmt->AddInfo.Type2Info);
    uint16_t                        lock_byte_start_addr = 0;

    /*  The position of the dynamic lock bits starts from 
        the first byte after the data area */
    lock_byte_start_addr = (uint16_t)(MFUL_INITIAL_BYTES_IGNORED + 
                        (ps_type2_info->OTPBytes[TYPE_2_MEM_SIZE_POSITION] * 8));

    ps_type2_info->LockBlockNumber = (uint8_t)(lock_byte_start_addr / 
                                                MFUL_BLOCK_SIZE_IN_BYTES);
    ps_type2_info->LockByteNumber = (uint8_t)(lock_byte_start_addr % 
                                                MFUL_BLOCK_SIZE_IN_BYTES);
    /* Default settings 
       NoOfLockBits = [(DataAreaSize - 48)/8] */
    ps_type2_info->NoOfLockBits = (uint8_t)
        (((ps_type2_info->OTPBytes[TYPE_2_MEM_SIZE_POSITION] * 8) - 48)/8);

    return result;
}

static 
NFCSTATUS 
phFriNfc_MfUL_ParseTLVs (
    phFriNfc_sNdefSmtCrdFmt_t       *NdefSmtCrdFmt, 
    uint8_t                         *data_to_parse, 
    uint8_t                         size_to_parse)
{
    NFCSTATUS                       result = NFCSTATUS_SUCCESS;
    static uint8_t                  lock_mem_ndef_index = 0;
    static uint8_t                  skip_lock_mem_size = 0;
    static uint16_t                 card_size_remaining = 0;
    static uint16_t                 ndef_data_size = 0;
    static phFriNfc_MfUL_Parse_t    parse_tlv = LOCK_TLV_T;
    uint8_t                         parse_index = 0;
    
    if ((0 == card_size_remaining) && (0 == parse_index))
    {
        /* card size is calculated only once */
        card_size_remaining = (uint16_t)
            (NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes[TYPE_2_MEM_SIZE_POSITION] * 8);
    }

    while ((parse_index < size_to_parse)
        && (NFCSTATUS_SUCCESS == result)  
        && (NDEF_TLV_V != parse_tlv)
        && (0 != card_size_remaining))
    {
        if (0 == skip_lock_mem_size)
        {
            /* Skip the lock TLVs, so get the lock bits */
            skip_lock_mem_size = phFriNfc_MfUL_GetSkipSize (NdefSmtCrdFmt, 
                                        NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock, 
                                        parse_index);
        }

        if (0 != skip_lock_mem_size)
        {
            if (skip_lock_mem_size >= (size_to_parse - parse_index))
            {
                /* if skip size is more than the size to parse, then  */
                card_size_remaining = (uint16_t)(card_size_remaining - 
                                    (size_to_parse - parse_index));
                skip_lock_mem_size = (uint8_t)(skip_lock_mem_size - 
                                            ((size_to_parse - parse_index)));
                parse_index = size_to_parse;
            }
            else
            {
                card_size_remaining = (uint16_t)(card_size_remaining - 
                                        skip_lock_mem_size);

                parse_index = (uint8_t)(parse_index + skip_lock_mem_size);
                skip_lock_mem_size = 0;
            }
        }
        else
        {
            switch (parse_tlv)
            {
                case LOCK_TLV_T:
                {
                    switch (*(data_to_parse + parse_index))
                    {
                        case MFUL_NULL_TLV:
                        {
                            /* Do nothing, parse further */
                            break;
                        }

                        case LOCK_CTRL_TYPE_IN_TLV:
                        {
                            parse_tlv = LOCK_TLV_L;
                            break;
                        }

                        case NDEF_TYPE_IN_TLV:
                        {
                            parse_tlv = NDEF_TLV_L;
                            /* Default lock bytes shall be taken */
                            NdefSmtCrdFmt->AddInfo.Type2Info.DefaultLockBytesFlag = 
                                                                            TRUE;
                            result = phFriNfc_MfUL_GetDefaultLockBytesInfo (NdefSmtCrdFmt);
                            break;
                        }

                        default:
                        {
                            parse_tlv = LOCK_TLV_T;
                            result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                                NFCSTATUS_NO_NDEF_SUPPORT);
                            break;
                        }
                    }
                    break;
                }

                case LOCK_TLV_L:
                {
                    if (LOCK_CTRL_LEN_IN_TLV == *(data_to_parse + parse_index))
                    {
                        parse_tlv = LOCK_TLV_V;
                    }
                    else
                    {
                        skip_lock_mem_size = 0;
                        parse_tlv = LOCK_TLV_T;
                        result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                                NFCSTATUS_NO_NDEF_SUPPORT);
                    }
                    break;
                }

                case LOCK_TLV_V:
                {
                    switch (lock_mem_ndef_index)
                    {
                        case 0:
                        case 1:
                        {
                            NdefSmtCrdFmt->AddInfo.Type2Info.DefaultLockBytesFlag = 
                                                                                FALSE;
                            NdefSmtCrdFmt->AddInfo.Type2Info.DynLockBytes[lock_mem_ndef_index] = 
                                            *(data_to_parse + parse_index);
                            lock_mem_ndef_index = (uint8_t)(lock_mem_ndef_index + 1);
                            break;
                        }

                        case 2:
                        {
                            NdefSmtCrdFmt->AddInfo.Type2Info.DynLockBytes[lock_mem_ndef_index] = 
                                            *(data_to_parse + parse_index);
                            parse_tlv = NDEF_TLV_T;
                            lock_mem_ndef_index = 0;
                            result = phFriNfc_MfUL_GetLockBytesInfo (NdefSmtCrdFmt);
                            break;
                        }

                        default:
                        {
                            skip_lock_mem_size = 0;
                            parse_tlv = LOCK_TLV_T;
                            result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                                NFCSTATUS_NO_NDEF_SUPPORT);
                            break;
                        }
                    }
                    break;
                } /* switch (lock_mem_ndef_index) in case LOCK_TLV_V */

                case NDEF_TLV_T:
                {
                    switch (*(data_to_parse + parse_index))
                    {
                        case MFUL_NULL_TLV:
                        {
                            /* Do nothing, parse further */
                            break;
                        }

                        case NDEF_TYPE_IN_TLV:
                        {
                            parse_tlv = NDEF_TLV_L;
                            break;
                        }

                        default:
                        {
                            skip_lock_mem_size = 0;
                            parse_tlv = LOCK_TLV_T;
                            result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                                NFCSTATUS_NO_NDEF_SUPPORT);
                            break;
                        }
                    }
                    break;
                } /* switch (*(data_to_parse + parse_index)) in case NDEF_TLV_T */

                case NDEF_TLV_L:
                {
                    switch (lock_mem_ndef_index)
                    {
                        case 0:
                        {
                            if (THREE_BYTE_LENGTH_FIELD == *(data_to_parse + parse_index))
                            {
                                lock_mem_ndef_index = (uint8_t)(lock_mem_ndef_index + 1);
                            }
                            else
                            {
                                ndef_data_size = *(data_to_parse + parse_index);
                                parse_tlv = NDEF_TLV_V;
                                lock_mem_ndef_index = 0;
                            }
                            break;
                        }

                        case 1:
                        {
                            ndef_data_size = (uint16_t)(*(data_to_parse + parse_index) << 8);
                            break;
                        }

                        case 2:
                        {
                            ndef_data_size = (uint16_t)(ndef_data_size | 
                                                        *(data_to_parse + parse_index));
                            parse_tlv = NDEF_TLV_V;
                            lock_mem_ndef_index = 0;
                            break;
                        }
                    } /* switch (lock_mem_ndef_index) in case NDEF_TLV_L */
                    break;
                }

                case NDEF_TLV_V:
                {
                    break;
                }
                
                default:
                {
                    skip_lock_mem_size = 0;
                    parse_tlv = LOCK_TLV_T;
                    result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                        NFCSTATUS_NO_NDEF_SUPPORT);
                    break;
                }
            } /* switch (parse_tlv) */
            
        } /* else part of if (0 != skip_lock_mem_size) */

        if (0 == card_size_remaining)
        {
            skip_lock_mem_size = 0;
            parse_tlv = LOCK_TLV_T;
            result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                NFCSTATUS_NO_NDEF_SUPPORT);
        }
        else if (NDEF_TLV_V != parse_tlv)
        {
            /* Increment the index */
            parse_index = (uint8_t)(parse_index + 1);
            /* card size is decremented as the memory area is parsed  */
            card_size_remaining = (uint16_t)(card_size_remaining - 1);
        }
        else
        {
            /* L field of the NDEF TLV
                L field can have 1 byte or also 3 bytes 
               */
            uint8_t length_to_deduct = 1;

            if ((NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes[TYPE_2_MEM_SIZE_POSITION] 
                * 8) >= THREE_BYTE_LENGTH_FIELD)
            {
                length_to_deduct = 3;
            }
            /* parse_tlv has reached the VALUE field of the NDEF TLV */
            if ((card_size_remaining - length_to_deduct) < ndef_data_size)
            {
                result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                    NFCSTATUS_NO_NDEF_SUPPORT);
            }

            lock_mem_ndef_index = 0;
            skip_lock_mem_size = 0;
            card_size_remaining = 0;
        }
    } /* while ((parse_index < size_to_parse)
        && (NFCSTATUS_SUCCESS != result)  
        && (NDEF_TLV_V != parse_tlv)
        && (0 != card_size_remaining)) */
    
    if ((NDEF_TLV_V == parse_tlv) || (NFCSTATUS_SUCCESS != result))
    {
        parse_tlv = LOCK_TLV_T;
    }
    else
    {
        NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_RO_NDEF_PARSE_RD_BYTES;
        NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 
                        (NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock + 4);

        result = phFriNfc_MfUL_H_WrRd (NdefSmtCrdFmt);
    }

    if (NFCSTATUS_PENDING != result)
    {
        lock_mem_ndef_index = 0;
        skip_lock_mem_size = 0;
        card_size_remaining = 0;
    }
    return result;
}

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

#endif /* #ifdef FRINFC_READONLY_NDEF */

static NFCSTATUS phFriNfc_MfUL_H_WrRd( phFriNfc_sNdefSmtCrdFmt_t *NdefSmtCrdFmt )
{
    NFCSTATUS   Result = NFCSTATUS_SUCCESS;

    /* Fill the send buffer */
    phFriNfc_MfUL_H_fillSendBuf(NdefSmtCrdFmt, 
                            NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock);

    /* Call transceive */
    Result = phFriNfc_MfUL_H_Transceive (NdefSmtCrdFmt);
    
    return Result;
}

static NFCSTATUS phFriNfc_MfUL_H_Transceive(phFriNfc_sNdefSmtCrdFmt_t    *NdefSmtCrdFmt)
{
    NFCSTATUS   Result = NFCSTATUS_SUCCESS;

    /* set the data for additional data exchange*/
    NdefSmtCrdFmt->psDepAdditionalInfo.DepFlags.MetaChaining = 0;
    NdefSmtCrdFmt->psDepAdditionalInfo.DepFlags.NADPresent = 0;
    NdefSmtCrdFmt->psDepAdditionalInfo.NAD = 0;

    /*set the completion routines for the card operations*/
    NdefSmtCrdFmt->SmtCrdFmtCompletionInfo.CompletionRoutine = phFriNfc_NdefSmtCrd_Process;
    NdefSmtCrdFmt->SmtCrdFmtCompletionInfo.Context = NdefSmtCrdFmt;

    *NdefSmtCrdFmt->SendRecvLength = PH_FRINFC_SMTCRDFMT_MAX_SEND_RECV_BUF_SIZE;

    /* Call the Overlapped HAL Transceive function */ 
    Result = phFriNfc_OvrHal_Transceive(    NdefSmtCrdFmt->LowerDevice,
                                            &NdefSmtCrdFmt->SmtCrdFmtCompletionInfo,
                                            NdefSmtCrdFmt->psRemoteDevInfo,
                                            NdefSmtCrdFmt->Cmd,
                                            &NdefSmtCrdFmt->psDepAdditionalInfo,
                                            NdefSmtCrdFmt->SendRecvBuf,
                                            NdefSmtCrdFmt->SendLength,
                                            NdefSmtCrdFmt->SendRecvBuf,
                                            NdefSmtCrdFmt->SendRecvLength);
    return Result;
}

static void phFriNfc_MfUL_H_fillSendBuf( phFriNfc_sNdefSmtCrdFmt_t *NdefSmtCrdFmt,
                                 uint8_t                    BlockNo)
{
#ifdef PH_NDEF_MIFARE_ULC
    uint8_t     NDEFTLV1[4] = {0x01, 0x03, 0xA0, 0x10}; 
    uint8_t     NDEFTLV2[4] = {0x44, 0x03, 0x00, 0xFE}; 
#endif /* #ifdef PH_NDEF_MIFARE_ULC */
    uint8_t     NDEFTLV[4] = {0x03, 0x00, 0xFE, 0x00};


    
    
    /* First byte for send buffer is always the block number */
    NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_0] = (uint8_t)BlockNo;
    switch(NdefSmtCrdFmt->State)
    {
#ifdef FRINFC_READONLY_NDEF

        case PH_FRINFC_MFUL_FMT_RO_RD_16BYTES:
        {
#ifdef PH_HAL4_ENABLE
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareRead;
#else
        /* Read command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareRead;
#endif /* #ifdef PH_HAL4_ENABLE */
            *NdefSmtCrdFmt->SendRecvBuf = RD_LOCK_OTP_BLOCK_NUMBER;
            /* Send length for read command is always one */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_1;
            break;
        }

#ifdef PH_NDEF_MIFARE_ULC

        case PH_FRINFC_MFUL_FMT_RO_NDEF_PARSE_RD_BYTES:
        {
#ifdef PH_HAL4_ENABLE
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareRead;
#else
        /* Read command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareRead;
#endif /* #ifdef PH_HAL4_ENABLE */
            *NdefSmtCrdFmt->SendRecvBuf = 
                    NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock;
            /* Send length for read command is always one */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_1;
            break;
        }

        case PH_FRINFC_MFUL_FMT_RO_RD_DYN_LOCK_BYTES:
        {
#ifdef PH_HAL4_ENABLE
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareRead;
#else
        /* Read command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareRead;
#endif /* #ifdef PH_HAL4_ENABLE */
            *NdefSmtCrdFmt->SendRecvBuf = NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock;
            /* Send length for read command is always one */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_1;
            break;
        }

        case PH_FRINFC_MFUL_FMT_RO_WR_DYN_LOCK_BYTES:
        {
#ifdef PH_HAL4_ENABLE
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareWrite4;
#else
            /* Write command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareWrite4;
#endif /* #ifdef PH_HAL4_ENABLE */

            /* Send length for read command is always one */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_5;
            *NdefSmtCrdFmt->SendRecvBuf = NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock;
            (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1],
                         NdefSmtCrdFmt->AddInfo.Type2Info.DynLockBytes,
                         PH_FRINFC_MFUL_FMT_VAL_4);
            break;
        }

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

        case PH_FRINFC_MFUL_FMT_RO_WR_LOCK_BYTES:
        {
#ifdef PH_HAL4_ENABLE
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareWrite4;
#else
            /* Read command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareWrite4;
#endif /* #ifdef PH_HAL4_ENABLE */

            /* Send length for read command is always one */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_5;
            *NdefSmtCrdFmt->SendRecvBuf = RD_LOCK_OTP_BLOCK_NUMBER;
            (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1],
                         NdefSmtCrdFmt->AddInfo.Type2Info.LockBytes,
                         PH_FRINFC_MFUL_FMT_VAL_4);
            break;
        }

        case PH_FRINFC_MFUL_FMT_RO_WR_OTP_BYTES:
        {
#ifdef PH_HAL4_ENABLE
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareWrite4;
#else
            /* Read command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareWrite4;
#endif /* #ifdef PH_HAL4_ENABLE */

            /* Send length for read command is always one */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_5;
            *NdefSmtCrdFmt->SendRecvBuf = OTP_BLOCK_NUMBER;
            (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1],
                         NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                         PH_FRINFC_MFUL_FMT_VAL_4);
            break;
        }

#endif /* #ifdef FRINFC_READONLY_NDEF */

    case PH_FRINFC_MFUL_FMT_RD_16BYTES:
#ifdef PH_HAL4_ENABLE
        NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareRead;
#else
        /* Read command */
        NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareRead;
#endif /* #ifdef PH_HAL4_ENABLE */
        /* Send length for read command is always one */
        NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_1;
        break;

    case PH_FRINFC_MFUL_FMT_WR_OTPBYTES:
        /* Send length for read command is always Five */
        NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_5;
        /* Write command */
#ifdef PH_HAL4_ENABLE
        NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareWrite4;
#else
        NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareWrite4;
#endif /* #ifdef PH_HAL4_ENABLE */
        /* Copy the OTP bytes */
        (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1], 
                    NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                    PH_FRINFC_MFUL_FMT_VAL_4);
        break;

    case PH_FRINFC_MFUL_FMT_WR_TLV:
#ifndef PH_NDEF_MIFARE_ULC      
    default:
#endif /* #ifndef PH_NDEF_MIFARE_ULC */    
        /* Send length for read command is always Five */
        NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_5;
        /* Write command */
#ifdef PH_HAL4_ENABLE
        NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareWrite4;
#else
        NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareCmdListMifareWrite4;
#endif /* #ifdef PH_HAL4_ENABLE */        
        /* Copy the NDEF TLV */
#ifdef PH_NDEF_MIFARE_ULC

        if (NdefSmtCrdFmt->CardType == PH_FRINFC_NDEFMAP_MIFARE_ULC_CARD)
        {
            (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1], 
                    NDEFTLV1,
                    PH_FRINFC_MFUL_FMT_VAL_4);
        }
        else if (NdefSmtCrdFmt->CardType == PH_FRINFC_NDEFMAP_MIFARE_UL_CARD)
        {
            (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1], 
                NDEFTLV,
                PH_FRINFC_MFUL_FMT_VAL_4);
        }
        else
        {
        }
#else
        (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1], 
                    NDEFTLV,
                    PH_FRINFC_MFUL_FMT_VAL_4);

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

        break;

#ifdef PH_NDEF_MIFARE_ULC
    case PH_FRINFC_MFUL_FMT_WR_TLV1:
        if (NdefSmtCrdFmt->CardType == PH_FRINFC_NDEFMAP_MIFARE_ULC_CARD)
        {
            /* Send length for write command is always Five */
            NdefSmtCrdFmt->SendLength = PH_FRINFC_MFUL_FMT_VAL_5;
            /* Write command */
            NdefSmtCrdFmt->Cmd.MfCmd = phHal_eMifareWrite4;
            (void)memcpy(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_1], 
                        NDEFTLV2,
                        PH_FRINFC_MFUL_FMT_VAL_4);
        }
        break;
    default:
        break;
#endif /* #ifdef PH_NDEF_MIFARE_ULC */

    
    }
}

static NFCSTATUS phFriNfc_MfUL_H_ProRd16Bytes( phFriNfc_sNdefSmtCrdFmt_t *NdefSmtCrdFmt )
{
    NFCSTATUS   Result = PHNFCSTVAL(CID_FRI_NFC_NDEF_SMTCRDFMT,
                                    NFCSTATUS_FORMAT_ERROR);
    uint32_t    memcompare = PH_FRINFC_MFUL_FMT_VAL_0;
    uint8_t     ZeroBuf[] = {0x00, 0x00, 0x00, 0x00};

#ifdef PH_NDEF_MIFARE_ULC
    uint8_t                 OTPByteUL[] = PH_FRINFC_MFUL_FMT_OTP_BYTES;
    uint8_t                 OTPByteULC[] = PH_FRINFC_MFULC_FMT_OTP_BYTES;
#endif /* #ifdef PH_NDEF_MIFARE_ULC */

    /* Check the lock bits (byte number 2 and 3 of block number 2) */
    if ((NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_2] == 
        PH_FRINFC_MFUL_FMT_LOCK_BITS_VAL) && 
        (NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_3] == 
        PH_FRINFC_MFUL_FMT_LOCK_BITS_VAL))
    {

#ifdef PH_NDEF_MIFARE_ULC

        if (NdefSmtCrdFmt->SendRecvBuf[8] == 0x02 && 
            NdefSmtCrdFmt->SendRecvBuf[9] == 0x00) 
        {
            NdefSmtCrdFmt->CardType = PH_FRINFC_NDEFMAP_MIFARE_ULC_CARD;
            
            (void)memcpy(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                        OTPByteULC,
                        sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes));
        }
        else if (NdefSmtCrdFmt->SendRecvBuf[8] == 0xFF && 
                NdefSmtCrdFmt->SendRecvBuf[9] == 0xFF)
        {
            NdefSmtCrdFmt->CardType = PH_FRINFC_NDEFMAP_MIFARE_UL_CARD;
            
            (void)memcpy(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                        OTPByteUL,
                        sizeof(NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes));
        }
        else
        {
            NdefSmtCrdFmt->CardType = PH_FRINFC_NDEFMAP_MIFARE_UL_CARD;
        }

#endif /* #ifdef PH_NDEF_MIFARE_ULC */

        memcompare = (uint32_t) 
                    MemCompare1(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_4],
                            NdefSmtCrdFmt->AddInfo.Type2Info.OTPBytes,
                            PH_FRINFC_MFUL_FMT_VAL_4);

        if (memcompare == PH_FRINFC_MFUL_FMT_VAL_0)
        {
            /* Write NDEF TLV in block number 4 */
            NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 
                                                PH_FRINFC_MFUL_FMT_VAL_4;
            /* Card already have the OTP bytes so write TLV */
            NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_WR_TLV;
        }
        else
        {
            /* IS the card new, OTP bytes = {0x00, 0x00, 0x00, 0x00} */
            memcompare = (uint32_t)MemCompare1(&NdefSmtCrdFmt->SendRecvBuf[PH_FRINFC_MFUL_FMT_VAL_4],
                                ZeroBuf,
                                PH_FRINFC_MFUL_FMT_VAL_4);
            /* If OTP bytes are Zero then the card is Zero */
            if (memcompare == PH_FRINFC_MFUL_FMT_VAL_0)
            {
                /* Write OTP bytes in block number 3 */
                NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 
                                                PH_FRINFC_MFUL_FMT_VAL_3;
                /* Card already have the OTP bytes so write TLV */
                NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_WR_OTPBYTES;
            }
        }
    }

    

#ifdef PH_NDEF_MIFARE_ULC
    if(
        ((NdefSmtCrdFmt->State == PH_FRINFC_MFUL_FMT_WR_TLV) || 
        (NdefSmtCrdFmt->State == PH_FRINFC_MFUL_FMT_WR_OTPBYTES)) &&
        ((NdefSmtCrdFmt->CardType == PH_FRINFC_NDEFMAP_MIFARE_ULC_CARD) ||
        (NdefSmtCrdFmt->CardType == PH_FRINFC_NDEFMAP_MIFARE_UL_CARD))
        )
#else
    if((NdefSmtCrdFmt->State == PH_FRINFC_MFUL_FMT_WR_TLV) || 
        (NdefSmtCrdFmt->State == PH_FRINFC_MFUL_FMT_WR_OTPBYTES))
#endif /* #ifdef PH_NDEF_MIFARE_ULC */        
    {
        Result = phFriNfc_MfUL_H_WrRd(NdefSmtCrdFmt);
    }
    return Result;
}

static NFCSTATUS phFriNfc_MfUL_H_ProWrOTPBytes( phFriNfc_sNdefSmtCrdFmt_t *NdefSmtCrdFmt )
{
    NFCSTATUS   Result = NFCSTATUS_SUCCESS;
    /* Card already have the OTP bytes so write TLV */
    NdefSmtCrdFmt->State = PH_FRINFC_MFUL_FMT_WR_TLV;

    /* Write NDEF TLV in block number 4 */
    NdefSmtCrdFmt->AddInfo.Type2Info.CurrentBlock = 
                                PH_FRINFC_MFUL_FMT_VAL_4;

    Result = phFriNfc_MfUL_H_WrRd(NdefSmtCrdFmt);
    return Result;
}