2011-09-18 11:19:35 +03:00
/*
* The NFC Controller Interface is the communication protocol between an
* NFC Controller ( NFCC ) and a Device Host ( DH ) .
*
* Copyright ( C ) 2011 Texas Instruments , Inc .
*
* Written by Ilan Elias < ilane @ ti . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation
*
* 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 General Public License
2013-12-06 08:56:16 -08:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2011-09-18 11:19:35 +03:00
*
*/
2011-12-14 16:43:05 +01:00
# define pr_fmt(fmt) KBUILD_MODNAME ": %s: " fmt, __func__
2011-11-29 11:37:32 -08:00
2011-09-18 11:19:35 +03:00
# include <linux/types.h>
# include <linux/interrupt.h>
# include <linux/wait.h>
# include <linux/bitops.h>
# include <linux/skbuff.h>
# include "../nfc.h"
# include <net/nfc/nci.h>
# include <net/nfc/nci_core.h>
# include <linux/nfc.h>
/* Complete data exchange transaction and forward skb to nfc core */
2012-03-05 01:03:54 +01:00
void nci_data_exchange_complete ( struct nci_dev * ndev , struct sk_buff * skb ,
2011-09-18 11:19:35 +03:00
int err )
{
data_exchange_cb_t cb = ndev - > data_exchange_cb ;
void * cb_context = ndev - > data_exchange_cb_context ;
2011-11-29 11:37:35 -08:00
pr_debug ( " len %d, err %d \n " , skb ? skb - > len : 0 , err ) ;
2011-09-18 11:19:35 +03:00
2012-01-17 12:03:50 +02:00
/* data exchange is complete, stop the data timer */
del_timer_sync ( & ndev - > data_timer ) ;
clear_bit ( NCI_DATA_EXCHANGE_TO , & ndev - > flags ) ;
2011-09-18 11:19:35 +03:00
if ( cb ) {
ndev - > data_exchange_cb = NULL ;
2012-05-07 12:31:24 +02:00
ndev - > data_exchange_cb_context = NULL ;
2011-09-18 11:19:35 +03:00
/* forward skb to nfc core */
cb ( cb_context , skb , err ) ;
} else if ( skb ) {
2011-11-29 11:37:32 -08:00
pr_err ( " no rx callback, dropping rx data... \n " ) ;
2011-09-18 11:19:35 +03:00
/* no waiting callback, free skb */
kfree_skb ( skb ) ;
}
2011-09-22 11:36:19 +03:00
clear_bit ( NCI_DATA_EXCHANGE , & ndev - > flags ) ;
2011-09-18 11:19:35 +03:00
}
/* ----------------- NCI TX Data ----------------- */
static inline void nci_push_data_hdr ( struct nci_dev * ndev ,
2012-03-05 01:03:54 +01:00
__u8 conn_id ,
struct sk_buff * skb ,
__u8 pbf )
2011-09-18 11:19:35 +03:00
{
struct nci_data_hdr * hdr ;
int plen = skb - > len ;
hdr = ( struct nci_data_hdr * ) skb_push ( skb , NCI_DATA_HDR_SIZE ) ;
hdr - > conn_id = conn_id ;
hdr - > rfu = 0 ;
hdr - > plen = plen ;
nci_mt_set ( ( __u8 * ) hdr , NCI_MT_DATA_PKT ) ;
nci_pbf_set ( ( __u8 * ) hdr , pbf ) ;
}
static int nci_queue_tx_data_frags ( struct nci_dev * ndev ,
2012-03-05 01:03:54 +01:00
__u8 conn_id ,
struct sk_buff * skb ) {
2011-09-18 11:19:35 +03:00
int total_len = skb - > len ;
unsigned char * data = skb - > data ;
unsigned long flags ;
struct sk_buff_head frags_q ;
struct sk_buff * skb_frag ;
int frag_len ;
int rc = 0 ;
2011-11-29 11:37:35 -08:00
pr_debug ( " conn_id 0x%x, total_len %d \n " , conn_id , total_len ) ;
2011-09-18 11:19:35 +03:00
__skb_queue_head_init ( & frags_q ) ;
while ( total_len ) {
2011-11-09 12:09:14 +02:00
frag_len =
min_t ( int , total_len , ndev - > max_data_pkt_payload_size ) ;
2011-09-18 11:19:35 +03:00
skb_frag = nci_skb_alloc ( ndev ,
2012-03-05 01:03:54 +01:00
( NCI_DATA_HDR_SIZE + frag_len ) ,
GFP_KERNEL ) ;
2011-09-18 11:19:35 +03:00
if ( skb_frag = = NULL ) {
rc = - ENOMEM ;
goto free_exit ;
}
skb_reserve ( skb_frag , NCI_DATA_HDR_SIZE ) ;
/* first, copy the data */
memcpy ( skb_put ( skb_frag , frag_len ) , data , frag_len ) ;
/* second, set the header */
nci_push_data_hdr ( ndev , conn_id , skb_frag ,
2012-03-05 01:03:54 +01:00
( ( total_len = = frag_len ) ?
( NCI_PBF_LAST ) : ( NCI_PBF_CONT ) ) ) ;
2011-09-18 11:19:35 +03:00
__skb_queue_tail ( & frags_q , skb_frag ) ;
data + = frag_len ;
total_len - = frag_len ;
2011-11-29 11:37:33 -08:00
pr_debug ( " frag_len %d, remaining total_len %d \n " ,
frag_len , total_len ) ;
2011-09-18 11:19:35 +03:00
}
/* queue all fragments atomically */
spin_lock_irqsave ( & ndev - > tx_q . lock , flags ) ;
while ( ( skb_frag = __skb_dequeue ( & frags_q ) ) ! = NULL )
__skb_queue_tail ( & ndev - > tx_q , skb_frag ) ;
spin_unlock_irqrestore ( & ndev - > tx_q . lock , flags ) ;
/* free the original skb */
kfree_skb ( skb ) ;
goto exit ;
free_exit :
while ( ( skb_frag = __skb_dequeue ( & frags_q ) ) ! = NULL )
kfree_skb ( skb_frag ) ;
exit :
return rc ;
}
/* Send NCI data */
int nci_send_data ( struct nci_dev * ndev , __u8 conn_id , struct sk_buff * skb )
{
int rc = 0 ;
2011-11-29 11:37:35 -08:00
pr_debug ( " conn_id 0x%x, plen %d \n " , conn_id , skb - > len ) ;
2011-09-18 11:19:35 +03:00
/* check if the packet need to be fragmented */
2011-11-09 12:09:14 +02:00
if ( skb - > len < = ndev - > max_data_pkt_payload_size ) {
2011-09-18 11:19:35 +03:00
/* no need to fragment packet */
nci_push_data_hdr ( ndev , conn_id , skb , NCI_PBF_LAST ) ;
skb_queue_tail ( & ndev - > tx_q , skb ) ;
} else {
/* fragment packet and queue the fragments */
rc = nci_queue_tx_data_frags ( ndev , conn_id , skb ) ;
if ( rc ) {
2011-11-29 11:37:32 -08:00
pr_err ( " failed to fragment tx data packet \n " ) ;
2011-09-18 11:19:35 +03:00
goto free_exit ;
}
}
queue_work ( ndev - > tx_wq , & ndev - > tx_work ) ;
goto exit ;
free_exit :
kfree_skb ( skb ) ;
exit :
return rc ;
}
/* ----------------- NCI RX Data ----------------- */
static void nci_add_rx_data_frag ( struct nci_dev * ndev ,
2012-03-05 01:03:54 +01:00
struct sk_buff * skb ,
__u8 pbf )
2011-09-18 11:19:35 +03:00
{
int reassembly_len ;
int err = 0 ;
if ( ndev - > rx_data_reassembly ) {
reassembly_len = ndev - > rx_data_reassembly - > len ;
/* first, make enough room for the already accumulated data */
if ( skb_cow_head ( skb , reassembly_len ) ) {
2011-11-29 11:37:32 -08:00
pr_err ( " error adding room for accumulated rx data \n " ) ;
2011-09-18 11:19:35 +03:00
kfree_skb ( skb ) ;
2012-05-07 12:31:24 +02:00
skb = NULL ;
2011-09-18 11:19:35 +03:00
kfree_skb ( ndev - > rx_data_reassembly ) ;
2012-05-07 12:31:24 +02:00
ndev - > rx_data_reassembly = NULL ;
2011-09-18 11:19:35 +03:00
err = - ENOMEM ;
goto exit ;
}
/* second, combine the two fragments */
memcpy ( skb_push ( skb , reassembly_len ) ,
2012-03-05 01:03:54 +01:00
ndev - > rx_data_reassembly - > data ,
reassembly_len ) ;
2011-09-18 11:19:35 +03:00
/* third, free old reassembly */
kfree_skb ( ndev - > rx_data_reassembly ) ;
2012-05-07 12:31:24 +02:00
ndev - > rx_data_reassembly = NULL ;
2011-09-18 11:19:35 +03:00
}
if ( pbf = = NCI_PBF_CONT ) {
/* need to wait for next fragment, store skb and exit */
ndev - > rx_data_reassembly = skb ;
return ;
}
exit :
nci_data_exchange_complete ( ndev , skb , err ) ;
}
/* Rx Data packet */
void nci_rx_data_packet ( struct nci_dev * ndev , struct sk_buff * skb )
{
__u8 pbf = nci_pbf ( skb - > data ) ;
2011-11-29 11:37:35 -08:00
pr_debug ( " len %d \n " , skb - > len ) ;
2011-09-18 11:19:35 +03:00
2011-11-29 11:37:33 -08:00
pr_debug ( " NCI RX: MT=data, PBF=%d, conn_id=%d, plen=%d \n " ,
nci_pbf ( skb - > data ) ,
nci_conn_id ( skb - > data ) ,
nci_plen ( skb - > data ) ) ;
2011-09-18 11:19:35 +03:00
/* strip the nci data header */
skb_pull ( skb , NCI_DATA_HDR_SIZE ) ;
2014-07-22 19:48:40 +02:00
if ( ndev - > target_active_prot = = NFC_PROTO_MIFARE | |
ndev - > target_active_prot = = NFC_PROTO_JEWEL | |
ndev - > target_active_prot = = NFC_PROTO_FELICA | |
ndev - > target_active_prot = = NFC_PROTO_ISO15693 ) {
2011-09-18 11:19:35 +03:00
/* frame I/F => remove the status byte */
2014-07-22 19:48:40 +02:00
pr_debug ( " frame I/F => remove the status byte \n " ) ;
2011-09-18 11:19:35 +03:00
skb_trim ( skb , ( skb - > len - 1 ) ) ;
}
nci_add_rx_data_frag ( ndev , skb , pbf ) ;
}