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 >
*
* Acknowledgements :
* This file is based on hci_event . c , which was written
* by Maxim Krasnyansky .
*
* 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
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
*/
2011-11-29 11:37:32 -08:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2011-09-18 11:19:35 +03:00
# include <linux/types.h>
# include <linux/interrupt.h>
# include <linux/bitops.h>
# include <linux/skbuff.h>
# include "../nfc.h"
# include <net/nfc/nci.h>
# include <net/nfc/nci_core.h>
/* Handle NCI Response packets */
static void nci_core_reset_rsp_packet ( struct nci_dev * ndev , struct sk_buff * skb )
{
struct nci_core_reset_rsp * rsp = ( void * ) skb - > data ;
2011-11-29 11:37:33 -08:00
pr_debug ( " entry, status 0x%x \n " , rsp - > status ) ;
2011-09-18 11:19:35 +03:00
2011-11-09 12:09:14 +02:00
if ( rsp - > status = = NCI_STATUS_OK ) {
2011-09-18 11:19:35 +03:00
ndev - > nci_ver = rsp - > nci_ver ;
2011-11-29 11:37:33 -08:00
pr_debug ( " nci_ver 0x%x, config_status 0x%x \n " ,
rsp - > nci_ver , rsp - > config_status ) ;
2011-11-09 12:09:14 +02:00
}
2011-09-18 11:19:35 +03:00
nci_req_complete ( ndev , rsp - > status ) ;
}
static void nci_core_init_rsp_packet ( struct nci_dev * ndev , struct sk_buff * skb )
{
struct nci_core_init_rsp_1 * rsp_1 = ( void * ) skb - > data ;
struct nci_core_init_rsp_2 * rsp_2 ;
2011-11-29 11:37:33 -08:00
pr_debug ( " entry, status 0x%x \n " , rsp_1 - > status ) ;
2011-09-18 11:19:35 +03:00
if ( rsp_1 - > status ! = NCI_STATUS_OK )
2011-11-09 12:09:14 +02:00
goto exit ;
2011-09-18 11:19:35 +03:00
ndev - > nfcc_features = __le32_to_cpu ( rsp_1 - > nfcc_features ) ;
ndev - > num_supported_rf_interfaces = rsp_1 - > num_supported_rf_interfaces ;
if ( ndev - > num_supported_rf_interfaces >
2011-11-09 12:09:14 +02:00
NCI_MAX_SUPPORTED_RF_INTERFACES ) {
2011-09-18 11:19:35 +03:00
ndev - > num_supported_rf_interfaces =
NCI_MAX_SUPPORTED_RF_INTERFACES ;
}
memcpy ( ndev - > supported_rf_interfaces ,
rsp_1 - > supported_rf_interfaces ,
ndev - > num_supported_rf_interfaces ) ;
2011-11-09 12:09:14 +02:00
rsp_2 = ( void * ) ( skb - > data + 6 + rsp_1 - > num_supported_rf_interfaces ) ;
2011-09-18 11:19:35 +03:00
ndev - > max_logical_connections =
rsp_2 - > max_logical_connections ;
ndev - > max_routing_table_size =
__le16_to_cpu ( rsp_2 - > max_routing_table_size ) ;
2011-11-09 12:09:14 +02:00
ndev - > max_ctrl_pkt_payload_len =
rsp_2 - > max_ctrl_pkt_payload_len ;
ndev - > max_size_for_large_params =
__le16_to_cpu ( rsp_2 - > max_size_for_large_params ) ;
ndev - > max_data_pkt_payload_size =
rsp_2 - > max_data_pkt_payload_size ;
ndev - > initial_num_credits =
rsp_2 - > initial_num_credits ;
ndev - > manufact_id =
rsp_2 - > manufact_id ;
ndev - > manufact_specific_info =
__le32_to_cpu ( rsp_2 - > manufact_specific_info ) ;
atomic_set ( & ndev - > credits_cnt , ndev - > initial_num_credits ) ;
2011-09-18 11:19:35 +03:00
2011-11-29 11:37:33 -08:00
pr_debug ( " nfcc_features 0x%x \n " ,
ndev - > nfcc_features ) ;
pr_debug ( " num_supported_rf_interfaces %d \n " ,
ndev - > num_supported_rf_interfaces ) ;
pr_debug ( " supported_rf_interfaces[0] 0x%x \n " ,
ndev - > supported_rf_interfaces [ 0 ] ) ;
pr_debug ( " supported_rf_interfaces[1] 0x%x \n " ,
ndev - > supported_rf_interfaces [ 1 ] ) ;
pr_debug ( " supported_rf_interfaces[2] 0x%x \n " ,
ndev - > supported_rf_interfaces [ 2 ] ) ;
pr_debug ( " supported_rf_interfaces[3] 0x%x \n " ,
ndev - > supported_rf_interfaces [ 3 ] ) ;
pr_debug ( " max_logical_connections %d \n " ,
ndev - > max_logical_connections ) ;
pr_debug ( " max_routing_table_size %d \n " ,
ndev - > max_routing_table_size ) ;
pr_debug ( " max_ctrl_pkt_payload_len %d \n " ,
ndev - > max_ctrl_pkt_payload_len ) ;
pr_debug ( " max_size_for_large_params %d \n " ,
ndev - > max_size_for_large_params ) ;
pr_debug ( " max_data_pkt_payload_size %d \n " ,
ndev - > max_data_pkt_payload_size ) ;
pr_debug ( " initial_num_credits %d \n " ,
ndev - > initial_num_credits ) ;
pr_debug ( " manufact_id 0x%x \n " ,
ndev - > manufact_id ) ;
pr_debug ( " manufact_specific_info 0x%x \n " ,
ndev - > manufact_specific_info ) ;
2011-11-09 12:09:14 +02:00
exit :
2011-09-18 11:19:35 +03:00
nci_req_complete ( ndev , rsp_1 - > status ) ;
}
static void nci_rf_disc_map_rsp_packet ( struct nci_dev * ndev ,
struct sk_buff * skb )
{
__u8 status = skb - > data [ 0 ] ;
2011-11-29 11:37:33 -08:00
pr_debug ( " entry, status 0x%x \n " , status ) ;
2011-09-18 11:19:35 +03:00
nci_req_complete ( ndev , status ) ;
}
static void nci_rf_disc_rsp_packet ( struct nci_dev * ndev , struct sk_buff * skb )
{
__u8 status = skb - > data [ 0 ] ;
2011-11-29 11:37:33 -08:00
pr_debug ( " entry, status 0x%x \n " , status ) ;
2011-09-18 11:19:35 +03:00
if ( status = = NCI_STATUS_OK )
set_bit ( NCI_DISCOVERY , & ndev - > flags ) ;
nci_req_complete ( ndev , status ) ;
}
static void nci_rf_deactivate_rsp_packet ( struct nci_dev * ndev ,
struct sk_buff * skb )
{
__u8 status = skb - > data [ 0 ] ;
2011-11-29 11:37:33 -08:00
pr_debug ( " entry, status 0x%x \n " , status ) ;
2011-09-18 11:19:35 +03:00
clear_bit ( NCI_DISCOVERY , & ndev - > flags ) ;
nci_req_complete ( ndev , status ) ;
}
void nci_rsp_packet ( struct nci_dev * ndev , struct sk_buff * skb )
{
__u16 rsp_opcode = nci_opcode ( skb - > data ) ;
/* we got a rsp, stop the cmd timer */
del_timer ( & ndev - > cmd_timer ) ;
2011-11-29 11:37:33 -08:00
pr_debug ( " NCI RX: MT=rsp, PBF=%d, GID=0x%x, OID=0x%x, plen=%d \n " ,
nci_pbf ( skb - > data ) ,
nci_opcode_gid ( rsp_opcode ) ,
nci_opcode_oid ( rsp_opcode ) ,
nci_plen ( skb - > data ) ) ;
2011-09-18 11:19:35 +03:00
/* strip the nci control header */
skb_pull ( skb , NCI_CTRL_HDR_SIZE ) ;
switch ( rsp_opcode ) {
case NCI_OP_CORE_RESET_RSP :
nci_core_reset_rsp_packet ( ndev , skb ) ;
break ;
case NCI_OP_CORE_INIT_RSP :
nci_core_init_rsp_packet ( ndev , skb ) ;
break ;
case NCI_OP_RF_DISCOVER_MAP_RSP :
nci_rf_disc_map_rsp_packet ( ndev , skb ) ;
break ;
case NCI_OP_RF_DISCOVER_RSP :
nci_rf_disc_rsp_packet ( ndev , skb ) ;
break ;
case NCI_OP_RF_DEACTIVATE_RSP :
nci_rf_deactivate_rsp_packet ( ndev , skb ) ;
break ;
default :
2011-11-29 11:37:32 -08:00
pr_err ( " unknown rsp opcode 0x%x \n " , rsp_opcode ) ;
2011-09-18 11:19:35 +03:00
break ;
}
kfree_skb ( skb ) ;
/* trigger the next cmd */
atomic_set ( & ndev - > cmd_cnt , 1 ) ;
if ( ! skb_queue_empty ( & ndev - > cmd_q ) )
queue_work ( ndev - > cmd_wq , & ndev - > cmd_work ) ;
}