2014-01-06 12:58:19 -08:00
/*
* Marvell NFC driver : major functions
*
2015-10-26 10:27:37 +01:00
* Copyright ( C ) 2014 - 2015 Marvell International Ltd .
2014-01-06 12:58:19 -08:00
*
* This software file ( the " File " ) is distributed by Marvell International
* Ltd . under the terms of the GNU General Public License Version 2 , June 1991
* ( the " License " ) . You may use , redistribute and / or modify this File in
* accordance with the terms and conditions of the License , a copy of which
* is available on the worldwide web at
* http : //www.gnu.org/licenses/old-licenses/gpl-2.0.txt.
*
* THE FILE IS DISTRIBUTED AS - IS , WITHOUT WARRANTY OF ANY KIND , AND THE
* IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE
* ARE EXPRESSLY DISCLAIMED . The License provides additional details about
* this warranty disclaimer .
*/
# include <linux/module.h>
2015-06-11 11:25:46 +02:00
# include <linux/gpio.h>
# include <linux/delay.h>
2015-06-11 14:00:19 +02:00
# include <linux/of_gpio.h>
2014-01-06 12:58:19 -08:00
# include <linux/nfc.h>
# include <net/nfc/nci.h>
# include <net/nfc/nci_core.h>
# include "nfcmrvl.h"
static int nfcmrvl_nci_open ( struct nci_dev * ndev )
{
struct nfcmrvl_private * priv = nci_get_drvdata ( ndev ) ;
int err ;
if ( test_and_set_bit ( NFCMRVL_NCI_RUNNING , & priv - > flags ) )
return 0 ;
2015-10-26 10:27:41 +01:00
/* Reset possible fault of previous session */
clear_bit ( NFCMRVL_PHY_ERROR , & priv - > flags ) ;
2014-01-06 12:58:19 -08:00
err = priv - > if_ops - > nci_open ( priv ) ;
if ( err )
clear_bit ( NFCMRVL_NCI_RUNNING , & priv - > flags ) ;
return err ;
}
static int nfcmrvl_nci_close ( struct nci_dev * ndev )
{
struct nfcmrvl_private * priv = nci_get_drvdata ( ndev ) ;
if ( ! test_and_clear_bit ( NFCMRVL_NCI_RUNNING , & priv - > flags ) )
return 0 ;
priv - > if_ops - > nci_close ( priv ) ;
return 0 ;
}
static int nfcmrvl_nci_send ( struct nci_dev * ndev , struct sk_buff * skb )
{
struct nfcmrvl_private * priv = nci_get_drvdata ( ndev ) ;
nfc_info ( priv - > dev , " send entry, len %d \n " , skb - > len ) ;
skb - > dev = ( void * ) ndev ;
2015-06-11 14:00:19 +02:00
if ( priv - > config . hci_muxed ) {
2015-06-11 11:25:43 +02:00
unsigned char * hdr ;
unsigned char len = skb - > len ;
hdr = ( char * ) skb_push ( skb , NFCMRVL_HCI_EVENT_HEADER_SIZE ) ;
hdr [ 0 ] = NFCMRVL_HCI_COMMAND_CODE ;
hdr [ 1 ] = NFCMRVL_HCI_OGF ;
hdr [ 2 ] = NFCMRVL_HCI_OCF ;
hdr [ 3 ] = len ;
}
2014-01-06 12:58:19 -08:00
return priv - > if_ops - > nci_send ( priv , skb ) ;
}
2014-01-06 12:58:20 -08:00
static int nfcmrvl_nci_setup ( struct nci_dev * ndev )
{
2015-06-12 15:35:54 +02:00
__u8 val = 1 ;
nci_set_config ( ndev , NFCMRVL_PB_BAIL_OUT , 1 , & val ) ;
2014-01-06 12:58:20 -08:00
return 0 ;
}
2015-10-26 10:27:39 +01:00
static int nfcmrvl_nci_fw_download ( struct nci_dev * ndev ,
const char * firmware_name )
{
return nfcmrvl_fw_dnld_start ( ndev , firmware_name ) ;
}
2014-01-06 12:58:19 -08:00
static struct nci_ops nfcmrvl_nci_ops = {
. open = nfcmrvl_nci_open ,
. close = nfcmrvl_nci_close ,
. send = nfcmrvl_nci_send ,
2014-01-06 12:58:20 -08:00
. setup = nfcmrvl_nci_setup ,
2015-10-26 10:27:39 +01:00
. fw_download = nfcmrvl_nci_fw_download ,
2014-01-06 12:58:19 -08:00
} ;
2015-10-26 10:27:40 +01:00
struct nfcmrvl_private * nfcmrvl_nci_register_dev ( enum nfcmrvl_phy phy ,
void * drv_data ,
2015-06-11 14:00:19 +02:00
struct nfcmrvl_if_ops * ops ,
struct device * dev ,
struct nfcmrvl_platform_data * pdata )
2014-01-06 12:58:19 -08:00
{
struct nfcmrvl_private * priv ;
int rc ;
2015-10-26 10:27:40 +01:00
int headroom ;
int tailroom ;
2014-01-06 12:58:19 -08:00
u32 protocols ;
priv = kzalloc ( sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return ERR_PTR ( - ENOMEM ) ;
priv - > drv_data = drv_data ;
priv - > if_ops = ops ;
priv - > dev = dev ;
2015-10-26 10:27:40 +01:00
priv - > phy = phy ;
2015-06-11 11:25:46 +02:00
2015-06-11 14:00:19 +02:00
memcpy ( & priv - > config , pdata , sizeof ( * pdata ) ) ;
if ( priv - > config . reset_n_io ) {
2015-06-11 11:25:46 +02:00
rc = devm_gpio_request_one ( dev ,
2015-06-11 14:00:19 +02:00
priv - > config . reset_n_io ,
2015-06-11 11:25:46 +02:00
GPIOF_OUT_INIT_LOW ,
" nfcmrvl_reset_n " ) ;
if ( rc < 0 )
nfc_err ( dev , " failed to request reset_n io \n " ) ;
}
2015-06-11 11:25:43 +02:00
2015-10-26 10:27:40 +01:00
headroom = tailroom = 0 ;
2015-06-11 14:00:19 +02:00
if ( priv - > config . hci_muxed )
2015-10-26 10:27:40 +01:00
headroom + = NFCMRVL_HCI_EVENT_HEADER_SIZE ;
2014-01-06 12:58:19 -08:00
protocols = NFC_PROTO_JEWEL_MASK
2015-06-11 14:00:19 +02:00
| NFC_PROTO_MIFARE_MASK
| NFC_PROTO_FELICA_MASK
2014-01-06 12:58:19 -08:00
| NFC_PROTO_ISO14443_MASK
| NFC_PROTO_ISO14443_B_MASK
2015-06-12 15:35:52 +02:00
| NFC_PROTO_ISO15693_MASK
2014-01-06 12:58:19 -08:00
| NFC_PROTO_NFC_DEP_MASK ;
2015-06-11 11:25:43 +02:00
priv - > ndev = nci_allocate_device ( & nfcmrvl_nci_ops , protocols ,
2015-10-26 10:27:40 +01:00
headroom , tailroom ) ;
2014-01-06 12:58:19 -08:00
if ( ! priv - > ndev ) {
2015-04-07 00:17:00 -07:00
nfc_err ( dev , " nci_allocate_device failed \n " ) ;
2014-01-08 10:52:27 -08:00
rc = - ENOMEM ;
goto error ;
2014-01-06 12:58:19 -08:00
}
nci_set_drvdata ( priv - > ndev , priv ) ;
rc = nci_register_device ( priv - > ndev ) ;
if ( rc ) {
2015-04-07 00:17:00 -07:00
nfc_err ( dev , " nci_register_device failed %d \n " , rc ) ;
2015-10-26 10:27:39 +01:00
goto error_free_dev ;
}
/* Ensure that controller is powered off */
nfcmrvl_chip_halt ( priv ) ;
rc = nfcmrvl_fw_dnld_init ( priv ) ;
if ( rc ) {
nfc_err ( dev , " failed to initialize FW download %d \n " , rc ) ;
goto error_free_dev ;
2014-01-06 12:58:19 -08:00
}
nfc_info ( dev , " registered with nci successfully \n " ) ;
return priv ;
2014-01-08 10:52:27 -08:00
2015-10-26 10:27:39 +01:00
error_free_dev :
nci_free_device ( priv - > ndev ) ;
2014-01-08 10:52:27 -08:00
error :
kfree ( priv ) ;
return ERR_PTR ( rc ) ;
2014-01-06 12:58:19 -08:00
}
EXPORT_SYMBOL_GPL ( nfcmrvl_nci_register_dev ) ;
void nfcmrvl_nci_unregister_dev ( struct nfcmrvl_private * priv )
{
struct nci_dev * ndev = priv - > ndev ;
2015-10-26 10:27:39 +01:00
if ( priv - > ndev - > nfc_dev - > fw_download_in_progress )
nfcmrvl_fw_dnld_abort ( priv ) ;
nfcmrvl_fw_dnld_deinit ( priv ) ;
2014-01-06 12:58:19 -08:00
nci_unregister_device ( ndev ) ;
nci_free_device ( ndev ) ;
kfree ( priv ) ;
}
EXPORT_SYMBOL_GPL ( nfcmrvl_nci_unregister_dev ) ;
2015-06-11 11:25:44 +02:00
int nfcmrvl_nci_recv_frame ( struct nfcmrvl_private * priv , struct sk_buff * skb )
2014-01-06 12:58:19 -08:00
{
2015-06-11 14:00:19 +02:00
if ( priv - > config . hci_muxed ) {
2015-06-11 11:25:43 +02:00
if ( skb - > data [ 0 ] = = NFCMRVL_HCI_EVENT_CODE & &
skb - > data [ 1 ] = = NFCMRVL_HCI_NFC_EVENT_CODE ) {
/* Data packet, let's extract NCI payload */
skb_pull ( skb , NFCMRVL_HCI_EVENT_HEADER_SIZE ) ;
} else {
/* Skip this packet */
kfree_skb ( skb ) ;
return 0 ;
}
}
2015-10-26 10:27:39 +01:00
if ( priv - > ndev - > nfc_dev - > fw_download_in_progress ) {
nfcmrvl_fw_dnld_recv_frame ( priv , skb ) ;
return 0 ;
}
2015-06-11 11:25:44 +02:00
if ( test_bit ( NFCMRVL_NCI_RUNNING , & priv - > flags ) )
nci_recv_frame ( priv - > ndev , skb ) ;
else {
/* Drop this packet since nobody wants it */
kfree_skb ( skb ) ;
return 0 ;
}
2014-01-06 12:58:19 -08:00
2015-06-11 11:25:44 +02:00
return 0 ;
2014-01-06 12:58:19 -08:00
}
EXPORT_SYMBOL_GPL ( nfcmrvl_nci_recv_frame ) ;
2015-06-11 11:25:46 +02:00
void nfcmrvl_chip_reset ( struct nfcmrvl_private * priv )
{
2015-10-26 10:27:41 +01:00
/* Reset possible fault of previous session */
clear_bit ( NFCMRVL_PHY_ERROR , & priv - > flags ) ;
2015-06-11 11:25:46 +02:00
2015-06-11 14:00:19 +02:00
if ( priv - > config . reset_n_io ) {
2015-06-11 11:25:46 +02:00
nfc_info ( priv - > dev , " reset the chip \n " ) ;
2015-06-11 14:00:19 +02:00
gpio_set_value ( priv - > config . reset_n_io , 0 ) ;
2015-06-11 11:25:46 +02:00
usleep_range ( 5000 , 10000 ) ;
2015-06-11 14:00:19 +02:00
gpio_set_value ( priv - > config . reset_n_io , 1 ) ;
2015-06-11 11:25:46 +02:00
} else
nfc_info ( priv - > dev , " no reset available on this interface \n " ) ;
}
2015-10-26 10:27:39 +01:00
void nfcmrvl_chip_halt ( struct nfcmrvl_private * priv )
{
if ( priv - > config . reset_n_io )
gpio_set_value ( priv - > config . reset_n_io , 0 ) ;
}
2015-06-11 14:00:19 +02:00
# ifdef CONFIG_OF
int nfcmrvl_parse_dt ( struct device_node * node ,
struct nfcmrvl_platform_data * pdata )
{
int reset_n_io ;
reset_n_io = of_get_named_gpio ( node , " reset-n-io " , 0 ) ;
if ( reset_n_io < 0 ) {
pr_info ( " no reset-n-io config \n " ) ;
reset_n_io = 0 ;
} else if ( ! gpio_is_valid ( reset_n_io ) ) {
pr_err ( " invalid reset-n-io GPIO \n " ) ;
return reset_n_io ;
}
pdata - > reset_n_io = reset_n_io ;
if ( of_find_property ( node , " hci-muxed " , NULL ) )
pdata - > hci_muxed = 1 ;
else
pdata - > hci_muxed = 0 ;
return 0 ;
}
# else
int nfcmrvl_parse_dt ( struct device_node * node ,
struct nfcmrvl_platform_data * pdata )
{
return - ENODEV ;
}
# endif
EXPORT_SYMBOL_GPL ( nfcmrvl_parse_dt ) ;
2014-01-06 12:58:19 -08:00
MODULE_AUTHOR ( " Marvell International Ltd. " ) ;
2015-10-26 10:27:37 +01:00
MODULE_DESCRIPTION ( " Marvell NFC driver " ) ;
2014-01-06 12:58:19 -08:00
MODULE_LICENSE ( " GPL v2 " ) ;