2009-08-27 12:49:45 +02:00
/**
2009-10-12 16:52:26 +02:00
* Public platform independent Near Field Communication ( NFC ) library
*
* Copyright ( C ) 2009 , Roel Verdult
*
* This program is free software : you can redistribute it and / or modify it
* under the terms of the GNU Lesser General Public License as published by the
* Free Software Foundation , either version 3 of the License , or ( at your
* option ) any later version .
*
* This program is distributed in the hope that it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>
*
*
2009-08-27 12:49:45 +02:00
* @ file libnfc . h
* @ brief libnfc interface
*
2009-10-12 16:52:26 +02:00
* Provide all usefull functions ( API ) to handle NFC devices .
2009-08-27 12:49:45 +02:00
*/
2009-10-13 14:43:52 +02:00
# ifndef _LIBNFC_H_
2009-10-12 16:52:26 +02:00
# define _LIBNFC_H_
2009-05-27 12:13:19 +02:00
# include <stdint.h>
2009-05-27 16:05:07 +02:00
# include <stdbool.h>
2009-05-27 12:13:19 +02:00
2009-11-10 10:47:59 +01:00
2009-11-09 12:23:33 +01:00
# include "nfc-types.h"
2009-04-29 14:47:41 +02:00
2009-11-02 08:46:29 +01:00
# ifdef __cplusplus
# define NFCAPI extern "C" {
# endif // __cplusplus
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_device_t * nfc_connect ( nfc_device_desc_t * pndd )
2009-08-27 12:49:45 +02:00
* @ brief Connect to a NFC device
2009-10-02 11:52:02 +02:00
* @ param pndd Device description if specific device is wanted , NULL otherwise
2009-11-09 12:23:33 +01:00
* @ return Returns pointer to a nfc_device_t struct if successfull ; otherwise returns INVALID_DEVICE_INFO value .
2009-08-27 12:49:45 +02:00
*
2009-10-02 11:52:02 +02:00
* If \ a pndd is NULL , the first available NFC device is claimed by libnfc .
2009-08-27 12:49:45 +02:00
* It will automatically search the system using all available drivers to determine a device is free .
2009-09-04 15:24:34 +02:00
*
2009-10-02 11:52:02 +02:00
* If \ a pndd is passed then libnfc will try to claim the right device using information provided by this struct .
2009-09-04 15:24:34 +02:00
*
2009-11-09 12:23:33 +01:00
* When it has successfully claimed a NFC device , memory is allocated to save the device information . It will return a pointer to a nfc_device_t struct .
2009-08-27 12:49:45 +02:00
* This pointer should be supplied by every next function of libnfc that should perform an action with this device .
*/
2009-11-09 12:23:33 +01:00
nfc_device_t * nfc_connect ( nfc_device_desc_t * pndd ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn void nfc_disconnect ( nfc_device_t * pnd )
2009-08-27 12:49:45 +02:00
* @ brief Disconnect from a NFC device
2009-11-09 12:23:33 +01:00
* @ param pnd nfc_device_t struct pointer that represent currently used device
2009-08-27 12:49:45 +02:00
*
2009-11-09 12:23:33 +01:00
* Initiator is disconnected and the device , including allocated nfc_device_t struct , is released .
2009-08-27 12:49:45 +02:00
*/
2009-11-09 12:23:33 +01:00
void nfc_disconnect ( nfc_device_t * pnd ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_configure ( nfc_device_t * pnd , const dev_config_option dco , const bool bEnable )
2009-08-27 12:49:45 +02:00
* @ brief Configure advanced NFC device settings
* @ return Returns true if action was successfully performed ; otherwise returns false .
2009-11-09 12:23:33 +01:00
* @ param pnd nfc_device_t struct pointer that represent currently used device
2009-08-27 12:49:45 +02:00
* @ param dco dev_config_option struct that contains options to set to device
* @ param bEnable boolean
*
* Configures parameters and registers that control for example timing , modulation , frame and error handling .
* There are different categories for configuring the PN53X chip features ( handle , activate , infinite and accept ) .
* These are defined to organize future settings that will become available when they are needed .
*/
2009-11-09 12:23:33 +01:00
bool nfc_configure ( nfc_device_t * pnd , const dev_config_option dco , const bool bEnable ) ;
2009-04-29 14:47:41 +02:00
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_init ( const nfc_device_t * pnd )
2009-08-27 12:49:45 +02:00
* @ brief Initialize NFC device as initiator ( reader )
* @ return Returns true if action was successfully performed ; otherwise returns false .
2009-11-09 12:23:33 +01:00
* @ param pnd nfc_device_t struct pointer that represent currently used device
2009-08-27 12:49:45 +02:00
*
* The NFC device is configured to function as RFID reader . After initialization it can be used to communicate to passive RFID tags and active NFC devices . The reader will act as initiator to communicate peer 2 peer ( NFCIP ) to other active NFC devices .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_init ( const nfc_device_t * pnd ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_select_tag ( const nfc_device_t * pnd , const init_modulation im , const byte_t * pbtInitData , const size_t szInitDataLen , tag_info * pti )
2009-08-27 12:49:45 +02:00
* @ brief Select a passive or emulated tag
* @ return Returns true if action was successfully performed ; otherwise returns false .
2009-11-09 12:23:33 +01:00
* @ param pnd nfc_device_t struct pointer that represent currently used device
2009-08-27 12:49:45 +02:00
* @ param im Desired modulation
2009-09-24 16:33:42 +02:00
* @ param pbtInitData Optional initiator data used for Felica , ISO14443B , Topaz Polling or for ISO14443A selecting a specific UID .
2009-10-02 11:52:02 +02:00
* @ param szInitDataLen Length of initiator data \ a pbtInitData .
2009-08-27 12:49:45 +02:00
*
* The NFC device will try to find the available passive tags . Some NFC devices are capable to emulate passive tags . The standards ( ISO18092 and ECMA - 340 ) describe the modulation that can be used for reader to passive communications . The chip needs to know with what kind of tag it is dealing with , therefore the initial modulation and speed ( 106 , 212 or 424 kbps ) should be supplied .
* @ note For every initial modulation type there is a different collection of information returned ( in tag_info pointer pti ) They all fit in the data - type which is called tag_info . This is a union which contains the tag information that belongs to the according initial modulation type .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_select_tag ( const nfc_device_t * pnd , const init_modulation im , const byte_t * pbtInitData , const size_t szInitDataLen , tag_info * pti ) ;
2009-08-27 12:49:45 +02:00
2009-09-03 15:47:26 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_select_dep_target ( const nfc_device_t * pnd , const init_modulation im , const byte_t * pbtPidData , const size_t szPidDataLen , const byte_t * pbtNFCID3i , const size_t szNFCID3iDataLen , const byte_t * pbtGbData , const size_t szGbDataLen , tag_info * pti ) ;
2009-09-03 15:47:26 +02:00
* @ brief Select a target and request active or passive mode for DEP ( Data Exchange Protocol )
* @ return Returns true if action was successfully performed ; otherwise returns false .
2009-11-09 12:23:33 +01:00
* @ param pnd nfc_device_t struct pointer that represent currently used device
2009-09-03 15:47:26 +02:00
* @ param im Desired modulation ( IM_ACTIVE_DEP or IM_PASSIVE_DEP for active , respectively passive mode )
* @ param pbtPidData passive initiator data , 4 or 5 bytes long , ( optional , only for IM_PASSIVE_DEP , can be NULL )
* @ param pbtNFCID3i the NFCID3 , 10 bytes long , of the initiator ( optional , can be NULL )
* @ param pbtGbData generic data of the initiator , max 48 bytes long , ( optional , can be NULL )
*
* The NFC device will try to find the available target . The standards ( ISO18092 and ECMA - 340 ) describe the modulation that can be used for reader to passive communications .
* @ note tag_info_dep will be returned when the target was acquired successfully .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_select_dep_target ( const nfc_device_t * pnd , const init_modulation im , const byte_t * pbtPidData , const size_t szPidDataLen , const byte_t * pbtNFCID3i , const size_t szNFCID3iDataLen , const byte_t * pbtGbData , const size_t szGbDataLen , tag_info * pti ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_deselect_tag ( const nfc_device_t * pnd ) ;
2009-08-27 12:49:45 +02:00
* @ brief Deselect a selected passive or emulated tag
* @ return Returns true if action was successfully performed ; otherwise returns false .
2009-11-09 12:23:33 +01:00
* @ param pnd nfc_device_t struct pointer that represent currently used device
2009-08-27 12:49:45 +02:00
*
* After selecting and communicating with a passive tag , this function could be used to deactivate and release the tag . This is very useful when there are multiple tags available in the field . It is possible to use the nfc_initiator_select_tag ( ) function to select the first available tag , test it for the available features and support , deselect it and skip to the next tag until the correct tag is found .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_deselect_tag ( const nfc_device_t * pnd ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_transceive_bits ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxBits , const byte_t * pbtTxPar , byte_t * pbtRx , size_t * pszRxBits , byte_t * pbtRxPar )
2009-08-27 12:49:45 +02:00
* @ brief Transceive raw bit - frames
* @ return Returns true if action was successfully performed ; otherwise returns false .
* @ param pbtTx contains a byte array of the frame that needs to be transmitted .
2009-10-02 11:52:02 +02:00
* @ param szTxBits contains the length in bits .
* @ note For example the REQA ( 0x26 ) command ( first anti - collision command of ISO14443 - A ) must be precise 7 bits long . This is not possible by using nfc_initiator_transceive_bytes ( ) . With that function you can only communicate frames that consist of full bytes . When you send a full byte ( 8 bits + 1 parity ) with the value of REQA ( 0x26 ) , a tag will simply not respond . More information about this can be found in the anti - colision example .
2009-08-27 12:49:45 +02:00
* @ param pbtTxPar parameter contains a byte array of the corresponding parity bits needed to send per byte .
2009-10-02 11:52:02 +02:00
* @ note For example if you send the SELECT_ALL ( 0x93 , 0x20 ) = [ 10010011 , 00100000 ] command , you have to supply the following parity bytes ( 0x01 , 0x00 ) to define the correct odd parity bits . This is only an example to explain how it works , if you just are sending two bytes with ISO14443 - A compliant parity bits you better can use the nfc_initiator_transceive_bytes ( ) function .
* @ returns The received response from the tag will be stored in the parameters ( pbtRx , pszRxBits and pbtRxPar ) . They work the same way as the corresponding parameters for transmission .
2009-08-27 12:49:45 +02:00
*
* The NFC reader will transmit low - level messages where only the modulation is handled by the PN53X chip . Construction of the frame ( data , CRC and parity ) is completely done by libnfc . This can be very useful for testing purposes . Some protocols ( e . g . MIFARE Classic ) require to violate the ISO14443 - A standard by sending incorrect parity and CRC bytes . Using this feature you are able to simulate these frames .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_transceive_bits ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxBits , const byte_t * pbtTxPar , byte_t * pbtRx , size_t * pszRxBits , byte_t * pbtRxPar ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_transceive_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen , byte_t * pbtRx , size_t * pszRxLen )
2009-08-27 12:49:45 +02:00
* @ brief Transceive byte and APDU frames
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* The reader will transmit the supplied bytes in pbtTx to the target ( tag ) . It waits for the response and stores the received bytes in the pbtRx byte array . The parity bits are handled by the PN53X chip . The CRC can be generated automatically or handled manually . Using this function , frames can be communicated very fast via the NFC reader to the tag . Tests show that on average this way of communicating is much faster than using the regular driver / middle - ware ( often supplied by manufacturers ) .
* @ warning The configuration option DCO_HANDLE_PARITY must be set to true ( the default value ) .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_transceive_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen , byte_t * pbtRx , size_t * pszRxLen ) ;
2009-08-27 12:49:45 +02:00
2009-09-03 15:47:26 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_transceive_dep_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen , byte_t * pbtRx , size_t * pszRxLen )
2009-09-03 15:47:26 +02:00
* @ brief Transceive data
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* The reader will transmit the supplied ( data ) bytes in pbtTx to the target ( tag ) . It waits for the response and stores the received bytes in the pbtRx byte array . The difference between this function and nfc_initiator_transceive_bytes is that here pbtTx and pbtRx contain * only * the data sent and received and not any additional commands , that is all handled internally by the PN53X .
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_transceive_dep_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen , byte_t * pbtRx , size_t * pszRxLen ) ;
2009-09-03 15:47:26 +02:00
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_initiator_mifare_cmd ( const nfc_device_t * pnd , const mifare_cmd mc , const uint8_t ui8Block , mifare_param * pmp )
2009-08-27 12:49:45 +02:00
* @ brief Execute a MIFARE Classic Command
* @ return Returns true if action was successfully performed ; otherwise returns false .
* @ param pmp Some commands need additional information . This information should be supplied in the mifare_param union .
*
* The specified MIFARE command will be executed on the tag . There are different commands possible , they all require the destination block number .
* @ note There are three different types of information ( Authenticate , Data and Value ) .
*
* First an authentication must take place using Key A or B . It requires a 48 bit Key ( 6 bytes ) and the UID . They are both used to initialize the internal cipher - state of the PN53X chip ( http : //libnfc.org/hardware/pn53x-chip). After a successful authentication it will be possible to execute other commands (e.g. Read/Write). The MIFARE Classic Specification (http://www.nxp.com/acrobat/other/identification/M001053_MF1ICS50_rev5_3.pdf) explains more about this process.
*/
2009-11-09 12:23:33 +01:00
bool nfc_initiator_mifare_cmd ( const nfc_device_t * pnd , const mifare_cmd mc , const uint8_t ui8Block , mifare_param * pmp ) ;
2009-05-27 12:13:19 +02:00
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_init ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxBits )
2009-08-27 12:49:45 +02:00
* @ brief Initialize NFC device as an emulated tag
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* This functionality allows the NFC device to act as an emulated tag . There seems to be quite some options available for this feature . Not all of the PN53X modulations are tested and documented at the moment . At the moment it could best be seen as a preliminary functionality .
*
* @ warning Be aware that this function will wait ( hang ) until a command is received that is not part of the anti - collision . The RATS command for example would wake up the emulator . After this is received , the send and receive functions can be used .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_init ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxBits ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_receive_bits ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxBits , byte_t * pbtRxPar )
2009-08-27 12:49:45 +02:00
* @ brief Receive bit - frames
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* This function makes it possible to receive ( raw ) bit - frames . It returns all the messages that are stored in the FIFO buffer of the PN53X chip . It does not require to send any frame and thereby could be used to snoop frames that are transmitted by a nearby reader . Check out the DCO_ACCEPT_MULTIPLE_FRAMES configuration option to avoid losing transmitted frames .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_receive_bits ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxBits , byte_t * pbtRxPar ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_receive_bytes ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxLen )
2009-08-27 12:49:45 +02:00
* @ brief Receive bytes and APDU frames
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* The main receive function that returns the received frames from a nearby reader .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_receive_bytes ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxLen ) ;
2009-08-27 12:49:45 +02:00
2009-09-03 15:47:26 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_receive_dep_bytes ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxLen )
2009-09-03 15:47:26 +02:00
* @ brief Receive data
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* The main receive function that returns the received data from a nearby reader . The difference between this function and nfc_target_receive_bytes is that here pbtRx contains * only * the data received and not any additional commands , that is all handled internally by the PN53X .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_receive_dep_bytes ( const nfc_device_t * pnd , byte_t * pbtRx , size_t * pszRxLen ) ;
2009-09-03 15:47:26 +02:00
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_send_bits ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxBits , const byte_t * pbtTxPar )
2009-08-27 12:49:45 +02:00
* @ brief Send raw bit - frames
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* This function can be used to transmit ( raw ) bit - frames to the reader .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_send_bits ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxBits , const byte_t * pbtTxPar ) ;
2009-08-27 12:49:45 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_send_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen )
2009-08-27 12:49:45 +02:00
* @ brief Send bytes and APDU frames
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* To communicate byte frames and APDU responses to the reader , this function could be used .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_send_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen ) ;
2009-02-12 13:39:05 +01:00
2009-09-03 15:47:26 +02:00
/**
2009-11-09 12:23:33 +01:00
* @ fn nfc_target_send_dep_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen )
2009-09-03 15:47:26 +02:00
* @ brief Send data
* @ return Returns true if action was successfully performed ; otherwise returns false .
*
* To communicate data to the reader , this function could be used . The difference between this function and nfc_target_send_bytes is that here pbtTx contains * only * the data sent and not any additional commands , that is all handled internally by the PN53X .
*/
2009-11-09 12:23:33 +01:00
bool nfc_target_send_dep_bytes ( const nfc_device_t * pnd , const byte_t * pbtTx , const size_t szTxLen ) ;
2009-09-03 15:47:26 +02:00
2009-11-02 15:05:03 +01:00
/**
* @ fn const char * nfc_version ( void )
* @ brief Returns the library version
* @ return Returns a string with the library version
*/
const char * nfc_version ( void ) ;
2009-11-02 08:46:29 +01:00
# ifdef __cplusplus
}
# endif // __cplusplus
2009-02-12 13:39:05 +01:00
# endif // _LIBNFC_H_