2012-04-10 19:43:06 +02:00
/*
* Copyright ( C ) 2012 Intel Corporation . All rights reserved .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 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 General Public License
2013-12-06 08:56:16 -08:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2012-04-10 19:43:06 +02:00
*/
# define pr_fmt(fmt) "hci: %s: " fmt, __func__
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/nfc.h>
# include <net/nfc/nfc.h>
# include <net/nfc/hci.h>
2012-09-13 17:10:00 +02:00
# include <net/nfc/llc.h>
2012-04-10 19:43:06 +02:00
# include "hci.h"
/* Largest headroom needed for outgoing HCI commands */
# define HCI_CMDS_HEADROOM 1
2012-10-17 16:50:10 +02:00
int nfc_hci_result_to_errno ( u8 result )
2012-05-03 15:35:25 +02:00
{
switch ( result ) {
case NFC_HCI_ANY_OK :
return 0 ;
2012-10-17 16:48:21 +02:00
case NFC_HCI_ANY_E_REG_PAR_UNKNOWN :
return - EOPNOTSUPP ;
2012-05-03 15:35:25 +02:00
case NFC_HCI_ANY_E_TIMEOUT :
return - ETIME ;
default :
return - 1 ;
}
}
2012-10-17 16:50:10 +02:00
EXPORT_SYMBOL ( nfc_hci_result_to_errno ) ;
2012-05-03 15:35:25 +02:00
2015-01-27 01:18:12 +01:00
void nfc_hci_reset_pipes ( struct nfc_hci_dev * hdev )
{
int i = 0 ;
for ( i = 0 ; i < NFC_HCI_MAX_PIPES ; i + + ) {
hdev - > pipes [ i ] . gate = NFC_HCI_INVALID_GATE ;
hdev - > pipes [ i ] . dest_host = NFC_HCI_INVALID_HOST ;
}
memset ( hdev - > gate2pipe , NFC_HCI_INVALID_PIPE , sizeof ( hdev - > gate2pipe ) ) ;
}
EXPORT_SYMBOL ( nfc_hci_reset_pipes ) ;
void nfc_hci_reset_pipes_per_host ( struct nfc_hci_dev * hdev , u8 host )
{
int i = 0 ;
for ( i = 0 ; i < NFC_HCI_MAX_PIPES ; i + + ) {
if ( hdev - > pipes [ i ] . dest_host ! = host )
continue ;
hdev - > pipes [ i ] . gate = NFC_HCI_INVALID_GATE ;
hdev - > pipes [ i ] . dest_host = NFC_HCI_INVALID_HOST ;
}
}
EXPORT_SYMBOL ( nfc_hci_reset_pipes_per_host ) ;
2012-04-10 19:43:06 +02:00
static void nfc_hci_msg_tx_work ( struct work_struct * work )
{
struct nfc_hci_dev * hdev = container_of ( work , struct nfc_hci_dev ,
msg_tx_work ) ;
struct hci_msg * msg ;
struct sk_buff * skb ;
int r = 0 ;
mutex_lock ( & hdev - > msg_tx_mutex ) ;
2012-11-26 18:06:27 +01:00
if ( hdev - > shutting_down )
goto exit ;
2012-04-10 19:43:06 +02:00
if ( hdev - > cmd_pending_msg ) {
if ( timer_pending ( & hdev - > cmd_timer ) = = 0 ) {
if ( hdev - > cmd_pending_msg - > cb )
2012-09-11 10:41:41 +02:00
hdev - > cmd_pending_msg - > cb ( hdev - >
2012-04-10 19:43:06 +02:00
cmd_pending_msg - >
2012-09-11 10:41:41 +02:00
cb_context ,
NULL ,
- ETIME ) ;
2012-04-10 19:43:06 +02:00
kfree ( hdev - > cmd_pending_msg ) ;
hdev - > cmd_pending_msg = NULL ;
2012-10-17 15:23:39 +02:00
} else {
2012-04-10 19:43:06 +02:00
goto exit ;
2012-10-17 15:23:39 +02:00
}
2012-04-10 19:43:06 +02:00
}
next_msg :
if ( list_empty ( & hdev - > msg_tx_queue ) )
goto exit ;
msg = list_first_entry ( & hdev - > msg_tx_queue , struct hci_msg , msg_l ) ;
list_del ( & msg - > msg_l ) ;
pr_debug ( " msg_tx_queue has a cmd to send \n " ) ;
while ( ( skb = skb_dequeue ( & msg - > msg_frags ) ) ! = NULL ) {
2012-09-18 19:45:48 +02:00
r = nfc_llc_xmit_from_hci ( hdev - > llc , skb ) ;
2012-04-10 19:43:06 +02:00
if ( r < 0 ) {
kfree_skb ( skb ) ;
skb_queue_purge ( & msg - > msg_frags ) ;
if ( msg - > cb )
2012-09-11 10:41:41 +02:00
msg - > cb ( msg - > cb_context , NULL , r ) ;
2012-04-10 19:43:06 +02:00
kfree ( msg ) ;
break ;
}
}
if ( r )
goto next_msg ;
if ( msg - > wait_response = = false ) {
kfree ( msg ) ;
goto next_msg ;
}
hdev - > cmd_pending_msg = msg ;
mod_timer ( & hdev - > cmd_timer , jiffies +
msecs_to_jiffies ( hdev - > cmd_pending_msg - > completion_delay ) ) ;
exit :
mutex_unlock ( & hdev - > msg_tx_mutex ) ;
}
static void nfc_hci_msg_rx_work ( struct work_struct * work )
{
struct nfc_hci_dev * hdev = container_of ( work , struct nfc_hci_dev ,
msg_rx_work ) ;
struct sk_buff * skb ;
struct hcp_message * message ;
u8 pipe ;
u8 type ;
u8 instruction ;
while ( ( skb = skb_dequeue ( & hdev - > msg_rx_queue ) ) ! = NULL ) {
pipe = skb - > data [ 0 ] ;
skb_pull ( skb , NFC_HCI_HCP_PACKET_HEADER_LEN ) ;
message = ( struct hcp_message * ) skb - > data ;
type = HCP_MSG_GET_TYPE ( message - > header ) ;
instruction = HCP_MSG_GET_CMD ( message - > header ) ;
skb_pull ( skb , NFC_HCI_HCP_MESSAGE_HEADER_LEN ) ;
nfc_hci_hcp_message_rx ( hdev , pipe , type , instruction , skb ) ;
}
}
2012-05-03 15:59:37 +02:00
static void __nfc_hci_cmd_completion ( struct nfc_hci_dev * hdev , int err ,
struct sk_buff * skb )
2012-04-10 19:43:06 +02:00
{
del_timer_sync ( & hdev - > cmd_timer ) ;
if ( hdev - > cmd_pending_msg - > cb )
2012-09-11 10:41:41 +02:00
hdev - > cmd_pending_msg - > cb ( hdev - > cmd_pending_msg - > cb_context ,
skb , err ) ;
2012-04-10 19:43:06 +02:00
else
kfree_skb ( skb ) ;
kfree ( hdev - > cmd_pending_msg ) ;
hdev - > cmd_pending_msg = NULL ;
2012-10-02 16:01:31 -07:00
schedule_work ( & hdev - > msg_tx_work ) ;
2012-05-03 15:59:37 +02:00
}
void nfc_hci_resp_received ( struct nfc_hci_dev * hdev , u8 result ,
struct sk_buff * skb )
{
mutex_lock ( & hdev - > msg_tx_mutex ) ;
if ( hdev - > cmd_pending_msg = = NULL ) {
kfree_skb ( skb ) ;
goto exit ;
}
__nfc_hci_cmd_completion ( hdev , nfc_hci_result_to_errno ( result ) , skb ) ;
2012-04-10 19:43:06 +02:00
exit :
mutex_unlock ( & hdev - > msg_tx_mutex ) ;
}
void nfc_hci_cmd_received ( struct nfc_hci_dev * hdev , u8 pipe , u8 cmd ,
struct sk_buff * skb )
{
2015-01-27 01:18:12 +01:00
u8 gate = hdev - > pipes [ pipe ] . gate ;
2015-01-27 01:18:14 +01:00
u8 status = NFC_HCI_ANY_OK ;
struct hci_create_pipe_resp * create_info ;
struct hci_delete_pipe_noti * delete_info ;
struct hci_all_pipe_cleared_noti * cleared_info ;
2014-11-13 00:30:42 +01:00
pr_debug ( " from gate %x pipe %x cmd %x \n " , gate , pipe , cmd ) ;
switch ( cmd ) {
case NFC_HCI_ADM_NOTIFY_PIPE_CREATED :
if ( skb - > len ! = 5 ) {
2015-01-27 01:18:14 +01:00
status = NFC_HCI_ANY_E_NOK ;
goto exit ;
2014-11-13 00:30:42 +01:00
}
2015-01-27 01:18:14 +01:00
create_info = ( struct hci_create_pipe_resp * ) skb - > data ;
2014-11-13 00:30:42 +01:00
2015-01-27 01:18:14 +01:00
/* Save the new created pipe and bind with local gate,
2014-11-13 00:30:42 +01:00
* the description for skb - > data [ 3 ] is destination gate id
* but since we received this cmd from host controller , we
* are the destination and it is our local gate
*/
2015-01-27 01:18:14 +01:00
hdev - > gate2pipe [ create_info - > dest_gate ] = create_info - > pipe ;
hdev - > pipes [ create_info - > pipe ] . gate = create_info - > dest_gate ;
hdev - > pipes [ create_info - > pipe ] . dest_host =
create_info - > src_host ;
2014-11-13 00:30:42 +01:00
break ;
case NFC_HCI_ANY_OPEN_PIPE :
2015-01-27 01:18:14 +01:00
if ( gate = = NFC_HCI_INVALID_GATE ) {
status = NFC_HCI_ANY_E_NOK ;
goto exit ;
}
break ;
case NFC_HCI_ADM_NOTIFY_PIPE_DELETED :
if ( skb - > len ! = 1 ) {
status = NFC_HCI_ANY_E_NOK ;
goto exit ;
}
delete_info = ( struct hci_delete_pipe_noti * ) skb - > data ;
hdev - > pipes [ delete_info - > pipe ] . gate = NFC_HCI_INVALID_GATE ;
hdev - > pipes [ delete_info - > pipe ] . dest_host = NFC_HCI_INVALID_HOST ;
2014-11-13 00:30:42 +01:00
break ;
2014-11-13 00:30:43 +01:00
case NFC_HCI_ADM_NOTIFY_ALL_PIPE_CLEARED :
2015-01-27 01:18:14 +01:00
if ( skb - > len ! = 1 ) {
status = NFC_HCI_ANY_E_NOK ;
goto exit ;
}
cleared_info = ( struct hci_all_pipe_cleared_noti * ) skb - > data ;
2015-01-27 01:18:13 +01:00
2015-01-27 01:18:14 +01:00
nfc_hci_reset_pipes_per_host ( hdev , cleared_info - > host ) ;
2014-11-13 00:30:43 +01:00
break ;
2014-11-13 00:30:42 +01:00
default :
pr_info ( " Discarded unknown cmd %x to gate %x \n " , cmd , gate ) ;
break ;
}
2015-01-27 01:18:15 +01:00
if ( hdev - > ops - > cmd_received )
hdev - > ops - > cmd_received ( hdev , pipe , cmd , skb ) ;
2015-01-27 01:18:14 +01:00
exit :
nfc_hci_hcp_message_tx ( hdev , pipe , NFC_HCI_HCP_RESPONSE ,
status , NULL , 0 , NULL , NULL , 0 ) ;
2012-04-10 19:43:06 +02:00
kfree_skb ( skb ) ;
}
2012-10-23 11:37:43 +02:00
u32 nfc_hci_sak_to_protocol ( u8 sak )
2012-04-10 19:43:06 +02:00
{
switch ( NFC_HCI_TYPE_A_SEL_PROT ( sak ) ) {
case NFC_HCI_TYPE_A_SEL_PROT_MIFARE :
return NFC_PROTO_MIFARE_MASK ;
case NFC_HCI_TYPE_A_SEL_PROT_ISO14443 :
return NFC_PROTO_ISO14443_MASK ;
case NFC_HCI_TYPE_A_SEL_PROT_DEP :
return NFC_PROTO_NFC_DEP_MASK ;
case NFC_HCI_TYPE_A_SEL_PROT_ISO14443_DEP :
return NFC_PROTO_ISO14443_MASK | NFC_PROTO_NFC_DEP_MASK ;
default :
return 0xffffffff ;
}
}
2012-10-23 11:37:43 +02:00
EXPORT_SYMBOL ( nfc_hci_sak_to_protocol ) ;
2012-04-10 19:43:06 +02:00
2012-09-27 17:32:55 +08:00
int nfc_hci_target_discovered ( struct nfc_hci_dev * hdev , u8 gate )
2012-04-10 19:43:06 +02:00
{
struct nfc_target * targets ;
struct sk_buff * atqa_skb = NULL ;
struct sk_buff * sak_skb = NULL ;
2012-07-12 20:27:54 +02:00
struct sk_buff * uid_skb = NULL ;
2012-04-10 19:43:06 +02:00
int r ;
pr_debug ( " from gate %d \n " , gate ) ;
targets = kzalloc ( sizeof ( struct nfc_target ) , GFP_KERNEL ) ;
if ( targets = = NULL )
return - ENOMEM ;
switch ( gate ) {
case NFC_HCI_RF_READER_A_GATE :
r = nfc_hci_get_param ( hdev , NFC_HCI_RF_READER_A_GATE ,
NFC_HCI_RF_READER_A_ATQA , & atqa_skb ) ;
if ( r < 0 )
goto exit ;
r = nfc_hci_get_param ( hdev , NFC_HCI_RF_READER_A_GATE ,
NFC_HCI_RF_READER_A_SAK , & sak_skb ) ;
if ( r < 0 )
goto exit ;
if ( atqa_skb - > len ! = 2 | | sak_skb - > len ! = 1 ) {
r = - EPROTO ;
goto exit ;
}
targets - > supported_protocols =
nfc_hci_sak_to_protocol ( sak_skb - > data [ 0 ] ) ;
if ( targets - > supported_protocols = = 0xffffffff ) {
r = - EPROTO ;
goto exit ;
}
2014-04-01 00:34:01 +02:00
targets - > sens_res = be16_to_cpu ( * ( __be16 * ) atqa_skb - > data ) ;
2012-04-10 19:43:06 +02:00
targets - > sel_res = sak_skb - > data [ 0 ] ;
2012-07-12 20:27:54 +02:00
r = nfc_hci_get_param ( hdev , NFC_HCI_RF_READER_A_GATE ,
NFC_HCI_RF_READER_A_UID , & uid_skb ) ;
if ( r < 0 )
goto exit ;
if ( uid_skb - > len = = 0 | | uid_skb - > len > NFC_NFCID1_MAXSIZE ) {
r = - EPROTO ;
goto exit ;
}
memcpy ( targets - > nfcid1 , uid_skb - > data , uid_skb - > len ) ;
targets - > nfcid1_len = uid_skb - > len ;
2012-04-10 19:43:06 +02:00
if ( hdev - > ops - > complete_target_discovered ) {
r = hdev - > ops - > complete_target_discovered ( hdev , gate ,
targets ) ;
if ( r < 0 )
goto exit ;
}
break ;
case NFC_HCI_RF_READER_B_GATE :
2012-07-04 00:14:04 +02:00
targets - > supported_protocols = NFC_PROTO_ISO14443_B_MASK ;
2012-04-10 19:43:06 +02:00
break ;
default :
if ( hdev - > ops - > target_from_gate )
r = hdev - > ops - > target_from_gate ( hdev , gate , targets ) ;
else
r = - EPROTO ;
if ( r < 0 )
goto exit ;
if ( hdev - > ops - > complete_target_discovered ) {
r = hdev - > ops - > complete_target_discovered ( hdev , gate ,
targets ) ;
if ( r < 0 )
goto exit ;
}
break ;
}
2012-09-27 17:32:56 +08:00
/* if driver set the new gate, we will skip the old one */
if ( targets - > hci_reader_gate = = 0x00 )
targets - > hci_reader_gate = gate ;
2012-04-10 19:43:06 +02:00
r = nfc_targets_found ( hdev - > ndev , targets , 1 ) ;
exit :
kfree ( targets ) ;
kfree_skb ( atqa_skb ) ;
kfree_skb ( sak_skb ) ;
2012-07-12 20:27:54 +02:00
kfree_skb ( uid_skb ) ;
2012-04-10 19:43:06 +02:00
return r ;
}
2012-09-27 17:32:55 +08:00
EXPORT_SYMBOL ( nfc_hci_target_discovered ) ;
2012-04-10 19:43:06 +02:00
void nfc_hci_event_received ( struct nfc_hci_dev * hdev , u8 pipe , u8 event ,
struct sk_buff * skb )
{
int r = 0 ;
2015-01-27 01:18:12 +01:00
u8 gate = hdev - > pipes [ pipe ] . gate ;
2012-10-17 16:49:12 +02:00
2015-01-27 01:18:12 +01:00
if ( gate = = NFC_HCI_INVALID_GATE ) {
2012-10-17 16:49:12 +02:00
pr_err ( " Discarded event %x to unopened pipe %x \n " , event , pipe ) ;
goto exit ;
}
2012-04-10 19:43:06 +02:00
2012-12-04 16:43:24 +01:00
if ( hdev - > ops - > event_received ) {
2015-01-27 01:18:11 +01:00
r = hdev - > ops - > event_received ( hdev , pipe , event , skb ) ;
2012-12-04 16:43:24 +01:00
if ( r < = 0 )
goto exit_noskb ;
}
2012-04-10 19:43:06 +02:00
switch ( event ) {
case NFC_HCI_EVT_TARGET_DISCOVERED :
if ( skb - > len < 1 ) { /* no status data? */
r = - EPROTO ;
goto exit ;
}
if ( skb - > data [ 0 ] = = 3 ) {
/* TODO: Multiple targets in field, none activated
* poll is supposedly stopped , but there is no
* single target to activate , so nothing to report
* up .
* if we need to restart poll , we must save the
* protocols from the initial poll and reuse here .
*/
}
if ( skb - > data [ 0 ] ! = 0 ) {
r = - EPROTO ;
goto exit ;
}
2012-10-17 16:49:12 +02:00
r = nfc_hci_target_discovered ( hdev , gate ) ;
2012-04-10 19:43:06 +02:00
break ;
default :
2012-12-04 16:43:24 +01:00
pr_info ( " Discarded unknown event %x to gate %x \n " , event , gate ) ;
r = - EINVAL ;
2012-04-10 19:43:06 +02:00
break ;
}
exit :
kfree_skb ( skb ) ;
2012-11-28 15:48:44 +01:00
exit_noskb :
2013-11-13 01:00:07 +01:00
if ( r )
nfc_hci_driver_failure ( hdev , r ) ;
2012-04-10 19:43:06 +02:00
}
2017-10-11 16:03:44 +05:30
static void nfc_hci_cmd_timeout ( struct timer_list * t )
2012-04-10 19:43:06 +02:00
{
2017-10-11 16:03:44 +05:30
struct nfc_hci_dev * hdev = from_timer ( hdev , t , cmd_timer ) ;
2012-04-10 19:43:06 +02:00
2012-10-02 16:01:31 -07:00
schedule_work ( & hdev - > msg_tx_work ) ;
2012-04-10 19:43:06 +02:00
}
static int hci_dev_connect_gates ( struct nfc_hci_dev * hdev , u8 gate_count ,
2012-06-05 14:42:11 +02:00
struct nfc_hci_gate * gates )
2012-04-10 19:43:06 +02:00
{
int r ;
while ( gate_count - - ) {
2012-06-05 14:42:11 +02:00
r = nfc_hci_connect_gate ( hdev , NFC_HCI_HOST_CONTROLLER_ID ,
gates - > gate , gates - > pipe ) ;
2012-04-10 19:43:06 +02:00
if ( r < 0 )
return r ;
2012-06-05 14:42:11 +02:00
gates + + ;
2012-04-10 19:43:06 +02:00
}
return 0 ;
}
static int hci_dev_session_init ( struct nfc_hci_dev * hdev )
{
struct sk_buff * skb = NULL ;
int r ;
2012-06-05 14:42:11 +02:00
if ( hdev - > init_data . gates [ 0 ] . gate ! = NFC_HCI_ADMIN_GATE )
return - EPROTO ;
2012-04-10 19:43:06 +02:00
r = nfc_hci_connect_gate ( hdev , NFC_HCI_HOST_CONTROLLER_ID ,
2012-06-05 14:42:11 +02:00
hdev - > init_data . gates [ 0 ] . gate ,
hdev - > init_data . gates [ 0 ] . pipe ) ;
2012-04-10 19:43:06 +02:00
if ( r < 0 )
goto exit ;
r = nfc_hci_get_param ( hdev , NFC_HCI_ADMIN_GATE ,
NFC_HCI_ADMIN_SESSION_IDENTITY , & skb ) ;
if ( r < 0 )
goto disconnect_all ;
2014-03-25 06:51:49 +01:00
if ( skb - > len & & skb - > len = = strlen ( hdev - > init_data . session_id ) & &
( memcmp ( hdev - > init_data . session_id , skb - > data ,
skb - > len ) = = 0 ) & & hdev - > ops - > load_session ) {
/* Restore gate<->pipe table from some proprietary location. */
2012-04-10 19:43:06 +02:00
2014-03-25 06:51:49 +01:00
r = hdev - > ops - > load_session ( hdev ) ;
2012-04-10 19:43:06 +02:00
2014-03-25 06:51:49 +01:00
if ( r < 0 )
goto disconnect_all ;
} else {
2012-04-10 19:43:06 +02:00
2014-03-25 06:51:49 +01:00
r = nfc_hci_disconnect_all_gates ( hdev ) ;
if ( r < 0 )
goto exit ;
2012-04-10 19:43:06 +02:00
2014-03-25 06:51:49 +01:00
r = hci_dev_connect_gates ( hdev , hdev - > init_data . gate_count ,
hdev - > init_data . gates ) ;
if ( r < 0 )
goto disconnect_all ;
r = nfc_hci_set_param ( hdev , NFC_HCI_ADMIN_GATE ,
NFC_HCI_ADMIN_SESSION_IDENTITY ,
hdev - > init_data . session_id ,
strlen ( hdev - > init_data . session_id ) ) ;
}
2012-04-10 19:43:06 +02:00
if ( r = = 0 )
goto exit ;
disconnect_all :
nfc_hci_disconnect_all_gates ( hdev ) ;
exit :
2012-08-28 21:02:40 +08:00
kfree_skb ( skb ) ;
2012-04-10 19:43:06 +02:00
return r ;
}
static int hci_dev_version ( struct nfc_hci_dev * hdev )
{
int r ;
struct sk_buff * skb ;
r = nfc_hci_get_param ( hdev , NFC_HCI_ID_MGMT_GATE ,
NFC_HCI_ID_MGMT_VERSION_SW , & skb ) ;
2012-10-17 16:48:21 +02:00
if ( r = = - EOPNOTSUPP ) {
pr_info ( " Software/Hardware info not available \n " ) ;
return 0 ;
}
2012-04-10 19:43:06 +02:00
if ( r < 0 )
return r ;
if ( skb - > len ! = 3 ) {
kfree_skb ( skb ) ;
return - EINVAL ;
}
hdev - > sw_romlib = ( skb - > data [ 0 ] & 0xf0 ) > > 4 ;
hdev - > sw_patch = skb - > data [ 0 ] & 0x0f ;
hdev - > sw_flashlib_major = skb - > data [ 1 ] ;
hdev - > sw_flashlib_minor = skb - > data [ 2 ] ;
kfree_skb ( skb ) ;
r = nfc_hci_get_param ( hdev , NFC_HCI_ID_MGMT_GATE ,
NFC_HCI_ID_MGMT_VERSION_HW , & skb ) ;
if ( r < 0 )
return r ;
if ( skb - > len ! = 3 ) {
kfree_skb ( skb ) ;
return - EINVAL ;
}
hdev - > hw_derivative = ( skb - > data [ 0 ] & 0xe0 ) > > 5 ;
hdev - > hw_version = skb - > data [ 0 ] & 0x1f ;
hdev - > hw_mpw = ( skb - > data [ 1 ] & 0xc0 ) > > 6 ;
hdev - > hw_software = skb - > data [ 1 ] & 0x3f ;
hdev - > hw_bsid = skb - > data [ 2 ] ;
kfree_skb ( skb ) ;
pr_info ( " SOFTWARE INFO: \n " ) ;
pr_info ( " RomLib : %d \n " , hdev - > sw_romlib ) ;
pr_info ( " Patch : %d \n " , hdev - > sw_patch ) ;
pr_info ( " FlashLib Major : %d \n " , hdev - > sw_flashlib_major ) ;
pr_info ( " FlashLib Minor : %d \n " , hdev - > sw_flashlib_minor ) ;
pr_info ( " HARDWARE INFO: \n " ) ;
pr_info ( " Derivative : %d \n " , hdev - > hw_derivative ) ;
pr_info ( " HW Version : %d \n " , hdev - > hw_version ) ;
pr_info ( " #MPW : %d \n " , hdev - > hw_mpw ) ;
pr_info ( " Software : %d \n " , hdev - > hw_software ) ;
pr_info ( " BSID Version : %d \n " , hdev - > hw_bsid ) ;
return 0 ;
}
static int hci_dev_up ( struct nfc_dev * nfc_dev )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
int r = 0 ;
if ( hdev - > ops - > open ) {
r = hdev - > ops - > open ( hdev ) ;
if ( r < 0 )
return r ;
}
2012-09-18 19:45:48 +02:00
r = nfc_llc_start ( hdev - > llc ) ;
if ( r < 0 )
goto exit_close ;
2012-04-10 19:43:06 +02:00
r = hci_dev_session_init ( hdev ) ;
if ( r < 0 )
2012-09-18 19:45:48 +02:00
goto exit_llc ;
2012-04-10 19:43:06 +02:00
r = nfc_hci_send_event ( hdev , NFC_HCI_RF_READER_A_GATE ,
NFC_HCI_EVT_END_OPERATION , NULL , 0 ) ;
if ( r < 0 )
2012-09-18 19:45:48 +02:00
goto exit_llc ;
2012-04-10 19:43:06 +02:00
if ( hdev - > ops - > hci_ready ) {
r = hdev - > ops - > hci_ready ( hdev ) ;
if ( r < 0 )
2012-09-18 19:45:48 +02:00
goto exit_llc ;
2012-04-10 19:43:06 +02:00
}
r = hci_dev_version ( hdev ) ;
if ( r < 0 )
2012-09-18 19:45:48 +02:00
goto exit_llc ;
return 0 ;
exit_llc :
nfc_llc_stop ( hdev - > llc ) ;
exit_close :
if ( hdev - > ops - > close )
hdev - > ops - > close ( hdev ) ;
2012-04-10 19:43:06 +02:00
return r ;
}
static int hci_dev_down ( struct nfc_dev * nfc_dev )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2012-09-18 19:45:48 +02:00
nfc_llc_stop ( hdev - > llc ) ;
2012-04-10 19:43:06 +02:00
if ( hdev - > ops - > close )
hdev - > ops - > close ( hdev ) ;
2015-01-27 01:18:12 +01:00
nfc_hci_reset_pipes ( hdev ) ;
2012-04-10 19:43:06 +02:00
return 0 ;
}
2012-05-15 15:57:06 +02:00
static int hci_start_poll ( struct nfc_dev * nfc_dev ,
u32 im_protocols , u32 tm_protocols )
2012-04-10 19:43:06 +02:00
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
if ( hdev - > ops - > start_poll )
2012-05-15 15:57:06 +02:00
return hdev - > ops - > start_poll ( hdev , im_protocols , tm_protocols ) ;
2012-04-10 19:43:06 +02:00
else
2012-05-07 12:31:31 +02:00
return nfc_hci_send_event ( hdev , NFC_HCI_RF_READER_A_GATE ,
2012-10-17 15:23:39 +02:00
NFC_HCI_EVT_READER_REQUESTED ,
NULL , 0 ) ;
2012-04-10 19:43:06 +02:00
}
static void hci_stop_poll ( struct nfc_dev * nfc_dev )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2014-05-20 22:21:57 +02:00
if ( hdev - > ops - > stop_poll )
hdev - > ops - > stop_poll ( hdev ) ;
else
nfc_hci_send_event ( hdev , NFC_HCI_RF_READER_A_GATE ,
NFC_HCI_EVT_END_OPERATION , NULL , 0 ) ;
2012-04-10 19:43:06 +02:00
}
2012-09-27 17:32:57 +08:00
static int hci_dep_link_up ( struct nfc_dev * nfc_dev , struct nfc_target * target ,
__u8 comm_mode , __u8 * gb , size_t gb_len )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2013-05-25 01:21:21 +02:00
if ( ! hdev - > ops - > dep_link_up )
return 0 ;
2012-09-27 17:32:57 +08:00
2013-05-25 01:21:21 +02:00
return hdev - > ops - > dep_link_up ( hdev , target , comm_mode ,
gb , gb_len ) ;
2012-09-27 17:32:57 +08:00
}
static int hci_dep_link_down ( struct nfc_dev * nfc_dev )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2013-05-25 01:21:21 +02:00
if ( ! hdev - > ops - > dep_link_down )
return 0 ;
2012-09-27 17:32:57 +08:00
2013-05-25 01:21:21 +02:00
return hdev - > ops - > dep_link_down ( hdev ) ;
2012-09-27 17:32:57 +08:00
}
2012-05-07 12:31:13 +02:00
static int hci_activate_target ( struct nfc_dev * nfc_dev ,
struct nfc_target * target , u32 protocol )
2012-04-10 19:43:06 +02:00
{
return 0 ;
}
2012-05-07 12:31:13 +02:00
static void hci_deactivate_target ( struct nfc_dev * nfc_dev ,
2015-10-25 22:54:43 +01:00
struct nfc_target * target ,
u8 mode )
2012-04-10 19:43:06 +02:00
{
}
2012-09-11 10:43:50 +02:00
# define HCI_CB_TYPE_TRANSCEIVE 1
static void hci_transceive_cb ( void * context , struct sk_buff * skb , int err )
{
struct nfc_hci_dev * hdev = context ;
switch ( hdev - > async_cb_type ) {
case HCI_CB_TYPE_TRANSCEIVE :
/*
* TODO : Check RF Error indicator to make sure data is valid .
* It seems that HCI cmd can complete without error , but data
* can be invalid if an RF error occured ? Ignore for now .
*/
if ( err = = 0 )
skb_trim ( skb , skb - > len - 1 ) ; /* RF Err ind */
hdev - > async_cb ( hdev - > async_cb_context , skb , err ) ;
break ;
default :
if ( err = = 0 )
kfree_skb ( skb ) ;
break ;
}
}
2012-05-16 15:55:48 +02:00
static int hci_transceive ( struct nfc_dev * nfc_dev , struct nfc_target * target ,
struct sk_buff * skb , data_exchange_cb_t cb ,
void * cb_context )
2012-04-10 19:43:06 +02:00
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
int r ;
2012-05-07 12:31:13 +02:00
pr_debug ( " target_idx=%d \n " , target - > idx ) ;
2012-04-10 19:43:06 +02:00
switch ( target - > hci_reader_gate ) {
case NFC_HCI_RF_READER_A_GATE :
case NFC_HCI_RF_READER_B_GATE :
2012-09-27 17:32:58 +08:00
if ( hdev - > ops - > im_transceive ) {
r = hdev - > ops - > im_transceive ( hdev , target , skb , cb ,
2012-09-11 10:43:50 +02:00
cb_context ) ;
2012-04-10 19:43:06 +02:00
if ( r < = 0 ) /* handled */
break ;
}
networking: make skb_push & __skb_push return void pointers
It seems like a historic accident that these return unsigned char *,
and in many places that means casts are required, more often than not.
Make these functions return void * and remove all the casts across
the tree, adding a (u8 *) cast only where the unsigned char pointer
was used directly, all done with the following spatch:
@@
expression SKB, LEN;
typedef u8;
identifier fn = { skb_push, __skb_push, skb_push_rcsum };
@@
- *(fn(SKB, LEN))
+ *(u8 *)fn(SKB, LEN)
@@
expression E, SKB, LEN;
identifier fn = { skb_push, __skb_push, skb_push_rcsum };
type T;
@@
- E = ((T *)(fn(SKB, LEN)))
+ E = fn(SKB, LEN)
@@
expression SKB, LEN;
identifier fn = { skb_push, __skb_push, skb_push_rcsum };
@@
- fn(SKB, LEN)[0]
+ *(u8 *)fn(SKB, LEN)
Note that the last part there converts from push(...)[0] to the
more idiomatic *(u8 *)push(...).
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2017-06-16 14:29:23 +02:00
* ( u8 * ) skb_push ( skb , 1 ) = 0 ; /* CTR, see spec:10.2.2.1 */
2012-09-11 10:43:50 +02:00
hdev - > async_cb_type = HCI_CB_TYPE_TRANSCEIVE ;
hdev - > async_cb = cb ;
hdev - > async_cb_context = cb_context ;
r = nfc_hci_send_cmd_async ( hdev , target - > hci_reader_gate ,
NFC_HCI_WR_XCHG_DATA , skb - > data ,
skb - > len , hci_transceive_cb , hdev ) ;
2012-04-10 19:43:06 +02:00
break ;
default :
2012-09-27 17:32:58 +08:00
if ( hdev - > ops - > im_transceive ) {
r = hdev - > ops - > im_transceive ( hdev , target , skb , cb ,
2012-09-11 10:43:50 +02:00
cb_context ) ;
2012-04-10 19:43:06 +02:00
if ( r = = 1 )
r = - ENOTSUPP ;
2012-10-17 15:23:39 +02:00
} else {
2012-04-10 19:43:06 +02:00
r = - ENOTSUPP ;
2012-10-17 15:23:39 +02:00
}
2012-09-11 10:43:50 +02:00
break ;
2012-04-10 19:43:06 +02:00
}
kfree_skb ( skb ) ;
2012-09-11 10:43:50 +02:00
return r ;
2012-04-10 19:43:06 +02:00
}
2012-10-08 14:54:39 +08:00
static int hci_tm_send ( struct nfc_dev * nfc_dev , struct sk_buff * skb )
2012-09-27 17:32:58 +08:00
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2013-05-25 01:21:21 +02:00
if ( ! hdev - > ops - > tm_send ) {
kfree_skb ( skb ) ;
return - ENOTSUPP ;
}
2012-12-04 16:44:25 +01:00
2013-05-25 01:21:21 +02:00
return hdev - > ops - > tm_send ( hdev , skb ) ;
2012-09-27 17:32:58 +08:00
}
2012-05-07 12:31:16 +02:00
static int hci_check_presence ( struct nfc_dev * nfc_dev ,
struct nfc_target * target )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2013-05-25 01:21:21 +02:00
if ( ! hdev - > ops - > check_presence )
return 0 ;
2012-05-07 12:31:16 +02:00
2013-05-25 01:21:21 +02:00
return hdev - > ops - > check_presence ( hdev , target ) ;
2012-05-07 12:31:16 +02:00
}
2013-05-10 11:57:06 +02:00
static int hci_discover_se ( struct nfc_dev * nfc_dev )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
if ( hdev - > ops - > discover_se )
return hdev - > ops - > discover_se ( hdev ) ;
return 0 ;
}
static int hci_enable_se ( struct nfc_dev * nfc_dev , u32 se_idx )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
if ( hdev - > ops - > enable_se )
return hdev - > ops - > enable_se ( hdev , se_idx ) ;
return 0 ;
}
static int hci_disable_se ( struct nfc_dev * nfc_dev , u32 se_idx )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
if ( hdev - > ops - > disable_se )
2013-06-18 01:48:26 +03:00
return hdev - > ops - > disable_se ( hdev , se_idx ) ;
2013-05-10 11:57:06 +02:00
return 0 ;
}
2014-11-13 00:30:34 +01:00
static int hci_se_io ( struct nfc_dev * nfc_dev , u32 se_idx ,
u8 * apdu , size_t apdu_length ,
se_io_cb_t cb , void * cb_context )
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
if ( hdev - > ops - > se_io )
return hdev - > ops - > se_io ( hdev , se_idx , apdu ,
apdu_length , cb , cb_context ) ;
return 0 ;
}
2012-09-18 19:45:48 +02:00
static void nfc_hci_failure ( struct nfc_hci_dev * hdev , int err )
{
mutex_lock ( & hdev - > msg_tx_mutex ) ;
if ( hdev - > cmd_pending_msg = = NULL ) {
nfc_driver_failure ( hdev - > ndev , err ) ;
goto exit ;
}
__nfc_hci_cmd_completion ( hdev , err , NULL ) ;
exit :
mutex_unlock ( & hdev - > msg_tx_mutex ) ;
}
static void nfc_hci_llc_failure ( struct nfc_hci_dev * hdev , int err )
{
nfc_hci_failure ( hdev , err ) ;
}
static void nfc_hci_recv_from_llc ( struct nfc_hci_dev * hdev , struct sk_buff * skb )
{
struct hcp_packet * packet ;
u8 type ;
u8 instruction ;
struct sk_buff * hcp_skb ;
u8 pipe ;
struct sk_buff * frag_skb ;
int msg_len ;
packet = ( struct hcp_packet * ) skb - > data ;
if ( ( packet - > header & ~ NFC_HCI_FRAGMENT ) = = 0 ) {
skb_queue_tail ( & hdev - > rx_hcp_frags , skb ) ;
return ;
}
/* it's the last fragment. Does it need re-aggregation? */
if ( skb_queue_len ( & hdev - > rx_hcp_frags ) ) {
pipe = packet - > header & NFC_HCI_FRAGMENT ;
skb_queue_tail ( & hdev - > rx_hcp_frags , skb ) ;
msg_len = 0 ;
skb_queue_walk ( & hdev - > rx_hcp_frags , frag_skb ) {
msg_len + = ( frag_skb - > len -
NFC_HCI_HCP_PACKET_HEADER_LEN ) ;
}
hcp_skb = nfc_alloc_recv_skb ( NFC_HCI_HCP_PACKET_HEADER_LEN +
msg_len , GFP_KERNEL ) ;
if ( hcp_skb = = NULL ) {
nfc_hci_failure ( hdev , - ENOMEM ) ;
return ;
}
networking: add and use skb_put_u8()
Joe and Bjørn suggested that it'd be nicer to not have the
cast in the fairly common case of doing
*(u8 *)skb_put(skb, 1) = c;
Add skb_put_u8() for this case, and use it across the code,
using the following spatch:
@@
expression SKB, C, S;
typedef u8;
identifier fn = {skb_put};
fresh identifier fn2 = fn ## "_u8";
@@
- *(u8 *)fn(SKB, S) = C;
+ fn2(SKB, C);
Note that due to the "S", the spatch isn't perfect, it should
have checked that S is 1, but there's also places that use a
sizeof expression like sizeof(var) or sizeof(u8) etc. Turns
out that nobody ever did something like
*(u8 *)skb_put(skb, 2) = c;
which would be wrong anyway since the second byte wouldn't be
initialized.
Suggested-by: Joe Perches <joe@perches.com>
Suggested-by: Bjørn Mork <bjorn@mork.no>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2017-06-16 14:29:24 +02:00
skb_put_u8 ( hcp_skb , pipe ) ;
2012-09-18 19:45:48 +02:00
skb_queue_walk ( & hdev - > rx_hcp_frags , frag_skb ) {
msg_len = frag_skb - > len - NFC_HCI_HCP_PACKET_HEADER_LEN ;
networking: introduce and use skb_put_data()
A common pattern with skb_put() is to just want to memcpy()
some data into the new space, introduce skb_put_data() for
this.
An spatch similar to the one for skb_put_zero() converts many
of the places using it:
@@
identifier p, p2;
expression len, skb, data;
type t, t2;
@@
(
-p = skb_put(skb, len);
+p = skb_put_data(skb, data, len);
|
-p = (t)skb_put(skb, len);
+p = skb_put_data(skb, data, len);
)
(
p2 = (t2)p;
-memcpy(p2, data, len);
|
-memcpy(p, data, len);
)
@@
type t, t2;
identifier p, p2;
expression skb, data;
@@
t *p;
...
(
-p = skb_put(skb, sizeof(t));
+p = skb_put_data(skb, data, sizeof(t));
|
-p = (t *)skb_put(skb, sizeof(t));
+p = skb_put_data(skb, data, sizeof(t));
)
(
p2 = (t2)p;
-memcpy(p2, data, sizeof(*p));
|
-memcpy(p, data, sizeof(*p));
)
@@
expression skb, len, data;
@@
-memcpy(skb_put(skb, len), data, len);
+skb_put_data(skb, data, len);
(again, manually post-processed to retain some comments)
Reviewed-by: Stephen Hemminger <stephen@networkplumber.org>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2017-06-16 14:29:20 +02:00
skb_put_data ( hcp_skb ,
frag_skb - > data + NFC_HCI_HCP_PACKET_HEADER_LEN ,
msg_len ) ;
2012-09-18 19:45:48 +02:00
}
skb_queue_purge ( & hdev - > rx_hcp_frags ) ;
} else {
packet - > header & = NFC_HCI_FRAGMENT ;
hcp_skb = skb ;
}
/* if this is a response, dispatch immediately to
* unblock waiting cmd context . Otherwise , enqueue to dispatch
* in separate context where handler can also execute command .
*/
packet = ( struct hcp_packet * ) hcp_skb - > data ;
type = HCP_MSG_GET_TYPE ( packet - > message . header ) ;
if ( type = = NFC_HCI_HCP_RESPONSE ) {
pipe = packet - > header ;
instruction = HCP_MSG_GET_CMD ( packet - > message . header ) ;
skb_pull ( hcp_skb , NFC_HCI_HCP_PACKET_HEADER_LEN +
NFC_HCI_HCP_MESSAGE_HEADER_LEN ) ;
nfc_hci_hcp_message_rx ( hdev , pipe , type , instruction , hcp_skb ) ;
} else {
skb_queue_tail ( & hdev - > msg_rx_queue , hcp_skb ) ;
2012-10-02 16:01:31 -07:00
schedule_work ( & hdev - > msg_rx_work ) ;
2012-09-18 19:45:48 +02:00
}
}
2013-07-31 01:19:43 +02:00
static int hci_fw_download ( struct nfc_dev * nfc_dev , const char * firmware_name )
2013-04-29 17:47:42 +02:00
{
struct nfc_hci_dev * hdev = nfc_get_drvdata ( nfc_dev ) ;
2013-07-31 01:19:43 +02:00
if ( ! hdev - > ops - > fw_download )
2013-05-25 01:21:21 +02:00
return - ENOTSUPP ;
2013-04-29 17:47:42 +02:00
2013-07-31 01:19:43 +02:00
return hdev - > ops - > fw_download ( hdev , firmware_name ) ;
2013-04-29 17:47:42 +02:00
}
2012-05-07 12:31:27 +02:00
static struct nfc_ops hci_nfc_ops = {
2012-04-10 19:43:06 +02:00
. dev_up = hci_dev_up ,
. dev_down = hci_dev_down ,
. start_poll = hci_start_poll ,
. stop_poll = hci_stop_poll ,
2012-09-27 17:32:57 +08:00
. dep_link_up = hci_dep_link_up ,
. dep_link_down = hci_dep_link_down ,
2012-04-10 19:43:06 +02:00
. activate_target = hci_activate_target ,
. deactivate_target = hci_deactivate_target ,
2012-05-16 15:55:48 +02:00
. im_transceive = hci_transceive ,
2012-09-27 17:32:58 +08:00
. tm_send = hci_tm_send ,
2012-05-07 12:31:16 +02:00
. check_presence = hci_check_presence ,
2013-07-31 01:19:43 +02:00
. fw_download = hci_fw_download ,
2013-05-10 11:57:06 +02:00
. discover_se = hci_discover_se ,
. enable_se = hci_enable_se ,
. disable_se = hci_disable_se ,
2014-11-13 00:30:34 +01:00
. se_io = hci_se_io ,
2012-04-10 19:43:06 +02:00
} ;
struct nfc_hci_dev * nfc_hci_allocate_device ( struct nfc_hci_ops * ops ,
struct nfc_hci_init_data * init_data ,
2012-12-18 14:15:49 +01:00
unsigned long quirks ,
2012-04-10 19:43:06 +02:00
u32 protocols ,
2012-09-18 19:45:48 +02:00
const char * llc_name ,
2012-04-10 19:43:06 +02:00
int tx_headroom ,
int tx_tailroom ,
int max_link_payload )
{
struct nfc_hci_dev * hdev ;
if ( ops - > xmit = = NULL )
return NULL ;
if ( protocols = = 0 )
return NULL ;
hdev = kzalloc ( sizeof ( struct nfc_hci_dev ) , GFP_KERNEL ) ;
if ( hdev = = NULL )
return NULL ;
2012-09-18 19:45:48 +02:00
hdev - > llc = nfc_llc_allocate ( llc_name , hdev , ops - > xmit ,
nfc_hci_recv_from_llc , tx_headroom ,
tx_tailroom , nfc_hci_llc_failure ) ;
if ( hdev - > llc = = NULL ) {
kfree ( hdev ) ;
return NULL ;
}
2013-05-07 19:22:11 +02:00
hdev - > ndev = nfc_allocate_device ( & hci_nfc_ops , protocols ,
2012-04-10 19:43:06 +02:00
tx_headroom + HCI_CMDS_HEADROOM ,
tx_tailroom ) ;
if ( ! hdev - > ndev ) {
2012-09-18 19:45:48 +02:00
nfc_llc_free ( hdev - > llc ) ;
2012-04-10 19:43:06 +02:00
kfree ( hdev ) ;
return NULL ;
}
hdev - > ops = ops ;
hdev - > max_data_link_payload = max_link_payload ;
hdev - > init_data = * init_data ;
nfc_set_drvdata ( hdev - > ndev , hdev ) ;
2015-01-27 01:18:12 +01:00
nfc_hci_reset_pipes ( hdev ) ;
2012-04-10 19:43:06 +02:00
2012-12-18 14:15:49 +01:00
hdev - > quirks = quirks ;
2012-04-10 19:43:06 +02:00
return hdev ;
}
EXPORT_SYMBOL ( nfc_hci_allocate_device ) ;
void nfc_hci_free_device ( struct nfc_hci_dev * hdev )
{
nfc_free_device ( hdev - > ndev ) ;
2012-09-18 19:45:48 +02:00
nfc_llc_free ( hdev - > llc ) ;
2012-04-10 19:43:06 +02:00
kfree ( hdev ) ;
}
EXPORT_SYMBOL ( nfc_hci_free_device ) ;
int nfc_hci_register_device ( struct nfc_hci_dev * hdev )
{
mutex_init ( & hdev - > msg_tx_mutex ) ;
INIT_LIST_HEAD ( & hdev - > msg_tx_queue ) ;
INIT_WORK ( & hdev - > msg_tx_work , nfc_hci_msg_tx_work ) ;
2017-10-11 16:03:44 +05:30
timer_setup ( & hdev - > cmd_timer , nfc_hci_cmd_timeout , 0 ) ;
2012-04-10 19:43:06 +02:00
skb_queue_head_init ( & hdev - > rx_hcp_frags ) ;
INIT_WORK ( & hdev - > msg_rx_work , nfc_hci_msg_rx_work ) ;
skb_queue_head_init ( & hdev - > msg_rx_queue ) ;
2012-08-22 16:22:16 -07:00
return nfc_register_device ( hdev - > ndev ) ;
2012-04-10 19:43:06 +02:00
}
EXPORT_SYMBOL ( nfc_hci_register_device ) ;
void nfc_hci_unregister_device ( struct nfc_hci_dev * hdev )
{
2012-07-07 00:53:18 +02:00
struct hci_msg * msg , * n ;
2012-04-10 19:43:06 +02:00
2012-11-26 18:06:27 +01:00
mutex_lock ( & hdev - > msg_tx_mutex ) ;
if ( hdev - > cmd_pending_msg ) {
if ( hdev - > cmd_pending_msg - > cb )
hdev - > cmd_pending_msg - > cb (
hdev - > cmd_pending_msg - > cb_context ,
NULL , - ESHUTDOWN ) ;
kfree ( hdev - > cmd_pending_msg ) ;
hdev - > cmd_pending_msg = NULL ;
}
hdev - > shutting_down = true ;
mutex_unlock ( & hdev - > msg_tx_mutex ) ;
del_timer_sync ( & hdev - > cmd_timer ) ;
cancel_work_sync ( & hdev - > msg_tx_work ) ;
cancel_work_sync ( & hdev - > msg_rx_work ) ;
nfc_unregister_device ( hdev - > ndev ) ;
2012-04-10 19:43:06 +02:00
skb_queue_purge ( & hdev - > rx_hcp_frags ) ;
skb_queue_purge ( & hdev - > msg_rx_queue ) ;
2012-07-07 00:53:18 +02:00
list_for_each_entry_safe ( msg , n , & hdev - > msg_tx_queue , msg_l ) {
2012-04-10 19:43:06 +02:00
list_del ( & msg - > msg_l ) ;
skb_queue_purge ( & msg - > msg_frags ) ;
kfree ( msg ) ;
}
}
EXPORT_SYMBOL ( nfc_hci_unregister_device ) ;
void nfc_hci_set_clientdata ( struct nfc_hci_dev * hdev , void * clientdata )
{
hdev - > clientdata = clientdata ;
}
EXPORT_SYMBOL ( nfc_hci_set_clientdata ) ;
void * nfc_hci_get_clientdata ( struct nfc_hci_dev * hdev )
{
return hdev - > clientdata ;
}
EXPORT_SYMBOL ( nfc_hci_get_clientdata ) ;
2012-06-11 13:36:52 +02:00
void nfc_hci_driver_failure ( struct nfc_hci_dev * hdev , int err )
{
nfc_hci_failure ( hdev , err ) ;
}
2012-04-30 18:21:51 +02:00
EXPORT_SYMBOL ( nfc_hci_driver_failure ) ;
2012-10-17 15:23:39 +02:00
void nfc_hci_recv_frame ( struct nfc_hci_dev * hdev , struct sk_buff * skb )
2012-04-10 19:43:06 +02:00
{
2012-09-18 19:45:48 +02:00
nfc_llc_rcv_from_drv ( hdev - > llc , skb ) ;
2012-04-10 19:43:06 +02:00
}
EXPORT_SYMBOL ( nfc_hci_recv_frame ) ;
2012-09-13 17:10:00 +02:00
static int __init nfc_hci_init ( void )
{
return nfc_llc_init ( ) ;
}
static void __exit nfc_hci_exit ( void )
{
nfc_llc_exit ( ) ;
}
2012-09-18 19:45:48 +02:00
subsys_initcall ( nfc_hci_init ) ;
2012-09-13 17:10:00 +02:00
module_exit ( nfc_hci_exit ) ;
2012-04-10 19:43:06 +02:00
MODULE_LICENSE ( " GPL " ) ;
2012-09-18 19:25:38 +02:00
MODULE_DESCRIPTION ( " NFC HCI Core " ) ;