2007-02-09 23:24:33 +09:00
/*
2005-04-16 15:20:36 -07:00
BlueZ - Bluetooth protocol stack for Linux
2010-05-28 11:53:46 -04:00
Copyright ( c ) 2000 - 2001 , 2010 , Code Aurora Forum . All rights reserved .
2005-04-16 15:20:36 -07:00
Written 2000 , 2001 by Maxim Krasnyansky < maxk @ qualcomm . 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 ;
THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS
OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS .
IN NO EVENT SHALL THE COPYRIGHT HOLDER ( S ) AND AUTHOR ( S ) BE LIABLE FOR ANY
2007-02-09 23:24:33 +09:00
CLAIM , OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES , OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN
ACTION OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF
2005-04-16 15:20:36 -07:00
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
2007-02-09 23:24:33 +09:00
ALL LIABILITY , INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS ,
COPYRIGHTS , TRADEMARKS OR OTHER RIGHTS , RELATING TO USE OF THIS
2005-04-16 15:20:36 -07:00
SOFTWARE IS DISCLAIMED .
*/
/* Bluetooth HCI event handling. */
# include <asm/unaligned.h>
# include <net/bluetooth/bluetooth.h>
# include <net/bluetooth/hci_core.h>
2012-08-09 09:52:30 +02:00
# include <net/bluetooth/mgmt.h>
2013-10-10 14:54:14 -07:00
2013-10-10 14:54:15 -07:00
# include "a2mp.h"
2013-10-10 14:54:14 -07:00
# include "amp.h"
2005-04-16 15:20:36 -07:00
/* Handle HCI Event packets */
2007-10-20 13:33:56 +02:00
static void hci_cc_inquiry_cancel ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-16 15:20:36 -07:00
2013-04-30 15:29:34 -03:00
if ( status )
2007-10-20 13:33:56 +02:00
return ;
2005-04-16 15:20:36 -07:00
2011-11-04 14:16:53 -03:00
clear_bit ( HCI_INQUIRY , & hdev - > flags ) ;
2013-03-27 20:04:56 -03:00
smp_mb__after_clear_bit ( ) ; /* wake_up_bit advises about this barrier */
wake_up_bit ( & hdev - > flags , HCI_INQUIRY ) ;
2011-11-04 14:16:53 -03:00
2007-10-20 13:33:56 +02:00
hci_conn_check_pending ( hdev ) ;
}
2006-11-18 22:14:22 +01:00
2012-03-21 00:03:35 -03:00
static void hci_cc_periodic_inq ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2012-03-21 00:03:36 -03:00
if ( status )
return ;
set_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) ;
2012-03-21 00:03:35 -03:00
}
2007-10-20 13:33:56 +02:00
static void hci_cc_exit_periodic_inq ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2006-11-18 22:14:22 +01:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-11-18 22:14:22 +01:00
2007-10-20 13:33:56 +02:00
if ( status )
return ;
2005-04-16 15:20:36 -07:00
2012-03-21 00:03:36 -03:00
clear_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) ;
2007-10-20 13:33:56 +02:00
hci_conn_check_pending ( hdev ) ;
}
2012-05-17 00:36:24 -03:00
static void hci_cc_remote_name_req_cancel ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
BT_DBG ( " %s " , hdev - > name ) ;
}
static void hci_cc_role_discovery ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_role_discovery * rp = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( rp - > handle ) ) ;
if ( conn ) {
if ( rp - > role )
conn - > link_mode & = ~ HCI_LM_MASTER ;
else
conn - > link_mode | = HCI_LM_MASTER ;
2005-04-16 15:20:36 -07:00
}
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
2005-04-16 15:20:36 -07:00
}
2008-07-14 20:13:47 +02:00
static void hci_cc_read_link_policy ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_link_policy * rp = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2008-07-14 20:13:47 +02:00
if ( rp - > status )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( rp - > handle ) ) ;
if ( conn )
conn - > link_policy = __le16_to_cpu ( rp - > policy ) ;
hci_dev_unlock ( hdev ) ;
}
2007-10-20 13:33:56 +02:00
static void hci_cc_write_link_policy ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_rp_write_link_policy * rp = ( void * ) skb - > data ;
2005-04-16 15:20:36 -07:00
struct hci_conn * conn ;
2006-07-03 10:02:33 +02:00
void * sent ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_LINK_POLICY ) ;
if ( ! sent )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( rp - > handle ) ) ;
2008-07-14 20:13:47 +02:00
if ( conn )
2008-05-02 16:25:46 -07:00
conn - > link_policy = get_unaligned_le16 ( sent + 2 ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
}
2005-04-16 15:20:36 -07:00
2012-05-17 00:36:24 -03:00
static void hci_cc_read_def_link_policy ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 20:13:47 +02:00
{
struct hci_rp_read_def_link_policy * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2008-07-14 20:13:47 +02:00
if ( rp - > status )
return ;
hdev - > link_policy = __le16_to_cpu ( rp - > policy ) ;
}
2012-05-17 00:36:24 -03:00
static void hci_cc_write_def_link_policy ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 20:13:47 +02:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
void * sent ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 20:13:47 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_DEF_LINK_POLICY ) ;
if ( ! sent )
return ;
if ( ! status )
hdev - > link_policy = get_unaligned_le16 ( sent ) ;
}
2007-10-20 13:33:56 +02:00
static void hci_cc_reset ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2006-07-03 10:02:33 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 10:02:33 +02:00
2011-03-16 15:36:29 -03:00
clear_bit ( HCI_RESET , & hdev - > flags ) ;
2012-02-21 17:55:47 +02:00
/* Reset all non-persistent flags */
2013-03-15 17:06:57 -05:00
hdev - > dev_flags & = ~ HCI_PERSISTENT_MASK ;
2012-02-23 16:50:05 +02:00
hdev - > discovery . state = DISCOVERY_STOPPED ;
2012-11-08 01:22:59 +01:00
hdev - > inq_tx_power = HCI_TX_POWER_INVALID ;
hdev - > adv_tx_power = HCI_TX_POWER_INVALID ;
2012-11-08 01:23:00 +01:00
memset ( hdev - > adv_data , 0 , sizeof ( hdev - > adv_data ) ) ;
hdev - > adv_data_len = 0 ;
2013-10-16 00:16:47 -07:00
memset ( hdev - > scan_rsp_data , 0 , sizeof ( hdev - > scan_rsp_data ) ) ;
hdev - > scan_rsp_data_len = 0 ;
2013-10-19 07:09:11 -07:00
hdev - > ssp_debug_mode = 0 ;
2007-10-20 13:33:56 +02:00
}
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
static void hci_cc_write_local_name ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
void * sent ;
2006-07-03 10:02:33 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_LOCAL_NAME ) ;
if ( ! sent )
return ;
2006-07-03 10:02:33 +02:00
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-02-22 18:17:32 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_set_local_name_complete ( hdev , sent , status ) ;
2012-02-22 21:06:55 +02:00
else if ( ! status )
memcpy ( hdev - > dev_name , sent , HCI_MAX_NAME_LENGTH ) ;
2012-02-22 18:17:32 +02:00
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
static void hci_cc_read_local_name ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_local_name * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
2012-02-22 20:14:22 +02:00
if ( test_bit ( HCI_SETUP , & hdev - > dev_flags ) )
memcpy ( hdev - > dev_name , rp - > name , HCI_MAX_NAME_LENGTH ) ;
2007-10-20 13:33:56 +02:00
}
static void hci_cc_write_auth_enable ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
void * sent ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_AUTH_ENABLE ) ;
if ( ! sent )
return ;
if ( ! status ) {
__u8 param = * ( ( __u8 * ) sent ) ;
if ( param = = AUTH_ENABLED )
set_bit ( HCI_AUTH , & hdev - > flags ) ;
else
clear_bit ( HCI_AUTH , & hdev - > flags ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-20 13:33:56 +02:00
2012-02-16 23:56:27 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_auth_enable_complete ( hdev , status ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-20 13:33:56 +02:00
static void hci_cc_write_encrypt_mode ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2005-04-16 15:20:36 -07:00
void * sent ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_ENCRYPT_MODE ) ;
if ( ! sent )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
if ( ! status ) {
__u8 param = * ( ( __u8 * ) sent ) ;
if ( param )
set_bit ( HCI_ENCRYPT , & hdev - > flags ) ;
else
clear_bit ( HCI_ENCRYPT , & hdev - > flags ) ;
}
}
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
static void hci_cc_write_scan_enable ( struct hci_dev * hdev , struct sk_buff * skb )
{
2011-11-04 00:17:45 +02:00
__u8 param , status = * ( ( __u8 * ) skb - > data ) ;
int old_pscan , old_iscan ;
2007-10-20 13:33:56 +02:00
void * sent ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_SCAN_ENABLE ) ;
if ( ! sent )
return ;
2005-04-16 15:20:36 -07:00
2011-11-04 00:17:45 +02:00
param = * ( ( __u8 * ) sent ) ;
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-08-09 09:52:29 +02:00
if ( status ) {
2011-11-08 20:40:14 +02:00
mgmt_write_scan_failed ( hdev , param , status ) ;
2011-11-07 22:16:03 +02:00
hdev - > discov_timeout = 0 ;
goto done ;
}
2013-10-02 13:43:14 +03:00
/* We need to ensure that we set this back on if someone changed
* the scan mode through a raw HCI socket .
*/
set_bit ( HCI_BREDR_ENABLED , & hdev - > dev_flags ) ;
2011-11-04 00:17:45 +02:00
old_pscan = test_and_clear_bit ( HCI_PSCAN , & hdev - > flags ) ;
old_iscan = test_and_clear_bit ( HCI_ISCAN , & hdev - > flags ) ;
if ( param & SCAN_INQUIRY ) {
set_bit ( HCI_ISCAN , & hdev - > flags ) ;
if ( ! old_iscan )
2011-11-08 20:40:14 +02:00
mgmt_discoverable ( hdev , 1 ) ;
2011-11-04 00:17:45 +02:00
} else if ( old_iscan )
2011-11-08 20:40:14 +02:00
mgmt_discoverable ( hdev , 0 ) ;
2011-11-04 00:17:45 +02:00
if ( param & SCAN_PAGE ) {
set_bit ( HCI_PSCAN , & hdev - > flags ) ;
if ( ! old_pscan )
2011-11-08 20:40:14 +02:00
mgmt_connectable ( hdev , 1 ) ;
2011-11-04 00:17:45 +02:00
} else if ( old_pscan )
2011-11-08 20:40:14 +02:00
mgmt_connectable ( hdev , 0 ) ;
2005-04-16 15:20:36 -07:00
2011-11-04 00:17:45 +02:00
done :
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
static void hci_cc_read_class_of_dev ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_class_of_dev * rp = ( void * ) skb - > data ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
memcpy ( hdev - > dev_class , rp - > dev_class , 3 ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s class 0x%.2x%.2x%.2x " , hdev - > name ,
2012-05-17 00:36:24 -03:00
hdev - > dev_class [ 2 ] , hdev - > dev_class [ 1 ] , hdev - > dev_class [ 0 ] ) ;
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
static void hci_cc_write_class_of_dev ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
void * sent ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_CLASS_OF_DEV ) ;
if ( ! sent )
return ;
2005-04-16 15:20:36 -07:00
2012-02-22 18:38:01 +01:00
hci_dev_lock ( hdev ) ;
if ( status = = 0 )
memcpy ( hdev - > dev_class , sent , 3 ) ;
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_set_class_of_dev_complete ( hdev , sent , status ) ;
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
static void hci_cc_read_voice_setting ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_voice_setting * rp = ( void * ) skb - > data ;
__u16 setting ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
setting = __le16_to_cpu ( rp - > voice_setting ) ;
2008-07-14 20:13:47 +02:00
if ( hdev - > voice_setting = = setting )
2007-10-20 13:33:56 +02:00
return ;
hdev - > voice_setting = setting ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s voice setting 0x%4.4x " , hdev - > name , setting ) ;
2007-10-20 13:33:56 +02:00
2011-12-14 22:58:44 -02:00
if ( hdev - > notify )
2007-10-20 13:33:56 +02:00
hdev - > notify ( hdev , HCI_NOTIFY_VOICE_SETTING ) ;
}
2012-05-23 04:04:21 -03:00
static void hci_cc_write_voice_setting ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2008-07-14 20:13:47 +02:00
__u16 setting ;
2007-10-20 13:33:56 +02:00
void * sent ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:47 +02:00
if ( status )
return ;
2007-10-20 13:33:56 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_VOICE_SETTING ) ;
if ( ! sent )
return ;
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:47 +02:00
setting = get_unaligned_le16 ( sent ) ;
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:47 +02:00
if ( hdev - > voice_setting = = setting )
return ;
hdev - > voice_setting = setting ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s voice setting 0x%4.4x " , hdev - > name , setting ) ;
2005-04-16 15:20:36 -07:00
2011-12-14 22:58:44 -02:00
if ( hdev - > notify )
2008-07-14 20:13:47 +02:00
hdev - > notify ( hdev , HCI_NOTIFY_VOICE_SETTING ) ;
2005-04-16 15:20:36 -07:00
}
2013-10-14 13:56:16 -07:00
static void hci_cc_read_num_supported_iac ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_read_num_supported_iac * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
if ( rp - > status )
return ;
hdev - > num_iac = rp - > num_iac ;
BT_DBG ( " %s num iac %d " , hdev - > name , hdev - > num_iac ) ;
}
2008-07-14 20:13:48 +02:00
static void hci_cc_write_ssp_mode ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-10-25 00:09:51 +03:00
struct hci_cp_write_ssp_mode * sent ;
2008-07-14 20:13:48 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 20:13:48 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_SSP_MODE ) ;
if ( ! sent )
return ;
2012-10-25 00:09:51 +03:00
if ( ! status ) {
if ( sent - > mode )
2013-04-17 15:00:51 +03:00
hdev - > features [ 1 ] [ 0 ] | = LMP_HOST_SSP ;
2012-10-25 00:09:51 +03:00
else
2013-04-17 15:00:51 +03:00
hdev - > features [ 1 ] [ 0 ] & = ~ LMP_HOST_SSP ;
2012-10-25 00:09:51 +03:00
}
2012-02-17 00:56:28 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-10-25 00:09:51 +03:00
mgmt_ssp_enable_complete ( hdev , sent - > mode , status ) ;
2012-02-22 12:38:31 +02:00
else if ( ! status ) {
2012-10-25 00:09:51 +03:00
if ( sent - > mode )
2012-02-22 12:38:31 +02:00
set_bit ( HCI_SSP_ENABLED , & hdev - > dev_flags ) ;
else
clear_bit ( HCI_SSP_ENABLED , & hdev - > dev_flags ) ;
}
2008-07-14 20:13:48 +02:00
}
2014-01-10 02:07:23 -08:00
static void hci_cc_write_sc_support ( struct hci_dev * hdev , struct sk_buff * skb )
{
u8 status = * ( ( u8 * ) skb - > data ) ;
struct hci_cp_write_sc_support * sent ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_SC_SUPPORT ) ;
if ( ! sent )
return ;
if ( ! status ) {
if ( sent - > support )
hdev - > features [ 1 ] [ 0 ] | = LMP_HOST_SC ;
else
hdev - > features [ 1 ] [ 0 ] & = ~ LMP_HOST_SC ;
}
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_sc_enable_complete ( hdev , sent - > support , status ) ;
else if ( ! status ) {
if ( sent - > support )
set_bit ( HCI_SC_ENABLED , & hdev - > dev_flags ) ;
else
clear_bit ( HCI_SC_ENABLED , & hdev - > dev_flags ) ;
}
}
2007-10-20 13:33:56 +02:00
static void hci_cc_read_local_version ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_local_version * rp = ( void * ) skb - > data ;
2006-09-23 09:57:20 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2006-09-23 09:57:20 +02:00
2007-10-20 13:33:56 +02:00
if ( rp - > status )
2013-03-05 20:37:49 +02:00
return ;
2006-09-23 09:57:20 +02:00
2013-10-18 12:04:50 -07:00
if ( test_bit ( HCI_SETUP , & hdev - > dev_flags ) ) {
hdev - > hci_ver = rp - > hci_ver ;
hdev - > hci_rev = __le16_to_cpu ( rp - > hci_rev ) ;
hdev - > lmp_ver = rp - > lmp_ver ;
hdev - > manufacturer = __le16_to_cpu ( rp - > manufacturer ) ;
hdev - > lmp_subver = __le16_to_cpu ( rp - > lmp_subver ) ;
}
2011-01-25 01:19:58 +02:00
}
2012-05-23 04:04:21 -03:00
static void hci_cc_read_local_commands ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_rp_read_local_commands * rp = ( void * ) skb - > data ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-16 15:20:36 -07:00
2013-10-31 04:54:33 -07:00
if ( rp - > status )
return ;
if ( test_bit ( HCI_SETUP , & hdev - > dev_flags ) )
2013-03-05 20:37:43 +02:00
memcpy ( hdev - > commands , rp - > commands , sizeof ( hdev - > commands ) ) ;
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2012-05-23 04:04:21 -03:00
static void hci_cc_read_local_features ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_rp_read_local_features * rp = ( void * ) skb - > data ;
2007-07-11 09:51:55 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
2007-07-11 09:51:55 +02:00
2007-10-20 13:33:56 +02:00
memcpy ( hdev - > features , rp - > features , 8 ) ;
2007-07-11 09:51:55 +02:00
2007-10-20 13:33:56 +02:00
/* Adjust default settings according to features
* supported by device . */
2005-04-16 15:20:36 -07:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 0 ] & LMP_3SLOT )
2007-10-20 13:33:56 +02:00
hdev - > pkt_type | = ( HCI_DM3 | HCI_DH3 ) ;
2005-04-16 15:20:36 -07:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 0 ] & LMP_5SLOT )
2007-10-20 13:33:56 +02:00
hdev - > pkt_type | = ( HCI_DM5 | HCI_DH5 ) ;
2005-04-16 15:20:36 -07:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 1 ] & LMP_HV2 ) {
2007-10-20 13:33:56 +02:00
hdev - > pkt_type | = ( HCI_HV2 ) ;
hdev - > esco_type | = ( ESCO_HV2 ) ;
}
2005-04-16 15:20:36 -07:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 1 ] & LMP_HV3 ) {
2007-10-20 13:33:56 +02:00
hdev - > pkt_type | = ( HCI_HV3 ) ;
hdev - > esco_type | = ( ESCO_HV3 ) ;
}
2005-04-16 15:20:36 -07:00
2012-07-24 15:03:49 -03:00
if ( lmp_esco_capable ( hdev ) )
2007-10-20 13:33:56 +02:00
hdev - > esco_type | = ( ESCO_EV3 ) ;
2006-07-03 10:02:29 +02:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 4 ] & LMP_EV4 )
2007-10-20 13:33:56 +02:00
hdev - > esco_type | = ( ESCO_EV4 ) ;
2006-07-03 10:02:29 +02:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 4 ] & LMP_EV5 )
2007-10-20 13:33:56 +02:00
hdev - > esco_type | = ( ESCO_EV5 ) ;
2005-04-16 15:20:36 -07:00
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 5 ] & LMP_EDR_ESCO_2M )
2009-02-06 09:13:37 +01:00
hdev - > esco_type | = ( ESCO_2EV3 ) ;
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 5 ] & LMP_EDR_ESCO_3M )
2009-02-06 09:13:37 +01:00
hdev - > esco_type | = ( ESCO_3EV3 ) ;
2013-04-17 15:00:51 +03:00
if ( hdev - > features [ 0 ] [ 5 ] & LMP_EDR_3S_ESCO )
2009-02-06 09:13:37 +01:00
hdev - > esco_type | = ( ESCO_2EV5 | ESCO_3EV5 ) ;
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2011-06-30 19:20:52 -03:00
static void hci_cc_read_local_ext_features ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-06-30 19:20:52 -03:00
{
struct hci_rp_read_local_ext_features * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-06-30 19:20:52 -03:00
if ( rp - > status )
2013-03-05 20:37:49 +02:00
return ;
2011-06-30 19:20:52 -03:00
2013-10-18 12:04:47 -07:00
if ( hdev - > max_page < rp - > max_page )
hdev - > max_page = rp - > max_page ;
2013-04-17 15:00:52 +03:00
2013-04-17 15:00:51 +03:00
if ( rp - > page < HCI_MAX_PAGES )
memcpy ( hdev - > features [ rp - > page ] , rp - > features , 8 ) ;
2011-06-30 19:20:52 -03:00
}
2011-11-24 14:52:02 +02:00
static void hci_cc_read_flow_control_mode ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-11-24 14:52:02 +02:00
{
struct hci_rp_read_flow_control_mode * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-11-24 14:52:02 +02:00
2013-03-05 20:37:49 +02:00
if ( ! rp - > status )
hdev - > flow_ctl_mode = rp - > mode ;
2011-11-24 14:52:02 +02:00
}
2007-10-20 13:33:56 +02:00
static void hci_cc_read_buffer_size ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_buffer_size * rp = ( void * ) skb - > data ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
if ( rp - > status )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hdev - > acl_mtu = __le16_to_cpu ( rp - > acl_mtu ) ;
hdev - > sco_mtu = rp - > sco_mtu ;
hdev - > acl_pkts = __le16_to_cpu ( rp - > acl_max_pkt ) ;
hdev - > sco_pkts = __le16_to_cpu ( rp - > sco_max_pkt ) ;
if ( test_bit ( HCI_QUIRK_FIXUP_BUFFER_SIZE , & hdev - > quirks ) ) {
hdev - > sco_mtu = 64 ;
hdev - > sco_pkts = 8 ;
2005-04-16 15:20:36 -07:00
}
2007-10-20 13:33:56 +02:00
hdev - > acl_cnt = hdev - > acl_pkts ;
hdev - > sco_cnt = hdev - > sco_pkts ;
2012-05-17 00:36:24 -03:00
BT_DBG ( " %s acl mtu %d:%d sco mtu %d:%d " , hdev - > name , hdev - > acl_mtu ,
hdev - > acl_pkts , hdev - > sco_mtu , hdev - > sco_pkts ) ;
2007-10-20 13:33:56 +02:00
}
static void hci_cc_read_bd_addr ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_read_bd_addr * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 13:33:56 +02:00
if ( ! rp - > status )
bacpy ( & hdev - > bdaddr , & rp - > bdaddr ) ;
2010-12-21 23:01:27 +02:00
}
2013-03-15 17:07:11 -05:00
static void hci_cc_read_page_scan_activity ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_read_page_scan_activity * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
if ( test_bit ( HCI_INIT , & hdev - > flags ) & & ! rp - > status ) {
hdev - > page_scan_interval = __le16_to_cpu ( rp - > interval ) ;
hdev - > page_scan_window = __le16_to_cpu ( rp - > window ) ;
}
}
2013-03-15 17:07:12 -05:00
static void hci_cc_write_page_scan_activity ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
u8 status = * ( ( u8 * ) skb - > data ) ;
struct hci_cp_write_page_scan_activity * sent ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
if ( status )
return ;
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_PAGE_SCAN_ACTIVITY ) ;
if ( ! sent )
return ;
hdev - > page_scan_interval = __le16_to_cpu ( sent - > interval ) ;
hdev - > page_scan_window = __le16_to_cpu ( sent - > window ) ;
}
2013-03-15 17:07:11 -05:00
static void hci_cc_read_page_scan_type ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_read_page_scan_type * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
if ( test_bit ( HCI_INIT , & hdev - > flags ) & & ! rp - > status )
hdev - > page_scan_type = rp - > type ;
}
2013-03-15 17:07:12 -05:00
static void hci_cc_write_page_scan_type ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
u8 status = * ( ( u8 * ) skb - > data ) ;
u8 * type ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
if ( status )
return ;
type = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_PAGE_SCAN_TYPE ) ;
if ( type )
hdev - > page_scan_type = * type ;
}
2011-12-07 15:56:51 +02:00
static void hci_cc_read_data_block_size ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-12-07 15:56:51 +02:00
{
struct hci_rp_read_data_block_size * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-12-07 15:56:51 +02:00
if ( rp - > status )
return ;
hdev - > block_mtu = __le16_to_cpu ( rp - > max_acl_len ) ;
hdev - > block_len = __le16_to_cpu ( rp - > block_len ) ;
hdev - > num_blocks = __le16_to_cpu ( rp - > num_blocks ) ;
hdev - > block_cnt = hdev - > num_blocks ;
BT_DBG ( " %s blk mtu %d cnt %d len %d " , hdev - > name , hdev - > block_mtu ,
2012-05-17 00:36:24 -03:00
hdev - > block_cnt , hdev - > block_len ) ;
2011-12-07 15:56:51 +02:00
}
2011-10-12 10:53:57 +03:00
static void hci_cc_read_local_amp_info ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-10-12 10:53:57 +03:00
{
struct hci_rp_read_local_amp_info * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-10-12 10:53:57 +03:00
if ( rp - > status )
2012-09-27 17:26:08 +03:00
goto a2mp_rsp ;
2011-10-12 10:53:57 +03:00
hdev - > amp_status = rp - > amp_status ;
hdev - > amp_total_bw = __le32_to_cpu ( rp - > total_bw ) ;
hdev - > amp_max_bw = __le32_to_cpu ( rp - > max_bw ) ;
hdev - > amp_min_latency = __le32_to_cpu ( rp - > min_latency ) ;
hdev - > amp_max_pdu = __le32_to_cpu ( rp - > max_pdu ) ;
hdev - > amp_type = rp - > amp_type ;
hdev - > amp_pal_cap = __le16_to_cpu ( rp - > pal_cap ) ;
hdev - > amp_assoc_size = __le16_to_cpu ( rp - > max_assoc_size ) ;
hdev - > amp_be_flush_to = __le32_to_cpu ( rp - > be_flush_to ) ;
hdev - > amp_max_flush_to = __le32_to_cpu ( rp - > max_flush_to ) ;
2012-09-27 17:26:08 +03:00
a2mp_rsp :
a2mp_send_getinfo_rsp ( hdev ) ;
2011-10-12 10:53:57 +03:00
}
2012-09-27 17:26:09 +03:00
static void hci_cc_read_local_amp_assoc ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_read_local_amp_assoc * rp = ( void * ) skb - > data ;
struct amp_assoc * assoc = & hdev - > loc_assoc ;
size_t rem_len , frag_len ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
if ( rp - > status )
goto a2mp_rsp ;
frag_len = skb - > len - sizeof ( * rp ) ;
rem_len = __le16_to_cpu ( rp - > rem_len ) ;
if ( rem_len > frag_len ) {
2012-09-28 14:44:23 +03:00
BT_DBG ( " frag_len %zu rem_len %zu " , frag_len , rem_len ) ;
2012-09-27 17:26:09 +03:00
memcpy ( assoc - > data + assoc - > offset , rp - > frag , frag_len ) ;
assoc - > offset + = frag_len ;
/* Read other fragments */
amp_read_loc_assoc_frag ( hdev , rp - > phy_handle ) ;
return ;
}
memcpy ( assoc - > data + assoc - > offset , rp - > frag , rem_len ) ;
assoc - > len = assoc - > offset + rem_len ;
assoc - > offset = 0 ;
a2mp_rsp :
/* Send A2MP Rsp when all fragments are received */
a2mp_send_getampassoc_rsp ( hdev , rp - > status ) ;
2012-09-27 17:26:22 +03:00
a2mp_send_create_phy_link_req ( hdev , rp - > status ) ;
2012-09-27 17:26:09 +03:00
}
2011-01-25 01:19:58 +02:00
static void hci_cc_read_inq_rsp_tx_power ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-01-25 01:19:58 +02:00
{
2012-03-11 19:27:21 -07:00
struct hci_rp_read_inq_rsp_tx_power * rp = ( void * ) skb - > data ;
2011-01-25 01:19:58 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2012-03-11 19:27:21 -07:00
if ( ! rp - > status )
hdev - > inq_tx_power = rp - > tx_power ;
2011-01-25 01:19:58 +02:00
}
2011-01-22 06:10:07 +02:00
static void hci_cc_pin_code_reply ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_pin_code_reply * rp = ( void * ) skb - > data ;
struct hci_cp_pin_code_reply * cp ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-01-22 06:10:07 +02:00
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-08 20:40:14 +02:00
mgmt_pin_code_reply_complete ( hdev , & rp - > bdaddr , rp - > status ) ;
2011-01-22 06:10:07 +02:00
2012-08-09 09:52:29 +02:00
if ( rp - > status )
2011-11-08 20:40:16 +02:00
goto unlock ;
2011-01-22 06:10:07 +02:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_PIN_CODE_REPLY ) ;
if ( ! cp )
2011-11-08 20:40:16 +02:00
goto unlock ;
2011-01-22 06:10:07 +02:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & cp - > bdaddr ) ;
if ( conn )
conn - > pin_length = cp - > pin_len ;
2011-11-08 20:40:16 +02:00
unlock :
hci_dev_unlock ( hdev ) ;
2011-01-22 06:10:07 +02:00
}
static void hci_cc_pin_code_neg_reply ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_pin_code_neg_reply * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-01-22 06:10:07 +02:00
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-08 20:40:14 +02:00
mgmt_pin_code_neg_reply_complete ( hdev , & rp - > bdaddr ,
2012-05-17 00:36:24 -03:00
rp - > status ) ;
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2011-01-22 06:10:07 +02:00
}
2011-11-08 20:40:16 +02:00
2011-02-10 22:38:48 -03:00
static void hci_cc_le_read_buffer_size ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_le_read_buffer_size * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-02-10 22:38:48 -03:00
if ( rp - > status )
return ;
hdev - > le_mtu = __le16_to_cpu ( rp - > le_mtu ) ;
hdev - > le_pkts = rp - > le_max_pkt ;
hdev - > le_cnt = hdev - > le_pkts ;
BT_DBG ( " %s le mtu %d:%d " , hdev - > name , hdev - > le_mtu , hdev - > le_pkts ) ;
}
2011-01-22 06:10:07 +02:00
2013-01-22 14:01:59 +02:00
static void hci_cc_le_read_local_features ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_le_read_local_features * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
if ( ! rp - > status )
memcpy ( hdev - > le_features , rp - > features , 8 ) ;
}
2012-10-19 20:57:49 +03:00
static void hci_cc_le_read_adv_tx_power ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_le_read_adv_tx_power * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2013-03-15 17:07:01 -05:00
if ( ! rp - > status )
2012-10-19 20:57:49 +03:00
hdev - > adv_tx_power = rp - > tx_power ;
}
2011-02-19 12:05:57 -03:00
static void hci_cc_user_confirm_reply ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_user_confirm_reply * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-02-19 12:05:57 -03:00
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-03-08 01:25:00 -03:00
mgmt_user_confirm_reply_complete ( hdev , & rp - > bdaddr , ACL_LINK , 0 ,
rp - > status ) ;
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2011-02-19 12:05:57 -03:00
}
static void hci_cc_user_confirm_neg_reply ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-02-19 12:05:57 -03:00
{
struct hci_rp_user_confirm_reply * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-02-19 12:05:57 -03:00
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-08 20:40:14 +02:00
mgmt_user_confirm_neg_reply_complete ( hdev , & rp - > bdaddr ,
2012-03-08 01:25:00 -03:00
ACL_LINK , 0 , rp - > status ) ;
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2011-02-19 12:05:57 -03:00
}
2011-11-23 08:28:34 -08:00
static void hci_cc_user_passkey_reply ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_user_confirm_reply * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-11-23 08:28:34 -08:00
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-02-09 15:26:12 +02:00
mgmt_user_passkey_reply_complete ( hdev , & rp - > bdaddr , ACL_LINK ,
2012-03-08 01:25:00 -03:00
0 , rp - > status ) ;
2011-11-23 08:28:34 -08:00
hci_dev_unlock ( hdev ) ;
}
static void hci_cc_user_passkey_neg_reply ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-11-23 08:28:34 -08:00
{
struct hci_rp_user_confirm_reply * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-11-23 08:28:34 -08:00
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-23 08:28:34 -08:00
mgmt_user_passkey_neg_reply_complete ( hdev , & rp - > bdaddr ,
2012-03-08 01:25:00 -03:00
ACL_LINK , 0 , rp - > status ) ;
2011-11-23 08:28:34 -08:00
hci_dev_unlock ( hdev ) ;
}
2014-01-10 02:07:26 -08:00
static void hci_cc_read_local_oob_data ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-03-22 13:12:21 +01:00
{
struct hci_rp_read_local_oob_data * rp = ( void * ) skb - > data ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-03-22 13:12:21 +01:00
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2014-01-10 02:07:26 -08:00
mgmt_read_local_oob_data_complete ( hdev , rp - > hash , rp - > randomizer ,
NULL , NULL , rp - > status ) ;
hci_dev_unlock ( hdev ) ;
}
static void hci_cc_read_local_oob_ext_data ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_read_local_oob_ext_data * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
hci_dev_lock ( hdev ) ;
mgmt_read_local_oob_data_complete ( hdev , rp - > hash192 , rp - > randomizer192 ,
rp - > hash256 , rp - > randomizer256 ,
rp - > status ) ;
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2011-03-22 13:12:21 +01:00
}
2014-02-19 19:52:13 -08:00
static void hci_cc_le_set_random_addr ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
bdaddr_t * sent ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
sent = hci_sent_cmd_data ( hdev , HCI_OP_LE_SET_RANDOM_ADDR ) ;
if ( ! sent )
return ;
hci_dev_lock ( hdev ) ;
if ( ! status )
bacpy ( & hdev - > random_addr , sent ) ;
hci_dev_unlock ( hdev ) ;
}
2012-11-08 01:23:01 +01:00
static void hci_cc_le_set_adv_enable ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 * sent , status = * ( ( __u8 * ) skb - > data ) ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
sent = hci_sent_cmd_data ( hdev , HCI_OP_LE_SET_ADV_ENABLE ) ;
if ( ! sent )
return ;
hci_dev_lock ( hdev ) ;
2014-02-24 14:52:17 +02:00
if ( ! status )
mgmt_advertising ( hdev , * sent ) ;
2012-11-08 01:23:01 +01:00
2013-03-15 17:07:01 -05:00
hci_dev_unlock ( hdev ) ;
2012-11-08 01:23:01 +01:00
}
2011-05-26 16:23:52 -03:00
static void hci_cc_le_set_scan_enable ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct sk_buff * skb )
2011-05-26 16:23:52 -03:00
{
struct hci_cp_le_set_scan_enable * cp ;
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-05-26 16:23:52 -03:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_LE_SET_SCAN_ENABLE ) ;
if ( ! cp )
return ;
2013-04-30 15:29:36 -03:00
if ( status )
return ;
2011-12-19 16:14:18 +02:00
switch ( cp - > enable ) {
2013-04-04 20:21:02 -03:00
case LE_SCAN_ENABLE :
2011-11-25 20:53:38 -03:00
set_bit ( HCI_LE_SCAN , & hdev - > dev_flags ) ;
2011-12-19 16:14:18 +02:00
break ;
2013-04-04 20:21:02 -03:00
case LE_SCAN_DISABLE :
2011-11-25 20:53:38 -03:00
clear_bit ( HCI_LE_SCAN , & hdev - > dev_flags ) ;
2011-12-19 16:14:18 +02:00
break ;
default :
BT_ERR ( " Used reserved LE_Scan_Enable param %d " , cp - > enable ) ;
break ;
2011-05-26 16:23:53 -03:00
}
2011-05-26 16:23:52 -03:00
}
2013-01-22 14:02:00 +02:00
static void hci_cc_le_read_white_list_size ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_le_read_white_list_size * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x size %u " , hdev - > name , rp - > status , rp - > size ) ;
if ( ! rp - > status )
hdev - > le_white_list_size = rp - > size ;
}
2013-01-22 14:02:01 +02:00
static void hci_cc_le_read_supported_states ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_le_read_supported_states * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
if ( ! rp - > status )
memcpy ( hdev - > le_states , rp - > le_states , 8 ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_cc_write_le_host_supported ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-06-30 19:20:53 -03:00
{
2012-02-22 16:37:11 +02:00
struct hci_cp_write_le_host_supported * sent ;
2011-06-30 19:20:53 -03:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-06-30 19:20:53 -03:00
2012-02-22 16:37:11 +02:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_LE_HOST_SUPPORTED ) ;
2012-02-28 01:07:22 +02:00
if ( ! sent )
2011-06-30 19:20:53 -03:00
return ;
2012-02-28 01:07:22 +02:00
if ( ! status ) {
2013-09-25 13:26:08 +03:00
if ( sent - > le ) {
2013-04-17 15:00:51 +03:00
hdev - > features [ 1 ] [ 0 ] | = LMP_HOST_LE ;
2013-09-25 13:26:08 +03:00
set_bit ( HCI_LE_ENABLED , & hdev - > dev_flags ) ;
} else {
2013-04-17 15:00:51 +03:00
hdev - > features [ 1 ] [ 0 ] & = ~ LMP_HOST_LE ;
2013-09-25 13:26:08 +03:00
clear_bit ( HCI_LE_ENABLED , & hdev - > dev_flags ) ;
2013-10-05 12:01:04 +02:00
clear_bit ( HCI_ADVERTISING , & hdev - > dev_flags ) ;
2013-09-25 13:26:08 +03:00
}
2012-10-24 21:11:59 +03:00
if ( sent - > simul )
2013-04-17 15:00:51 +03:00
hdev - > features [ 1 ] [ 0 ] | = LMP_HOST_LE_BREDR ;
2012-10-24 21:11:59 +03:00
else
2013-04-17 15:00:51 +03:00
hdev - > features [ 1 ] [ 0 ] & = ~ LMP_HOST_LE_BREDR ;
2012-02-28 01:07:22 +02:00
}
2011-06-30 19:20:53 -03:00
}
2012-09-27 17:26:20 +03:00
static void hci_cc_write_remote_amp_assoc ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_rp_write_remote_amp_assoc * rp = ( void * ) skb - > data ;
BT_DBG ( " %s status 0x%2.2x phy_handle 0x%2.2x " ,
hdev - > name , rp - > status , rp - > phy_handle ) ;
if ( rp - > status )
return ;
amp_write_rem_assoc_continue ( hdev , rp - > phy_handle ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_cs_inquiry ( struct hci_dev * hdev , __u8 status )
2007-10-20 13:33:56 +02:00
{
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 13:33:56 +02:00
if ( status ) {
hci_conn_check_pending ( hdev ) ;
2011-04-27 10:29:57 -04:00
return ;
}
2011-11-04 14:16:53 -03:00
set_bit ( HCI_INQUIRY , & hdev - > flags ) ;
2005-04-16 15:20:36 -07:00
}
2012-05-23 04:04:18 -03:00
static void hci_cs_create_conn ( struct hci_dev * hdev , __u8 status )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_cp_create_conn * cp ;
2005-04-16 15:20:36 -07:00
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 13:33:56 +02:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_CREATE_CONN ) ;
2005-04-16 15:20:36 -07:00
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & cp - > bdaddr ) ;
2012-09-25 12:49:43 +03:00
BT_DBG ( " %s bdaddr %pMR hcon %p " , hdev - > name , & cp - > bdaddr , conn ) ;
2005-04-16 15:20:36 -07:00
if ( status ) {
if ( conn & & conn - > state = = BT_CONNECT ) {
2006-10-15 17:30:56 +02:00
if ( status ! = 0x0c | | conn - > attempt > 2 ) {
conn - > state = BT_CLOSED ;
hci_proto_connect_cfm ( conn , status ) ;
hci_conn_del ( conn ) ;
} else
conn - > state = BT_CONNECT2 ;
2005-04-16 15:20:36 -07:00
}
} else {
if ( ! conn ) {
conn = hci_conn_add ( hdev , ACL_LINK , & cp - > bdaddr ) ;
if ( conn ) {
2012-01-16 09:49:58 +02:00
conn - > out = true ;
2005-04-16 15:20:36 -07:00
conn - > link_mode | = HCI_LM_MASTER ;
} else
2010-07-18 15:13:37 -03:00
BT_ERR ( " No memory for new connection " ) ;
2005-04-16 15:20:36 -07:00
}
}
hci_dev_unlock ( hdev ) ;
}
2007-10-20 13:33:56 +02:00
static void hci_cs_add_sco ( struct hci_dev * hdev , __u8 status )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_cp_add_sco * cp ;
struct hci_conn * acl , * sco ;
__u16 handle ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 14:55:10 +02:00
2007-10-20 13:33:56 +02:00
if ( ! status )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_ADD_SCO ) ;
if ( ! cp )
return ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
handle = __le16_to_cpu ( cp - > handle ) ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s handle 0x%4.4x " , hdev - > name , handle ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
acl = hci_conn_hash_lookup_handle ( hdev , handle ) ;
2011-01-11 17:20:20 +02:00
if ( acl ) {
sco = acl - > link ;
if ( sco ) {
sco - > state = BT_CLOSED ;
2005-04-16 15:20:36 -07:00
2011-01-11 17:20:20 +02:00
hci_proto_connect_cfm ( sco , status ) ;
hci_conn_del ( sco ) ;
}
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
}
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:49 +02:00
static void hci_cs_auth_requested ( struct hci_dev * hdev , __u8 status )
{
struct hci_cp_auth_requested * cp ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 20:13:49 +02:00
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_AUTH_REQUESTED ) ;
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
if ( conn ) {
if ( conn - > state = = BT_CONFIG ) {
hci_proto_connect_cfm ( conn , status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2008-07-14 20:13:49 +02:00
}
}
hci_dev_unlock ( hdev ) ;
}
static void hci_cs_set_conn_encrypt ( struct hci_dev * hdev , __u8 status )
{
struct hci_cp_set_conn_encrypt * cp ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 20:13:49 +02:00
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_SET_CONN_ENCRYPT ) ;
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
if ( conn ) {
if ( conn - > state = = BT_CONFIG ) {
hci_proto_connect_cfm ( conn , status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2008-07-14 20:13:49 +02:00
}
}
hci_dev_unlock ( hdev ) ;
}
2010-11-18 22:22:29 +02:00
static int hci_outgoing_auth_needed ( struct hci_dev * hdev ,
2012-05-17 00:36:24 -03:00
struct hci_conn * conn )
2010-11-18 22:22:28 +02:00
{
if ( conn - > state ! = BT_CONFIG | | ! conn - > out )
return 0 ;
2011-01-19 12:06:52 +05:30
if ( conn - > pending_sec_level = = BT_SECURITY_SDP )
2010-11-18 22:22:28 +02:00
return 0 ;
/* Only request authentication for SSP connections or non-SSP
2014-01-08 16:40:39 +02:00
* devices with sec_level MEDIUM or HIGH or if MITM protection
* is requested .
*/
2012-05-17 00:36:24 -03:00
if ( ! hci_conn_ssp_enabled ( conn ) & & ! ( conn - > auth_type & 0x01 ) & &
2014-01-08 16:40:39 +02:00
conn - > pending_sec_level ! = BT_SECURITY_HIGH & &
conn - > pending_sec_level ! = BT_SECURITY_MEDIUM )
2010-11-18 22:22:28 +02:00
return 0 ;
return 1 ;
}
2012-05-23 04:04:18 -03:00
static int hci_resolve_name ( struct hci_dev * hdev ,
2012-03-08 01:25:00 -03:00
struct inquiry_entry * e )
2012-01-04 15:44:20 +02:00
{
struct hci_cp_remote_name_req cp ;
memset ( & cp , 0 , sizeof ( cp ) ) ;
bacpy ( & cp . bdaddr , & e - > data . bdaddr ) ;
cp . pscan_rep_mode = e - > data . pscan_rep_mode ;
cp . pscan_mode = e - > data . pscan_mode ;
cp . clock_offset = e - > data . clock_offset ;
return hci_send_cmd ( hdev , HCI_OP_REMOTE_NAME_REQ , sizeof ( cp ) , & cp ) ;
}
2012-01-17 21:48:47 +02:00
static bool hci_resolve_next_name ( struct hci_dev * hdev )
2012-01-04 15:44:20 +02:00
{
struct discovery_state * discov = & hdev - > discovery ;
struct inquiry_entry * e ;
2012-01-17 21:48:47 +02:00
if ( list_empty ( & discov - > resolve ) )
return false ;
e = hci_inquiry_cache_lookup_resolve ( hdev , BDADDR_ANY , NAME_NEEDED ) ;
2012-07-19 10:26:09 +03:00
if ( ! e )
return false ;
2012-01-17 21:48:47 +02:00
if ( hci_resolve_name ( hdev , e ) = = 0 ) {
e - > name_state = NAME_PENDING ;
return true ;
}
return false ;
}
static void hci_check_pending_name ( struct hci_dev * hdev , struct hci_conn * conn ,
2012-03-08 01:25:00 -03:00
bdaddr_t * bdaddr , u8 * name , u8 name_len )
2012-01-17 21:48:47 +02:00
{
struct discovery_state * discov = & hdev - > discovery ;
struct inquiry_entry * e ;
if ( conn & & ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
2012-03-08 01:25:00 -03:00
mgmt_device_connected ( hdev , bdaddr , ACL_LINK , 0x00 , 0 , name ,
name_len , conn - > dev_class ) ;
2012-01-17 21:48:47 +02:00
if ( discov - > state = = DISCOVERY_STOPPED )
return ;
2012-01-04 15:44:20 +02:00
if ( discov - > state = = DISCOVERY_STOPPING )
goto discov_complete ;
if ( discov - > state ! = DISCOVERY_RESOLVING )
return ;
e = hci_inquiry_cache_lookup_resolve ( hdev , bdaddr , NAME_PENDING ) ;
2012-07-19 10:26:10 +03:00
/* If the device was not found in a list of found devices names of which
* are pending . there is no need to continue resolving a next name as it
* will be done upon receiving another Remote Name Request Complete
* Event */
if ( ! e )
return ;
list_del ( & e - > list ) ;
if ( name ) {
2012-01-04 15:44:20 +02:00
e - > name_state = NAME_KNOWN ;
2012-07-19 10:26:10 +03:00
mgmt_remote_name ( hdev , bdaddr , ACL_LINK , 0x00 ,
e - > data . rssi , name , name_len ) ;
2012-07-19 10:26:11 +03:00
} else {
e - > name_state = NAME_NOT_KNOWN ;
2012-01-04 15:44:20 +02:00
}
2012-01-17 21:48:47 +02:00
if ( hci_resolve_next_name ( hdev ) )
2012-01-04 15:44:20 +02:00
return ;
discov_complete :
hci_discovery_set_state ( hdev , DISCOVERY_STOPPED ) ;
}
2007-10-20 13:33:56 +02:00
static void hci_cs_remote_name_req ( struct hci_dev * hdev , __u8 status )
{
2010-11-18 22:22:29 +02:00
struct hci_cp_remote_name_req * cp ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2010-11-18 22:22:29 +02:00
/* If successful wait for the name req complete event before
* checking for the need to do authentication */
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_REMOTE_NAME_REQ ) ;
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
2012-01-17 21:48:47 +02:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & cp - > bdaddr ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-01-17 21:48:47 +02:00
hci_check_pending_name ( hdev , conn , & cp - > bdaddr , NULL , 0 ) ;
2012-01-04 15:44:20 +02:00
2011-04-28 11:28:55 -07:00
if ( ! conn )
goto unlock ;
if ( ! hci_outgoing_auth_needed ( hdev , conn ) )
goto unlock ;
2012-01-16 06:10:31 +02:00
if ( ! test_and_set_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ) {
2013-10-07 18:19:16 +02:00
struct hci_cp_auth_requested auth_cp ;
auth_cp . handle = __cpu_to_le16 ( conn - > handle ) ;
hci_send_cmd ( hdev , HCI_OP_AUTH_REQUESTED ,
sizeof ( auth_cp ) , & auth_cp ) ;
2010-11-18 22:22:29 +02:00
}
2011-04-28 11:28:55 -07:00
unlock :
2010-11-18 22:22:29 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:49 +02:00
static void hci_cs_read_remote_features ( struct hci_dev * hdev , __u8 status )
{
struct hci_cp_read_remote_features * cp ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 20:13:49 +02:00
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_READ_REMOTE_FEATURES ) ;
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
if ( conn ) {
if ( conn - > state = = BT_CONFIG ) {
hci_proto_connect_cfm ( conn , status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2008-07-14 20:13:49 +02:00
}
}
hci_dev_unlock ( hdev ) ;
}
static void hci_cs_read_remote_ext_features ( struct hci_dev * hdev , __u8 status )
{
struct hci_cp_read_remote_ext_features * cp ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 20:13:49 +02:00
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_READ_REMOTE_EXT_FEATURES ) ;
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
if ( conn ) {
if ( conn - > state = = BT_CONFIG ) {
hci_proto_connect_cfm ( conn , status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2008-07-14 20:13:49 +02:00
}
}
hci_dev_unlock ( hdev ) ;
}
2007-10-20 13:33:56 +02:00
static void hci_cs_setup_sync_conn ( struct hci_dev * hdev , __u8 status )
{
2007-10-20 14:55:10 +02:00
struct hci_cp_setup_sync_conn * cp ;
struct hci_conn * acl , * sco ;
__u16 handle ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 14:55:10 +02:00
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_SETUP_SYNC_CONN ) ;
if ( ! cp )
return ;
handle = __le16_to_cpu ( cp - > handle ) ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s handle 0x%4.4x " , hdev - > name , handle ) ;
2007-10-20 14:55:10 +02:00
hci_dev_lock ( hdev ) ;
acl = hci_conn_hash_lookup_handle ( hdev , handle ) ;
2011-01-11 17:20:20 +02:00
if ( acl ) {
sco = acl - > link ;
if ( sco ) {
sco - > state = BT_CLOSED ;
2007-10-20 14:55:10 +02:00
2011-01-11 17:20:20 +02:00
hci_proto_connect_cfm ( sco , status ) ;
hci_conn_del ( sco ) ;
}
2007-10-20 14:55:10 +02:00
}
hci_dev_unlock ( hdev ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-20 13:33:56 +02:00
static void hci_cs_sniff_mode ( struct hci_dev * hdev , __u8 status )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_cp_sniff_mode * cp ;
struct hci_conn * conn ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
if ( ! status )
return ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_SNIFF_MODE ) ;
if ( ! cp )
return ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
2010-07-26 10:06:00 -04:00
if ( conn ) {
2012-01-16 06:10:31 +02:00
clear_bit ( HCI_CONN_MODE_CHANGE_PEND , & conn - > flags ) ;
2006-07-03 10:02:33 +02:00
2012-01-16 06:10:31 +02:00
if ( test_and_clear_bit ( HCI_CONN_SCO_SETUP_PEND , & conn - > flags ) )
2010-07-26 10:06:00 -04:00
hci_sco_setup ( conn , status ) ;
}
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
}
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
static void hci_cs_exit_sniff_mode ( struct hci_dev * hdev , __u8 status )
{
struct hci_cp_exit_sniff_mode * cp ;
struct hci_conn * conn ;
2006-07-03 10:02:33 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
if ( ! status )
return ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_EXIT_SNIFF_MODE ) ;
if ( ! cp )
return ;
2006-07-03 10:02:33 +02:00
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
2010-07-26 10:06:00 -04:00
if ( conn ) {
2012-01-16 06:10:31 +02:00
clear_bit ( HCI_CONN_MODE_CHANGE_PEND , & conn - > flags ) ;
2005-04-16 15:20:36 -07:00
2012-01-16 06:10:31 +02:00
if ( test_and_clear_bit ( HCI_CONN_SCO_SETUP_PEND , & conn - > flags ) )
2010-07-26 10:06:00 -04:00
hci_sco_setup ( conn , status ) ;
}
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
2005-04-16 15:20:36 -07:00
}
2012-02-09 14:27:38 +02:00
static void hci_cs_disconnect ( struct hci_dev * hdev , u8 status )
{
struct hci_cp_disconnect * cp ;
struct hci_conn * conn ;
if ( ! status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_DISCONNECT ) ;
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
if ( conn )
mgmt_disconnect_failed ( hdev , & conn - > dst , conn - > type ,
2012-03-08 01:25:00 -03:00
conn - > dst_type , status ) ;
2012-02-09 14:27:38 +02:00
hci_dev_unlock ( hdev ) ;
}
2012-09-27 17:26:19 +03:00
static void hci_cs_create_phylink ( struct hci_dev * hdev , u8 status )
{
2012-09-27 17:26:20 +03:00
struct hci_cp_create_phy_link * cp ;
2012-09-27 17:26:19 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2012-09-27 17:26:20 +03:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_CREATE_PHY_LINK ) ;
if ( ! cp )
return ;
2012-10-31 15:46:33 +02:00
hci_dev_lock ( hdev ) ;
if ( status ) {
struct hci_conn * hcon ;
hcon = hci_conn_hash_lookup_handle ( hdev , cp - > phy_handle ) ;
if ( hcon )
hci_conn_del ( hcon ) ;
} else {
amp_write_remote_assoc ( hdev , cp - > phy_handle ) ;
}
hci_dev_unlock ( hdev ) ;
2012-09-27 17:26:19 +03:00
}
2012-09-27 17:26:24 +03:00
static void hci_cs_accept_phylink ( struct hci_dev * hdev , u8 status )
{
struct hci_cp_accept_phy_link * cp ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
if ( status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_ACCEPT_PHY_LINK ) ;
if ( ! cp )
return ;
amp_write_remote_assoc ( hdev , cp - > phy_handle ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_inquiry_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-01-04 15:44:20 +02:00
struct discovery_state * discov = & hdev - > discovery ;
struct inquiry_entry * e ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hci_conn_check_pending ( hdev ) ;
2011-11-04 14:16:53 -03:00
if ( ! test_and_clear_bit ( HCI_INQUIRY , & hdev - > flags ) )
return ;
2013-03-27 20:04:56 -03:00
smp_mb__after_clear_bit ( ) ; /* wake_up_bit advises about this barrier */
wake_up_bit ( & hdev - > flags , HCI_INQUIRY ) ;
2012-01-08 23:11:15 +02:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-01-04 15:44:20 +02:00
return ;
2011-11-08 20:40:16 +02:00
hci_dev_lock ( hdev ) ;
2012-01-04 15:44:20 +02:00
2012-02-17 20:39:37 -03:00
if ( discov - > state ! = DISCOVERY_FINDING )
2012-01-04 15:44:20 +02:00
goto unlock ;
if ( list_empty ( & discov - > resolve ) ) {
hci_discovery_set_state ( hdev , DISCOVERY_STOPPED ) ;
goto unlock ;
}
e = hci_inquiry_cache_lookup_resolve ( hdev , BDADDR_ANY , NAME_NEEDED ) ;
if ( e & & hci_resolve_name ( hdev , e ) = = 0 ) {
e - > name_state = NAME_PENDING ;
hci_discovery_set_state ( hdev , DISCOVERY_RESOLVING ) ;
} else {
hci_discovery_set_state ( hdev , DISCOVERY_STOPPED ) ;
}
unlock :
2011-11-08 20:40:16 +02:00
hci_dev_unlock ( hdev ) ;
2005-04-16 15:20:36 -07:00
}
2012-05-23 04:04:18 -03:00
static void hci_inquiry_result_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2005-08-09 20:27:49 -07:00
struct inquiry_data data ;
2007-10-20 13:33:56 +02:00
struct inquiry_info * info = ( void * ) ( skb - > data + 1 ) ;
2005-04-16 15:20:36 -07:00
int num_rsp = * ( ( __u8 * ) skb - > data ) ;
BT_DBG ( " %s num_rsp %d " , hdev - > name , num_rsp ) ;
2005-08-09 20:27:49 -07:00
if ( ! num_rsp )
return ;
2012-03-21 00:03:38 -03:00
if ( test_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) )
return ;
2005-04-16 15:20:36 -07:00
hci_dev_lock ( hdev ) ;
2005-08-09 20:27:49 -07:00
2011-03-30 23:57:16 +03:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2012-02-23 00:38:59 +02:00
bool name_known , ssp ;
2012-01-04 13:39:52 +02:00
2005-04-16 15:20:36 -07:00
bacpy ( & data . bdaddr , & info - > bdaddr ) ;
data . pscan_rep_mode = info - > pscan_rep_mode ;
data . pscan_period_mode = info - > pscan_period_mode ;
data . pscan_mode = info - > pscan_mode ;
memcpy ( data . dev_class , info - > dev_class , 3 ) ;
data . clock_offset = info - > clock_offset ;
data . rssi = 0x00 ;
2008-07-14 20:13:48 +02:00
data . ssp_mode = 0x00 ;
2012-01-04 13:39:52 +02:00
2012-02-23 00:38:59 +02:00
name_known = hci_inquiry_cache_update ( hdev , & data , false , & ssp ) ;
2011-11-09 13:58:58 +02:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 01:25:00 -03:00
info - > dev_class , 0 , ! name_known , ssp , NULL ,
0 ) ;
2005-04-16 15:20:36 -07:00
}
2005-08-09 20:27:49 -07:00
2005-04-16 15:20:36 -07:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_conn_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_conn_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2005-04-16 15:20:36 -07:00
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ev - > link_type , & ev - > bdaddr ) ;
2009-04-19 19:30:03 +02:00
if ( ! conn ) {
if ( ev - > link_type ! = SCO_LINK )
goto unlock ;
conn = hci_conn_hash_lookup_ba ( hdev , ESCO_LINK , & ev - > bdaddr ) ;
if ( ! conn )
goto unlock ;
conn - > type = SCO_LINK ;
}
2005-04-16 15:20:36 -07:00
if ( ! ev - > status ) {
conn - > handle = __le16_to_cpu ( ev - > handle ) ;
2008-07-14 20:13:49 +02:00
if ( conn - > type = = ACL_LINK ) {
conn - > state = BT_CONFIG ;
hci_conn_hold ( conn ) ;
2012-07-19 14:46:08 +02:00
if ( ! conn - > out & & ! hci_conn_ssp_enabled ( conn ) & &
! hci_find_link_key ( hdev , & ev - > bdaddr ) )
conn - > disc_timeout = HCI_PAIRING_TIMEOUT ;
else
conn - > disc_timeout = HCI_DISCONN_TIMEOUT ;
2008-07-14 20:13:49 +02:00
} else
conn - > state = BT_CONNECTED ;
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:51 +02:00
hci_conn_add_sysfs ( conn ) ;
2005-04-16 15:20:36 -07:00
if ( test_bit ( HCI_AUTH , & hdev - > flags ) )
conn - > link_mode | = HCI_LM_AUTH ;
if ( test_bit ( HCI_ENCRYPT , & hdev - > flags ) )
conn - > link_mode | = HCI_LM_ENCRYPT ;
2006-07-03 10:02:33 +02:00
/* Get remote features */
if ( conn - > type = = ACL_LINK ) {
struct hci_cp_read_remote_features cp ;
cp . handle = ev - > handle ;
2008-07-14 20:13:49 +02:00
hci_send_cmd ( hdev , HCI_OP_READ_REMOTE_FEATURES ,
2012-03-08 01:25:00 -03:00
sizeof ( cp ) , & cp ) ;
2006-07-03 10:02:33 +02:00
}
2005-04-16 15:20:36 -07:00
/* Set packet type for incoming connection */
2011-12-01 14:33:27 +02:00
if ( ! conn - > out & & hdev - > hci_ver < BLUETOOTH_VER_2_0 ) {
2005-04-16 15:20:36 -07:00
struct hci_cp_change_conn_ptype cp ;
cp . handle = ev - > handle ;
2008-07-14 20:13:46 +02:00
cp . pkt_type = cpu_to_le16 ( conn - > pkt_type ) ;
2012-03-08 01:25:00 -03:00
hci_send_cmd ( hdev , HCI_OP_CHANGE_CONN_PTYPE , sizeof ( cp ) ,
& cp ) ;
2005-04-16 15:20:36 -07:00
}
2011-01-22 06:09:08 +02:00
} else {
2005-04-16 15:20:36 -07:00
conn - > state = BT_CLOSED ;
2011-01-22 06:09:08 +02:00
if ( conn - > type = = ACL_LINK )
2014-02-18 14:22:20 -08:00
mgmt_connect_failed ( hdev , & conn - > dst , conn - > type ,
2012-03-08 01:25:00 -03:00
conn - > dst_type , ev - > status ) ;
2011-01-22 06:09:08 +02:00
}
2005-04-16 15:20:36 -07:00
2010-07-26 10:06:00 -04:00
if ( conn - > type = = ACL_LINK )
hci_sco_setup ( conn , ev - > status ) ;
2005-04-16 15:20:36 -07:00
2008-07-14 20:13:49 +02:00
if ( ev - > status ) {
hci_proto_connect_cfm ( conn , ev - > status ) ;
2005-04-16 15:20:36 -07:00
hci_conn_del ( conn ) ;
2009-01-15 21:57:03 +01:00
} else if ( ev - > link_type ! = ACL_LINK )
hci_proto_connect_cfm ( conn , ev - > status ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
unlock :
2005-04-16 15:20:36 -07:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
hci_conn_check_pending ( hdev ) ;
2005-04-16 15:20:36 -07:00
}
2012-05-23 04:04:18 -03:00
static void hci_conn_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_conn_request * ev = ( void * ) skb - > data ;
int mask = hdev - > link_mode ;
2012-11-21 10:51:12 +01:00
__u8 flags = 0 ;
2005-04-16 15:20:36 -07:00
2012-09-25 12:49:43 +03:00
BT_DBG ( " %s bdaddr %pMR type 0x%x " , hdev - > name , & ev - > bdaddr ,
2012-05-17 00:36:24 -03:00
ev - > link_type ) ;
2005-04-16 15:20:36 -07:00
2012-11-21 10:51:12 +01:00
mask | = hci_proto_connect_ind ( hdev , & ev - > bdaddr , ev - > link_type ,
& flags ) ;
2005-04-16 15:20:36 -07:00
2011-02-17 16:44:23 +01:00
if ( ( mask & HCI_LM_ACCEPT ) & &
2013-10-17 17:24:13 -07:00
! hci_blacklist_lookup ( hdev , & ev - > bdaddr , BDADDR_BREDR ) ) {
2007-10-20 13:33:56 +02:00
/* Connection accepted */
2008-07-14 20:13:47 +02:00
struct inquiry_entry * ie ;
2005-04-16 15:20:36 -07:00
struct hci_conn * conn ;
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
2007-10-20 14:55:10 +02:00
2010-11-22 13:21:37 +02:00
ie = hci_inquiry_cache_lookup ( hdev , & ev - > bdaddr ) ;
if ( ie )
2008-07-14 20:13:47 +02:00
memcpy ( ie - > data . dev_class , ev - > dev_class , 3 ) ;
2012-05-23 04:04:21 -03:00
conn = hci_conn_hash_lookup_ba ( hdev , ev - > link_type ,
& ev - > bdaddr ) ;
2007-10-20 13:33:56 +02:00
if ( ! conn ) {
2010-11-22 13:21:37 +02:00
conn = hci_conn_add ( hdev , ev - > link_type , & ev - > bdaddr ) ;
if ( ! conn ) {
2010-07-18 15:13:37 -03:00
BT_ERR ( " No memory for new connection " ) ;
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
return ;
2005-04-16 15:20:36 -07:00
}
}
2007-10-20 14:55:10 +02:00
2007-10-20 13:33:56 +02:00
memcpy ( conn - > dev_class , ev - > dev_class , 3 ) ;
2007-10-20 14:55:10 +02:00
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
2005-04-16 15:20:36 -07:00
2012-11-21 10:51:12 +01:00
if ( ev - > link_type = = ACL_LINK | |
( ! ( flags & HCI_PROTO_DEFER ) & & ! lmp_esco_capable ( hdev ) ) ) {
2007-10-20 14:55:10 +02:00
struct hci_cp_accept_conn_req cp ;
2012-11-21 10:51:12 +01:00
conn - > state = BT_CONNECT ;
2005-04-16 15:20:36 -07:00
2007-10-20 14:55:10 +02:00
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
if ( lmp_rswitch_capable ( hdev ) & & ( mask & HCI_LM_MASTER ) )
cp . role = 0x00 ; /* Become master */
else
cp . role = 0x01 ; /* Remain slave */
2012-03-08 01:25:00 -03:00
hci_send_cmd ( hdev , HCI_OP_ACCEPT_CONN_REQ , sizeof ( cp ) ,
& cp ) ;
2012-11-21 10:51:12 +01:00
} else if ( ! ( flags & HCI_PROTO_DEFER ) ) {
2007-10-20 14:55:10 +02:00
struct hci_cp_accept_sync_conn_req cp ;
2012-11-21 10:51:12 +01:00
conn - > state = BT_CONNECT ;
2007-10-20 14:55:10 +02:00
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2008-07-14 20:13:46 +02:00
cp . pkt_type = cpu_to_le16 ( conn - > pkt_type ) ;
2007-10-20 14:55:10 +02:00
2012-05-25 11:38:27 +03:00
cp . tx_bandwidth = __constant_cpu_to_le32 ( 0x00001f40 ) ;
cp . rx_bandwidth = __constant_cpu_to_le32 ( 0x00001f40 ) ;
cp . max_latency = __constant_cpu_to_le16 ( 0xffff ) ;
2007-10-20 14:55:10 +02:00
cp . content_format = cpu_to_le16 ( hdev - > voice_setting ) ;
cp . retrans_effort = 0xff ;
2005-04-16 15:20:36 -07:00
2007-10-20 14:55:10 +02:00
hci_send_cmd ( hdev , HCI_OP_ACCEPT_SYNC_CONN_REQ ,
2012-03-08 01:25:00 -03:00
sizeof ( cp ) , & cp ) ;
2012-11-21 10:51:12 +01:00
} else {
conn - > state = BT_CONNECT2 ;
hci_proto_connect_cfm ( conn , 0 ) ;
2007-10-20 14:55:10 +02:00
}
2007-10-20 13:33:56 +02:00
} else {
/* Connection rejected */
struct hci_cp_reject_conn_req cp ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2011-11-07 14:20:25 +02:00
cp . reason = HCI_ERROR_REJ_BAD_ADDR ;
2007-10-20 13:33:56 +02:00
hci_send_cmd ( hdev , HCI_OP_REJECT_CONN_REQ , sizeof ( cp ) , & cp ) ;
2005-04-16 15:20:36 -07:00
}
}
2012-08-09 09:52:30 +02:00
static u8 hci_to_mgmt_reason ( u8 err )
{
switch ( err ) {
case HCI_ERROR_CONNECTION_TIMEOUT :
return MGMT_DEV_DISCONN_TIMEOUT ;
case HCI_ERROR_REMOTE_USER_TERM :
case HCI_ERROR_REMOTE_LOW_RESOURCES :
case HCI_ERROR_REMOTE_POWER_OFF :
return MGMT_DEV_DISCONN_REMOTE ;
case HCI_ERROR_LOCAL_HOST_TERM :
return MGMT_DEV_DISCONN_LOCAL_HOST ;
default :
return MGMT_DEV_DISCONN_UNKNOWN ;
}
}
2012-05-23 04:04:18 -03:00
static void hci_disconn_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2006-07-03 10:02:33 +02:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_disconn_complete * ev = ( void * ) skb - > data ;
2013-11-07 17:36:09 -03:00
u8 reason = hci_to_mgmt_reason ( ev - > reason ) ;
2006-07-03 10:02:33 +02:00
struct hci_conn * conn ;
2014-02-24 14:52:18 +02:00
bool mgmt_connected ;
2013-11-07 17:36:10 -03:00
u8 type ;
2006-07-03 10:02:33 +02:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2006-07-03 10:02:33 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2011-01-20 12:34:39 +02:00
if ( ! conn )
goto unlock ;
2008-07-14 20:13:51 +02:00
2013-11-07 17:36:09 -03:00
if ( ev - > status ) {
mgmt_disconnect_failed ( hdev , & conn - > dst , conn - > type ,
conn - > dst_type , ev - > status ) ;
goto unlock ;
2011-11-10 15:54:39 +02:00
}
2011-01-20 12:34:39 +02:00
2013-11-07 17:36:10 -03:00
conn - > state = BT_CLOSED ;
2014-02-24 14:52:18 +02:00
mgmt_connected = test_and_clear_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) ;
mgmt_device_disconnected ( hdev , & conn - > dst , conn - > type , conn - > dst_type ,
reason , mgmt_connected ) ;
2013-11-07 17:36:09 -03:00
2013-11-07 17:36:10 -03:00
if ( conn - > type = = ACL_LINK & & conn - > flush_key )
hci_remove_link_key ( hdev , & conn - > dst ) ;
2013-10-05 12:01:06 +02:00
2013-11-07 17:36:10 -03:00
type = conn - > type ;
2013-10-05 12:01:06 +02:00
2013-11-07 17:36:10 -03:00
hci_proto_disconn_cfm ( conn , ev - > reason ) ;
hci_conn_del ( conn ) ;
/* Re-enable advertising if necessary, since it might
* have been disabled by the connection . From the
* HCI_LE_Set_Advertise_Enable command description in
* the core specification ( v4 .0 ) :
* " The Controller shall continue advertising until the Host
* issues an LE_Set_Advertise_Enable command with
* Advertising_Enable set to 0x00 ( Advertising is disabled )
* or until a connection is created or until the Advertising
* is timed out due to Directed Advertising . "
*/
if ( type = = LE_LINK )
mgmt_reenable_advertising ( hdev ) ;
2011-01-20 12:34:39 +02:00
unlock :
2006-07-03 10:02:33 +02:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_auth_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_auth_complete * ev = ( void * ) skb - > data ;
2006-07-03 10:02:33 +02:00
struct hci_conn * conn ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-16 15:20:36 -07:00
hci_dev_lock ( hdev ) ;
2006-07-03 10:02:33 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2011-05-31 15:49:26 +02:00
if ( ! conn )
goto unlock ;
if ( ! ev - > status ) {
2012-01-18 21:33:12 +02:00
if ( ! hci_conn_ssp_enabled ( conn ) & &
2012-05-17 00:36:24 -03:00
test_bit ( HCI_CONN_REAUTH_PEND , & conn - > flags ) ) {
2011-05-31 15:49:26 +02:00
BT_INFO ( " re-auth of legacy device is not possible. " ) ;
2011-02-19 12:06:00 -03:00
} else {
2011-05-31 15:49:26 +02:00
conn - > link_mode | = HCI_LM_AUTH ;
conn - > sec_level = conn - > pending_sec_level ;
2011-02-19 12:06:00 -03:00
}
2011-05-31 15:49:26 +02:00
} else {
2012-02-09 16:07:29 +02:00
mgmt_auth_failed ( hdev , & conn - > dst , conn - > type , conn - > dst_type ,
2012-03-08 01:25:00 -03:00
ev - > status ) ;
2011-05-31 15:49:26 +02:00
}
2005-04-16 15:20:36 -07:00
2012-01-16 06:10:31 +02:00
clear_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ;
clear_bit ( HCI_CONN_REAUTH_PEND , & conn - > flags ) ;
2005-04-16 15:20:36 -07:00
2011-05-31 15:49:26 +02:00
if ( conn - > state = = BT_CONFIG ) {
2012-01-18 21:33:12 +02:00
if ( ! ev - > status & & hci_conn_ssp_enabled ( conn ) ) {
2011-05-31 15:49:26 +02:00
struct hci_cp_set_conn_encrypt cp ;
cp . handle = ev - > handle ;
cp . encrypt = 0x01 ;
hci_send_cmd ( hdev , HCI_OP_SET_CONN_ENCRYPT , sizeof ( cp ) ,
2012-05-17 00:36:24 -03:00
& cp ) ;
2009-04-26 20:01:22 +02:00
} else {
2011-05-31 15:49:26 +02:00
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2009-04-26 20:01:22 +02:00
}
2011-05-31 15:49:26 +02:00
} else {
hci_auth_cfm ( conn , ev - > status ) ;
2009-04-26 20:01:22 +02:00
2011-05-31 15:49:26 +02:00
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_DISCONN_TIMEOUT ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2011-05-31 15:49:26 +02:00
}
2012-01-16 06:10:31 +02:00
if ( test_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ) {
2011-05-31 15:49:26 +02:00
if ( ! ev - > status ) {
struct hci_cp_set_conn_encrypt cp ;
cp . handle = ev - > handle ;
cp . encrypt = 0x01 ;
hci_send_cmd ( hdev , HCI_OP_SET_CONN_ENCRYPT , sizeof ( cp ) ,
2012-05-17 00:36:24 -03:00
& cp ) ;
2011-05-31 15:49:26 +02:00
} else {
2012-01-16 06:10:31 +02:00
clear_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ;
2011-05-31 15:49:26 +02:00
hci_encrypt_cfm ( conn , ev - > status , 0x00 ) ;
2005-04-16 15:20:36 -07:00
}
}
2011-05-31 15:49:26 +02:00
unlock :
2005-04-16 15:20:36 -07:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_remote_name_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2010-11-18 22:22:29 +02:00
struct hci_ev_remote_name * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s " , hdev - > name ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
hci_conn_check_pending ( hdev ) ;
2010-11-18 22:22:29 +02:00
hci_dev_lock ( hdev ) ;
2012-01-17 21:48:47 +02:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2012-01-04 15:44:20 +02:00
2012-01-17 21:48:47 +02:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
goto check_auth ;
2011-03-30 13:18:12 +03:00
2012-01-17 21:48:47 +02:00
if ( ev - > status = = 0 )
hci_check_pending_name ( hdev , conn , & ev - > bdaddr , ev - > name ,
2012-03-08 01:25:00 -03:00
strnlen ( ev - > name , HCI_MAX_NAME_LENGTH ) ) ;
2012-01-17 21:48:47 +02:00
else
hci_check_pending_name ( hdev , conn , & ev - > bdaddr , NULL , 0 ) ;
check_auth :
2011-04-28 11:28:55 -07:00
if ( ! conn )
goto unlock ;
if ( ! hci_outgoing_auth_needed ( hdev , conn ) )
goto unlock ;
2012-01-16 06:10:31 +02:00
if ( ! test_and_set_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ) {
2010-11-18 22:22:29 +02:00
struct hci_cp_auth_requested cp ;
cp . handle = __cpu_to_le16 ( conn - > handle ) ;
hci_send_cmd ( hdev , HCI_OP_AUTH_REQUESTED , sizeof ( cp ) , & cp ) ;
}
2011-04-28 11:28:55 -07:00
unlock :
2010-11-18 22:22:29 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2012-05-23 04:04:18 -03:00
static void hci_encrypt_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_ev_encrypt_change * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-16 15:20:36 -07:00
hci_dev_lock ( hdev ) ;
2006-07-03 10:02:33 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2014-01-31 16:24:27 -08:00
if ( ! conn )
goto unlock ;
2005-04-16 15:20:36 -07:00
2014-01-31 16:24:27 -08:00
if ( ! ev - > status ) {
if ( ev - > encrypt ) {
/* Encryption implies authentication */
conn - > link_mode | = HCI_LM_AUTH ;
conn - > link_mode | = HCI_LM_ENCRYPT ;
conn - > sec_level = conn - > pending_sec_level ;
2014-01-31 16:24:28 -08:00
2014-02-01 11:52:02 -08:00
/* P-256 authentication key implies FIPS */
if ( conn - > key_type = = HCI_LK_AUTH_COMBINATION_P256 )
conn - > link_mode | = HCI_LM_FIPS ;
2014-01-31 16:24:28 -08:00
if ( ( conn - > type = = ACL_LINK & & ev - > encrypt = = 0x02 ) | |
conn - > type = = LE_LINK )
set_bit ( HCI_CONN_AES_CCM , & conn - > flags ) ;
} else {
2014-01-31 16:24:27 -08:00
conn - > link_mode & = ~ HCI_LM_ENCRYPT ;
2014-01-31 16:24:28 -08:00
clear_bit ( HCI_CONN_AES_CCM , & conn - > flags ) ;
}
2014-01-31 16:24:27 -08:00
}
2012-05-13 03:20:07 -03:00
2014-01-31 16:24:27 -08:00
clear_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ;
2008-07-14 20:13:49 +02:00
2014-01-31 16:24:27 -08:00
if ( ev - > status & & conn - > state = = BT_CONNECTED ) {
hci_disconnect ( conn , HCI_ERROR_AUTH_FAILURE ) ;
hci_conn_drop ( conn ) ;
goto unlock ;
2005-04-16 15:20:36 -07:00
}
2014-01-31 16:24:27 -08:00
if ( conn - > state = = BT_CONFIG ) {
if ( ! ev - > status )
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
hci_conn_drop ( conn ) ;
} else
hci_encrypt_cfm ( conn , ev - > status , ev - > encrypt ) ;
2012-05-13 03:20:07 -03:00
unlock :
2005-04-16 15:20:36 -07:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_change_link_key_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_change_link_key_complete * ev = ( void * ) skb - > data ;
2006-07-03 10:02:33 +02:00
struct hci_conn * conn ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-16 15:20:36 -07:00
hci_dev_lock ( hdev ) ;
2006-07-03 10:02:33 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2005-04-16 15:20:36 -07:00
if ( conn ) {
if ( ! ev - > status )
conn - > link_mode | = HCI_LM_SECURE ;
2012-01-16 06:10:31 +02:00
clear_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ;
2005-04-16 15:20:36 -07:00
hci_key_change_cfm ( conn , ev - > status ) ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_remote_features_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_remote_features * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2010-11-10 17:11:51 +02:00
if ( ! conn )
goto unlock ;
2008-07-14 20:13:49 +02:00
2010-11-10 17:11:51 +02:00
if ( ! ev - > status )
2013-04-17 15:00:51 +03:00
memcpy ( conn - > features [ 0 ] , ev - > features , 8 ) ;
2010-11-10 17:11:51 +02:00
if ( conn - > state ! = BT_CONFIG )
goto unlock ;
if ( ! ev - > status & & lmp_ssp_capable ( hdev ) & & lmp_ssp_capable ( conn ) ) {
struct hci_cp_read_remote_ext_features cp ;
cp . handle = ev - > handle ;
cp . page = 0x01 ;
hci_send_cmd ( hdev , HCI_OP_READ_REMOTE_EXT_FEATURES ,
2012-05-17 00:36:24 -03:00
sizeof ( cp ) , & cp ) ;
2010-11-18 22:22:28 +02:00
goto unlock ;
}
2012-05-12 16:11:50 -03:00
if ( ! ev - > status & & ! test_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) ) {
2010-11-18 22:22:29 +02:00
struct hci_cp_remote_name_req cp ;
memset ( & cp , 0 , sizeof ( cp ) ) ;
bacpy ( & cp . bdaddr , & conn - > dst ) ;
cp . pscan_rep_mode = 0x02 ;
hci_send_cmd ( hdev , HCI_OP_REMOTE_NAME_REQ , sizeof ( cp ) , & cp ) ;
2012-01-17 21:48:47 +02:00
} else if ( ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
mgmt_device_connected ( hdev , & conn - > dst , conn - > type ,
2012-03-08 01:25:00 -03:00
conn - > dst_type , 0 , NULL , 0 ,
conn - > dev_class ) ;
2010-11-18 22:22:28 +02:00
2010-11-18 22:22:29 +02:00
if ( ! hci_outgoing_auth_needed ( hdev , conn ) ) {
2010-11-10 17:11:51 +02:00
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2008-07-14 20:13:49 +02:00
}
2007-10-20 13:33:56 +02:00
2010-11-10 17:11:51 +02:00
unlock :
2007-10-20 13:33:56 +02:00
hci_dev_unlock ( hdev ) ;
2005-04-16 15:20:36 -07:00
}
2012-05-23 04:04:18 -03:00
static void hci_cmd_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_ev_cmd_complete * ev = ( void * ) skb - > data ;
2013-03-05 20:37:48 +02:00
u8 status = skb - > data [ sizeof ( * ev ) ] ;
2007-10-20 13:33:56 +02:00
__u16 opcode ;
skb_pull ( skb , sizeof ( * ev ) ) ;
opcode = __le16_to_cpu ( ev - > opcode ) ;
switch ( opcode ) {
case HCI_OP_INQUIRY_CANCEL :
hci_cc_inquiry_cancel ( hdev , skb ) ;
break ;
2012-03-21 00:03:35 -03:00
case HCI_OP_PERIODIC_INQ :
hci_cc_periodic_inq ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_EXIT_PERIODIC_INQ :
hci_cc_exit_periodic_inq ( hdev , skb ) ;
break ;
case HCI_OP_REMOTE_NAME_REQ_CANCEL :
hci_cc_remote_name_req_cancel ( hdev , skb ) ;
break ;
case HCI_OP_ROLE_DISCOVERY :
hci_cc_role_discovery ( hdev , skb ) ;
break ;
2008-07-14 20:13:47 +02:00
case HCI_OP_READ_LINK_POLICY :
hci_cc_read_link_policy ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_WRITE_LINK_POLICY :
hci_cc_write_link_policy ( hdev , skb ) ;
break ;
2008-07-14 20:13:47 +02:00
case HCI_OP_READ_DEF_LINK_POLICY :
hci_cc_read_def_link_policy ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_DEF_LINK_POLICY :
hci_cc_write_def_link_policy ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_RESET :
hci_cc_reset ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_LOCAL_NAME :
hci_cc_write_local_name ( hdev , skb ) ;
break ;
case HCI_OP_READ_LOCAL_NAME :
hci_cc_read_local_name ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_AUTH_ENABLE :
hci_cc_write_auth_enable ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_ENCRYPT_MODE :
hci_cc_write_encrypt_mode ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_SCAN_ENABLE :
hci_cc_write_scan_enable ( hdev , skb ) ;
break ;
case HCI_OP_READ_CLASS_OF_DEV :
hci_cc_read_class_of_dev ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_CLASS_OF_DEV :
hci_cc_write_class_of_dev ( hdev , skb ) ;
break ;
case HCI_OP_READ_VOICE_SETTING :
hci_cc_read_voice_setting ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_VOICE_SETTING :
hci_cc_write_voice_setting ( hdev , skb ) ;
break ;
2013-10-14 13:56:16 -07:00
case HCI_OP_READ_NUM_SUPPORTED_IAC :
hci_cc_read_num_supported_iac ( hdev , skb ) ;
break ;
2008-07-14 20:13:48 +02:00
case HCI_OP_WRITE_SSP_MODE :
hci_cc_write_ssp_mode ( hdev , skb ) ;
break ;
2014-01-10 02:07:23 -08:00
case HCI_OP_WRITE_SC_SUPPORT :
hci_cc_write_sc_support ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_READ_LOCAL_VERSION :
hci_cc_read_local_version ( hdev , skb ) ;
break ;
case HCI_OP_READ_LOCAL_COMMANDS :
hci_cc_read_local_commands ( hdev , skb ) ;
break ;
case HCI_OP_READ_LOCAL_FEATURES :
hci_cc_read_local_features ( hdev , skb ) ;
break ;
2011-06-30 19:20:52 -03:00
case HCI_OP_READ_LOCAL_EXT_FEATURES :
hci_cc_read_local_ext_features ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_READ_BUFFER_SIZE :
hci_cc_read_buffer_size ( hdev , skb ) ;
break ;
case HCI_OP_READ_BD_ADDR :
hci_cc_read_bd_addr ( hdev , skb ) ;
break ;
2013-03-15 17:07:11 -05:00
case HCI_OP_READ_PAGE_SCAN_ACTIVITY :
hci_cc_read_page_scan_activity ( hdev , skb ) ;
break ;
2013-03-15 17:07:12 -05:00
case HCI_OP_WRITE_PAGE_SCAN_ACTIVITY :
hci_cc_write_page_scan_activity ( hdev , skb ) ;
break ;
2013-03-15 17:07:11 -05:00
case HCI_OP_READ_PAGE_SCAN_TYPE :
hci_cc_read_page_scan_type ( hdev , skb ) ;
break ;
2013-03-15 17:07:12 -05:00
case HCI_OP_WRITE_PAGE_SCAN_TYPE :
hci_cc_write_page_scan_type ( hdev , skb ) ;
break ;
2011-12-07 15:56:51 +02:00
case HCI_OP_READ_DATA_BLOCK_SIZE :
hci_cc_read_data_block_size ( hdev , skb ) ;
break ;
2011-11-24 14:52:02 +02:00
case HCI_OP_READ_FLOW_CONTROL_MODE :
hci_cc_read_flow_control_mode ( hdev , skb ) ;
break ;
2011-10-12 10:53:57 +03:00
case HCI_OP_READ_LOCAL_AMP_INFO :
hci_cc_read_local_amp_info ( hdev , skb ) ;
break ;
2012-09-27 17:26:09 +03:00
case HCI_OP_READ_LOCAL_AMP_ASSOC :
hci_cc_read_local_amp_assoc ( hdev , skb ) ;
break ;
2011-01-25 01:19:58 +02:00
case HCI_OP_READ_INQ_RSP_TX_POWER :
hci_cc_read_inq_rsp_tx_power ( hdev , skb ) ;
break ;
2011-01-22 06:10:07 +02:00
case HCI_OP_PIN_CODE_REPLY :
hci_cc_pin_code_reply ( hdev , skb ) ;
break ;
case HCI_OP_PIN_CODE_NEG_REPLY :
hci_cc_pin_code_neg_reply ( hdev , skb ) ;
break ;
2011-03-22 13:12:21 +01:00
case HCI_OP_READ_LOCAL_OOB_DATA :
2014-01-10 02:07:26 -08:00
hci_cc_read_local_oob_data ( hdev , skb ) ;
break ;
case HCI_OP_READ_LOCAL_OOB_EXT_DATA :
hci_cc_read_local_oob_ext_data ( hdev , skb ) ;
2011-03-22 13:12:21 +01:00
break ;
2011-02-10 22:38:48 -03:00
case HCI_OP_LE_READ_BUFFER_SIZE :
hci_cc_le_read_buffer_size ( hdev , skb ) ;
break ;
2013-01-22 14:01:59 +02:00
case HCI_OP_LE_READ_LOCAL_FEATURES :
hci_cc_le_read_local_features ( hdev , skb ) ;
break ;
2012-10-19 20:57:49 +03:00
case HCI_OP_LE_READ_ADV_TX_POWER :
hci_cc_le_read_adv_tx_power ( hdev , skb ) ;
break ;
2011-02-19 12:05:57 -03:00
case HCI_OP_USER_CONFIRM_REPLY :
hci_cc_user_confirm_reply ( hdev , skb ) ;
break ;
case HCI_OP_USER_CONFIRM_NEG_REPLY :
hci_cc_user_confirm_neg_reply ( hdev , skb ) ;
break ;
2011-11-23 08:28:34 -08:00
case HCI_OP_USER_PASSKEY_REPLY :
hci_cc_user_passkey_reply ( hdev , skb ) ;
break ;
case HCI_OP_USER_PASSKEY_NEG_REPLY :
hci_cc_user_passkey_neg_reply ( hdev , skb ) ;
2012-04-13 12:32:42 +02:00
break ;
2011-12-02 21:13:31 +09:00
2014-02-19 19:52:13 -08:00
case HCI_OP_LE_SET_RANDOM_ADDR :
hci_cc_le_set_random_addr ( hdev , skb ) ;
break ;
2012-11-08 01:23:01 +01:00
case HCI_OP_LE_SET_ADV_ENABLE :
hci_cc_le_set_adv_enable ( hdev , skb ) ;
break ;
2011-05-26 16:23:52 -03:00
case HCI_OP_LE_SET_SCAN_ENABLE :
hci_cc_le_set_scan_enable ( hdev , skb ) ;
break ;
2013-01-22 14:02:00 +02:00
case HCI_OP_LE_READ_WHITE_LIST_SIZE :
hci_cc_le_read_white_list_size ( hdev , skb ) ;
break ;
2013-01-22 14:02:01 +02:00
case HCI_OP_LE_READ_SUPPORTED_STATES :
hci_cc_le_read_supported_states ( hdev , skb ) ;
break ;
2011-06-30 19:20:53 -03:00
case HCI_OP_WRITE_LE_HOST_SUPPORTED :
hci_cc_write_le_host_supported ( hdev , skb ) ;
break ;
2012-09-27 17:26:20 +03:00
case HCI_OP_WRITE_REMOTE_AMP_ASSOC :
hci_cc_write_remote_amp_assoc ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
default :
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s opcode 0x%4.4x " , hdev - > name , opcode ) ;
2007-10-20 13:33:56 +02:00
break ;
}
2013-03-09 09:53:50 +02:00
if ( opcode ! = HCI_OP_NOP )
2011-02-16 16:32:41 +02:00
del_timer ( & hdev - > cmd_timer ) ;
2013-03-09 09:53:50 +02:00
hci_req_cmd_complete ( hdev , opcode , status ) ;
2013-03-05 20:37:48 +02:00
2012-12-11 08:51:19 +01:00
if ( ev - > ncmd & & ! test_bit ( HCI_RESET , & hdev - > flags ) ) {
2007-10-20 13:33:56 +02:00
atomic_set ( & hdev - > cmd_cnt , 1 ) ;
if ( ! skb_queue_empty ( & hdev - > cmd_q ) )
2011-12-14 23:53:47 -02:00
queue_work ( hdev - > workqueue , & hdev - > cmd_work ) ;
2007-10-20 13:33:56 +02:00
}
}
2012-05-23 04:04:18 -03:00
static void hci_cmd_status_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_ev_cmd_status * ev = ( void * ) skb - > data ;
__u16 opcode ;
skb_pull ( skb , sizeof ( * ev ) ) ;
opcode = __le16_to_cpu ( ev - > opcode ) ;
switch ( opcode ) {
case HCI_OP_INQUIRY :
hci_cs_inquiry ( hdev , ev - > status ) ;
break ;
case HCI_OP_CREATE_CONN :
hci_cs_create_conn ( hdev , ev - > status ) ;
break ;
case HCI_OP_ADD_SCO :
hci_cs_add_sco ( hdev , ev - > status ) ;
break ;
2008-07-14 20:13:49 +02:00
case HCI_OP_AUTH_REQUESTED :
hci_cs_auth_requested ( hdev , ev - > status ) ;
break ;
case HCI_OP_SET_CONN_ENCRYPT :
hci_cs_set_conn_encrypt ( hdev , ev - > status ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_REMOTE_NAME_REQ :
hci_cs_remote_name_req ( hdev , ev - > status ) ;
break ;
2008-07-14 20:13:49 +02:00
case HCI_OP_READ_REMOTE_FEATURES :
hci_cs_read_remote_features ( hdev , ev - > status ) ;
break ;
case HCI_OP_READ_REMOTE_EXT_FEATURES :
hci_cs_read_remote_ext_features ( hdev , ev - > status ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_OP_SETUP_SYNC_CONN :
hci_cs_setup_sync_conn ( hdev , ev - > status ) ;
break ;
case HCI_OP_SNIFF_MODE :
hci_cs_sniff_mode ( hdev , ev - > status ) ;
break ;
case HCI_OP_EXIT_SNIFF_MODE :
hci_cs_exit_sniff_mode ( hdev , ev - > status ) ;
break ;
2011-01-20 12:40:27 +02:00
case HCI_OP_DISCONNECT :
2012-02-09 14:27:38 +02:00
hci_cs_disconnect ( hdev , ev - > status ) ;
2011-01-20 12:40:27 +02:00
break ;
2012-09-27 17:26:19 +03:00
case HCI_OP_CREATE_PHY_LINK :
hci_cs_create_phylink ( hdev , ev - > status ) ;
break ;
2012-09-27 17:26:24 +03:00
case HCI_OP_ACCEPT_PHY_LINK :
hci_cs_accept_phylink ( hdev , ev - > status ) ;
break ;
2007-10-20 13:33:56 +02:00
default :
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s opcode 0x%4.4x " , hdev - > name , opcode ) ;
2007-10-20 13:33:56 +02:00
break ;
}
2013-03-09 09:53:50 +02:00
if ( opcode ! = HCI_OP_NOP )
2011-02-16 16:32:41 +02:00
del_timer ( & hdev - > cmd_timer ) ;
2013-04-03 21:50:29 +03:00
if ( ev - > status | |
( hdev - > sent_cmd & & ! bt_cb ( hdev - > sent_cmd ) - > req . event ) )
hci_req_cmd_complete ( hdev , opcode , ev - > status ) ;
2013-03-05 20:37:48 +02:00
2011-03-16 15:36:29 -03:00
if ( ev - > ncmd & & ! test_bit ( HCI_RESET , & hdev - > flags ) ) {
2007-10-20 13:33:56 +02:00
atomic_set ( & hdev - > cmd_cnt , 1 ) ;
if ( ! skb_queue_empty ( & hdev - > cmd_q ) )
2011-12-14 23:53:47 -02:00
queue_work ( hdev - > workqueue , & hdev - > cmd_work ) ;
2007-10-20 13:33:56 +02:00
}
}
2012-05-23 04:04:18 -03:00
static void hci_role_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_ev_role_change * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( conn ) {
if ( ! ev - > status ) {
if ( ev - > role )
conn - > link_mode & = ~ HCI_LM_MASTER ;
else
conn - > link_mode | = HCI_LM_MASTER ;
}
2012-01-16 06:10:31 +02:00
clear_bit ( HCI_CONN_RSWITCH_PEND , & conn - > flags ) ;
2007-10-20 13:33:56 +02:00
hci_role_switch_cfm ( conn , ev - > status , ev - > role ) ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_num_comp_pkts_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct hci_ev_num_comp_pkts * ev = ( void * ) skb - > data ;
int i ;
2011-12-19 16:31:29 +02:00
if ( hdev - > flow_ctl_mode ! = HCI_FLOW_CTL_MODE_PACKET_BASED ) {
BT_ERR ( " Wrong event for mode %d " , hdev - > flow_ctl_mode ) ;
return ;
}
2011-12-30 12:07:47 +02:00
if ( skb - > len < sizeof ( * ev ) | | skb - > len < sizeof ( * ev ) +
2012-05-17 00:36:24 -03:00
ev - > num_hndl * sizeof ( struct hci_comp_pkts_info ) ) {
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s bad parameters " , hdev - > name ) ;
return ;
}
2011-12-30 12:07:47 +02:00
BT_DBG ( " %s num_hndl %d " , hdev - > name , ev - > num_hndl ) ;
2011-12-19 16:31:30 +02:00
for ( i = 0 ; i < ev - > num_hndl ; i + + ) {
struct hci_comp_pkts_info * info = & ev - > handles [ i ] ;
2007-10-20 13:33:56 +02:00
struct hci_conn * conn ;
__u16 handle , count ;
2011-12-19 16:31:30 +02:00
handle = __le16_to_cpu ( info - > handle ) ;
count = __le16_to_cpu ( info - > count ) ;
2007-10-20 13:33:56 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , handle ) ;
2011-12-07 15:56:52 +02:00
if ( ! conn )
continue ;
conn - > sent - = count ;
switch ( conn - > type ) {
case ACL_LINK :
hdev - > acl_cnt + = count ;
if ( hdev - > acl_cnt > hdev - > acl_pkts )
hdev - > acl_cnt = hdev - > acl_pkts ;
break ;
case LE_LINK :
if ( hdev - > le_pkts ) {
hdev - > le_cnt + = count ;
if ( hdev - > le_cnt > hdev - > le_pkts )
hdev - > le_cnt = hdev - > le_pkts ;
} else {
2010-12-01 16:58:25 +02:00
hdev - > acl_cnt + = count ;
if ( hdev - > acl_cnt > hdev - > acl_pkts )
2007-10-20 13:33:56 +02:00
hdev - > acl_cnt = hdev - > acl_pkts ;
}
2011-12-07 15:56:52 +02:00
break ;
case SCO_LINK :
hdev - > sco_cnt + = count ;
if ( hdev - > sco_cnt > hdev - > sco_pkts )
hdev - > sco_cnt = hdev - > sco_pkts ;
break ;
default :
BT_ERR ( " Unknown type %d conn %p " , conn - > type , conn ) ;
break ;
2007-10-20 13:33:56 +02:00
}
}
2011-12-15 00:50:02 -02:00
queue_work ( hdev - > workqueue , & hdev - > tx_work ) ;
2007-10-20 13:33:56 +02:00
}
2012-10-10 17:38:29 +03:00
static struct hci_conn * __hci_conn_lookup_handle ( struct hci_dev * hdev ,
__u16 handle )
{
struct hci_chan * chan ;
switch ( hdev - > dev_type ) {
case HCI_BREDR :
return hci_conn_hash_lookup_handle ( hdev , handle ) ;
case HCI_AMP :
chan = hci_chan_lookup_handle ( hdev , handle ) ;
if ( chan )
return chan - > conn ;
break ;
default :
BT_ERR ( " %s unknown dev_type %d " , hdev - > name , hdev - > dev_type ) ;
break ;
}
return NULL ;
}
2012-05-23 04:04:18 -03:00
static void hci_num_comp_blocks_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2012-01-04 12:41:58 +02:00
{
struct hci_ev_num_comp_blocks * ev = ( void * ) skb - > data ;
int i ;
if ( hdev - > flow_ctl_mode ! = HCI_FLOW_CTL_MODE_BLOCK_BASED ) {
BT_ERR ( " Wrong event for mode %d " , hdev - > flow_ctl_mode ) ;
return ;
}
if ( skb - > len < sizeof ( * ev ) | | skb - > len < sizeof ( * ev ) +
2012-05-17 00:36:24 -03:00
ev - > num_hndl * sizeof ( struct hci_comp_blocks_info ) ) {
2012-01-04 12:41:58 +02:00
BT_DBG ( " %s bad parameters " , hdev - > name ) ;
return ;
}
BT_DBG ( " %s num_blocks %d num_hndl %d " , hdev - > name , ev - > num_blocks ,
2012-05-17 00:36:24 -03:00
ev - > num_hndl ) ;
2012-01-04 12:41:58 +02:00
for ( i = 0 ; i < ev - > num_hndl ; i + + ) {
struct hci_comp_blocks_info * info = & ev - > handles [ i ] ;
2012-10-10 17:38:29 +03:00
struct hci_conn * conn = NULL ;
2012-01-04 12:41:58 +02:00
__u16 handle , block_count ;
handle = __le16_to_cpu ( info - > handle ) ;
block_count = __le16_to_cpu ( info - > blocks ) ;
2012-10-10 17:38:29 +03:00
conn = __hci_conn_lookup_handle ( hdev , handle ) ;
2012-01-04 12:41:58 +02:00
if ( ! conn )
continue ;
conn - > sent - = block_count ;
switch ( conn - > type ) {
case ACL_LINK :
2012-10-10 17:38:30 +03:00
case AMP_LINK :
2012-01-04 12:41:58 +02:00
hdev - > block_cnt + = block_count ;
if ( hdev - > block_cnt > hdev - > num_blocks )
hdev - > block_cnt = hdev - > num_blocks ;
break ;
default :
BT_ERR ( " Unknown type %d conn %p " , conn - > type , conn ) ;
break ;
}
}
queue_work ( hdev - > workqueue , & hdev - > tx_work ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_mode_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2006-07-03 10:02:33 +02:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_mode_change * ev = ( void * ) skb - > data ;
2006-07-03 10:02:33 +02:00
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2006-07-03 10:02:33 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2007-10-20 13:33:56 +02:00
if ( conn ) {
conn - > mode = ev - > mode ;
2012-05-23 04:04:21 -03:00
if ( ! test_and_clear_bit ( HCI_CONN_MODE_CHANGE_PEND ,
& conn - > flags ) ) {
2007-10-20 13:33:56 +02:00
if ( conn - > mode = = HCI_CM_ACTIVE )
2012-01-16 06:47:28 +02:00
set_bit ( HCI_CONN_POWER_SAVE , & conn - > flags ) ;
2007-10-20 13:33:56 +02:00
else
2012-01-16 06:47:28 +02:00
clear_bit ( HCI_CONN_POWER_SAVE , & conn - > flags ) ;
2007-10-20 13:33:56 +02:00
}
2010-07-26 10:06:00 -04:00
2012-01-16 06:10:31 +02:00
if ( test_and_clear_bit ( HCI_CONN_SCO_SETUP_PEND , & conn - > flags ) )
2010-07-26 10:06:00 -04:00
hci_sco_setup ( conn , ev - > status ) ;
2006-07-03 10:02:33 +02:00
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_pin_code_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
2009-04-26 20:01:22 +02:00
struct hci_ev_pin_code_req * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s " , hdev - > name ) ;
2009-04-26 20:01:22 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2011-09-23 10:01:30 +02:00
if ( ! conn )
goto unlock ;
if ( conn - > state = = BT_CONNECTED ) {
2009-04-26 20:01:22 +02:00
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_PAIRING_TIMEOUT ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2009-04-26 20:01:22 +02:00
}
2012-01-08 23:11:15 +02:00
if ( ! test_bit ( HCI_PAIRABLE , & hdev - > dev_flags ) )
2011-01-04 15:40:05 +02:00
hci_send_cmd ( hdev , HCI_OP_PIN_CODE_NEG_REPLY ,
2012-05-17 00:36:24 -03:00
sizeof ( ev - > bdaddr ) , & ev - > bdaddr ) ;
2012-01-08 23:11:15 +02:00
else if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) ) {
2011-04-28 12:07:59 +02:00
u8 secure ;
if ( conn - > pending_sec_level = = BT_SECURITY_HIGH )
secure = 1 ;
else
secure = 0 ;
2011-11-08 20:40:14 +02:00
mgmt_pin_code_request ( hdev , & ev - > bdaddr , secure ) ;
2011-04-28 12:07:59 +02:00
}
2011-01-22 06:10:07 +02:00
2011-09-23 10:01:30 +02:00
unlock :
2009-04-26 20:01:22 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2012-05-23 04:04:18 -03:00
static void hci_link_key_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
2011-01-17 14:41:05 +02:00
struct hci_ev_link_key_req * ev = ( void * ) skb - > data ;
struct hci_cp_link_key_reply cp ;
struct hci_conn * conn ;
struct link_key * key ;
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s " , hdev - > name ) ;
2011-01-17 14:41:05 +02:00
2013-05-14 11:44:16 +03:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-01-17 14:41:05 +02:00
return ;
hci_dev_lock ( hdev ) ;
key = hci_find_link_key ( hdev , & ev - > bdaddr ) ;
if ( ! key ) {
2012-09-25 12:49:43 +03:00
BT_DBG ( " %s link key not found for %pMR " , hdev - > name ,
& ev - > bdaddr ) ;
2011-01-17 14:41:05 +02:00
goto not_found ;
}
2012-09-25 12:49:43 +03:00
BT_DBG ( " %s found key type %u for %pMR " , hdev - > name , key - > type ,
& ev - > bdaddr ) ;
2011-01-17 14:41:05 +02:00
2012-01-08 23:11:15 +02:00
if ( ! test_bit ( HCI_DEBUG_KEYS , & hdev - > dev_flags ) & &
2012-05-17 00:36:24 -03:00
key - > type = = HCI_LK_DEBUG_COMBINATION ) {
2011-01-17 14:41:05 +02:00
BT_DBG ( " %s ignoring debug key " , hdev - > name ) ;
goto not_found ;
}
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2011-04-28 12:07:56 +02:00
if ( conn ) {
2014-01-10 02:07:20 -08:00
if ( ( key - > type = = HCI_LK_UNAUTH_COMBINATION_P192 | |
key - > type = = HCI_LK_UNAUTH_COMBINATION_P256 ) & &
2012-05-17 00:36:24 -03:00
conn - > auth_type ! = 0xff & & ( conn - > auth_type & 0x01 ) ) {
2011-04-28 12:07:56 +02:00
BT_DBG ( " %s ignoring unauthenticated key " , hdev - > name ) ;
goto not_found ;
}
2011-01-17 14:41:05 +02:00
2011-04-28 12:07:56 +02:00
if ( key - > type = = HCI_LK_COMBINATION & & key - > pin_len < 16 & &
2012-05-17 00:36:24 -03:00
conn - > pending_sec_level = = BT_SECURITY_HIGH ) {
2012-05-23 04:04:21 -03:00
BT_DBG ( " %s ignoring key unauthenticated for high security " ,
hdev - > name ) ;
2011-04-28 12:07:56 +02:00
goto not_found ;
}
conn - > key_type = key - > type ;
conn - > pin_length = key - > pin_len ;
2011-01-17 14:41:05 +02:00
}
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2012-05-23 11:31:20 +03:00
memcpy ( cp . link_key , key - > val , HCI_LINK_KEY_SIZE ) ;
2011-01-17 14:41:05 +02:00
hci_send_cmd ( hdev , HCI_OP_LINK_KEY_REPLY , sizeof ( cp ) , & cp ) ;
hci_dev_unlock ( hdev ) ;
return ;
not_found :
hci_send_cmd ( hdev , HCI_OP_LINK_KEY_NEG_REPLY , 6 , & ev - > bdaddr ) ;
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2012-05-23 04:04:18 -03:00
static void hci_link_key_notify_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
2009-04-26 20:01:22 +02:00
struct hci_ev_link_key_notify * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2011-01-17 14:41:05 +02:00
u8 pin_len = 0 ;
2009-04-26 20:01:22 +02:00
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s " , hdev - > name ) ;
2009-04-26 20:01:22 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( conn ) {
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_DISCONN_TIMEOUT ;
2011-01-22 06:10:07 +02:00
pin_len = conn - > pin_length ;
2011-04-28 12:07:55 +02:00
if ( ev - > key_type ! = HCI_LK_CHANGED_COMBINATION )
conn - > key_type = ev - > key_type ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2009-04-26 20:01:22 +02:00
}
2013-05-14 11:44:16 +03:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-04-28 11:28:59 -07:00
hci_add_link_key ( hdev , conn , 1 , & ev - > bdaddr , ev - > link_key ,
2012-05-17 00:36:24 -03:00
ev - > key_type , pin_len ) ;
2011-01-17 14:41:05 +02:00
2009-04-26 20:01:22 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2012-05-23 04:04:18 -03:00
static void hci_clock_offset_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_clock_offset * ev = ( void * ) skb - > data ;
2006-07-03 10:02:33 +02:00
struct hci_conn * conn ;
2005-04-16 15:20:36 -07:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-16 15:20:36 -07:00
hci_dev_lock ( hdev ) ;
2006-07-03 10:02:33 +02:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2005-04-16 15:20:36 -07:00
if ( conn & & ! ev - > status ) {
struct inquiry_entry * ie ;
2010-11-22 13:21:37 +02:00
ie = hci_inquiry_cache_lookup ( hdev , & conn - > dst ) ;
if ( ie ) {
2005-04-16 15:20:36 -07:00
ie - > data . clock_offset = ev - > clock_offset ;
ie - > timestamp = jiffies ;
}
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_pkt_type_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2008-07-14 20:13:46 +02:00
{
struct hci_ev_pkt_type_change * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2008-07-14 20:13:46 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
if ( conn & & ! ev - > status )
conn - > pkt_type = __le16_to_cpu ( ev - > pkt_type ) ;
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_pscan_rep_mode_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-08-09 20:28:02 -07:00
{
2007-10-20 13:33:56 +02:00
struct hci_ev_pscan_rep_mode * ev = ( void * ) skb - > data ;
2005-08-09 20:28:02 -07:00
struct inquiry_entry * ie ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2010-11-22 13:21:37 +02:00
ie = hci_inquiry_cache_lookup ( hdev , & ev - > bdaddr ) ;
if ( ie ) {
2005-08-09 20:28:02 -07:00
ie - > data . pscan_rep_mode = ev - > pscan_rep_mode ;
ie - > timestamp = jiffies ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_inquiry_result_with_rssi_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
struct inquiry_data data ;
int num_rsp = * ( ( __u8 * ) skb - > data ) ;
2012-02-23 00:38:59 +02:00
bool name_known , ssp ;
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s num_rsp %d " , hdev - > name , num_rsp ) ;
if ( ! num_rsp )
return ;
2012-03-21 00:03:38 -03:00
if ( test_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) )
return ;
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
if ( ( skb - > len - 1 ) / num_rsp ! = sizeof ( struct inquiry_info_with_rssi ) ) {
2011-02-17 16:44:23 +01:00
struct inquiry_info_with_rssi_and_pscan_mode * info ;
info = ( void * ) ( skb - > data + 1 ) ;
2007-10-20 13:33:56 +02:00
2011-03-30 23:57:16 +03:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2007-10-20 13:33:56 +02:00
bacpy ( & data . bdaddr , & info - > bdaddr ) ;
data . pscan_rep_mode = info - > pscan_rep_mode ;
data . pscan_period_mode = info - > pscan_period_mode ;
data . pscan_mode = info - > pscan_mode ;
memcpy ( data . dev_class , info - > dev_class , 3 ) ;
data . clock_offset = info - > clock_offset ;
data . rssi = info - > rssi ;
2008-07-14 20:13:48 +02:00
data . ssp_mode = 0x00 ;
2012-01-04 13:39:52 +02:00
name_known = hci_inquiry_cache_update ( hdev , & data ,
2012-03-08 01:25:00 -03:00
false , & ssp ) ;
2011-11-09 13:58:58 +02:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 01:25:00 -03:00
info - > dev_class , info - > rssi ,
! name_known , ssp , NULL , 0 ) ;
2007-10-20 13:33:56 +02:00
}
} else {
struct inquiry_info_with_rssi * info = ( void * ) ( skb - > data + 1 ) ;
2011-03-30 23:57:16 +03:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2007-10-20 13:33:56 +02:00
bacpy ( & data . bdaddr , & info - > bdaddr ) ;
data . pscan_rep_mode = info - > pscan_rep_mode ;
data . pscan_period_mode = info - > pscan_period_mode ;
data . pscan_mode = 0x00 ;
memcpy ( data . dev_class , info - > dev_class , 3 ) ;
data . clock_offset = info - > clock_offset ;
data . rssi = info - > rssi ;
2008-07-14 20:13:48 +02:00
data . ssp_mode = 0x00 ;
2012-01-04 13:39:52 +02:00
name_known = hci_inquiry_cache_update ( hdev , & data ,
2012-03-08 01:25:00 -03:00
false , & ssp ) ;
2011-11-09 13:58:58 +02:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 01:25:00 -03:00
info - > dev_class , info - > rssi ,
! name_known , ssp , NULL , 0 ) ;
2007-10-20 13:33:56 +02:00
}
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_remote_ext_features_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
2008-07-14 20:13:48 +02:00
struct hci_ev_remote_ext_features * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s " , hdev - > name ) ;
2008-07-14 20:13:48 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2010-11-10 17:11:51 +02:00
if ( ! conn )
goto unlock ;
2008-07-14 20:13:48 +02:00
2013-04-17 15:00:51 +03:00
if ( ev - > page < HCI_MAX_PAGES )
memcpy ( conn - > features [ ev - > page ] , ev - > features , 8 ) ;
2010-11-10 17:11:51 +02:00
if ( ! ev - > status & & ev - > page = = 0x01 ) {
struct inquiry_entry * ie ;
2008-07-14 20:13:48 +02:00
2010-11-22 13:21:37 +02:00
ie = hci_inquiry_cache_lookup ( hdev , & conn - > dst ) ;
if ( ie )
2012-02-28 02:28:43 +02:00
ie - > data . ssp_mode = ( ev - > features [ 0 ] & LMP_HOST_SSP ) ;
2008-07-14 20:13:49 +02:00
2013-04-16 20:16:30 +05:30
if ( ev - > features [ 0 ] & LMP_HOST_SSP ) {
2012-01-16 06:47:28 +02:00
set_bit ( HCI_CONN_SSP_ENABLED , & conn - > flags ) ;
2013-04-16 20:16:30 +05:30
} else {
/* It is mandatory by the Bluetooth specification that
* Extended Inquiry Results are only used when Secure
* Simple Pairing is enabled , but some devices violate
* this .
*
* To make these devices work , the internal SSP
* enabled flag needs to be cleared if the remote host
* features do not indicate SSP support */
clear_bit ( HCI_CONN_SSP_ENABLED , & conn - > flags ) ;
}
2014-01-15 22:37:38 -08:00
if ( ev - > features [ 0 ] & LMP_HOST_SC )
set_bit ( HCI_CONN_SC_ENABLED , & conn - > flags ) ;
2010-11-10 17:11:51 +02:00
}
if ( conn - > state ! = BT_CONFIG )
goto unlock ;
2012-05-12 16:11:50 -03:00
if ( ! ev - > status & & ! test_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) ) {
2010-11-18 22:22:29 +02:00
struct hci_cp_remote_name_req cp ;
memset ( & cp , 0 , sizeof ( cp ) ) ;
bacpy ( & cp . bdaddr , & conn - > dst ) ;
cp . pscan_rep_mode = 0x02 ;
hci_send_cmd ( hdev , HCI_OP_REMOTE_NAME_REQ , sizeof ( cp ) , & cp ) ;
2012-01-17 21:48:47 +02:00
} else if ( ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
mgmt_device_connected ( hdev , & conn - > dst , conn - > type ,
2012-03-08 01:25:00 -03:00
conn - > dst_type , 0 , NULL , 0 ,
conn - > dev_class ) ;
2010-11-18 22:22:28 +02:00
2010-11-18 22:22:29 +02:00
if ( ! hci_outgoing_auth_needed ( hdev , conn ) ) {
2010-11-10 17:11:51 +02:00
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2008-07-14 20:13:48 +02:00
}
2010-11-10 17:11:51 +02:00
unlock :
2008-07-14 20:13:48 +02:00
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2012-05-23 04:04:18 -03:00
static void hci_sync_conn_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 13:33:56 +02:00
{
2007-10-20 14:55:10 +02:00
struct hci_ev_sync_conn_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2007-10-20 14:55:10 +02:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ev - > link_type , & ev - > bdaddr ) ;
2008-07-14 20:13:46 +02:00
if ( ! conn ) {
if ( ev - > link_type = = ESCO_LINK )
goto unlock ;
conn = hci_conn_hash_lookup_ba ( hdev , ESCO_LINK , & ev - > bdaddr ) ;
if ( ! conn )
goto unlock ;
conn - > type = SCO_LINK ;
}
2007-10-20 14:55:10 +02:00
2009-04-19 19:14:14 +02:00
switch ( ev - > status ) {
case 0x00 :
2007-10-20 14:55:10 +02:00
conn - > handle = __le16_to_cpu ( ev - > handle ) ;
conn - > state = BT_CONNECTED ;
2008-07-14 20:13:51 +02:00
hci_conn_add_sysfs ( conn ) ;
2009-04-19 19:14:14 +02:00
break ;
2013-08-19 14:24:02 +02:00
case 0x0d : /* Connection Rejected due to Limited Resources */
2010-02-16 11:29:44 -05:00
case 0x11 : /* Unsupported Feature or Parameter Value */
2009-04-19 19:14:14 +02:00
case 0x1c : /* SCO interval rejected */
2010-02-03 11:42:26 -08:00
case 0x1a : /* Unsupported Remote Feature */
2009-04-19 19:14:14 +02:00
case 0x1f : /* Unspecified error */
2013-08-19 14:24:03 +02:00
if ( conn - > out ) {
2009-04-19 19:14:14 +02:00
conn - > pkt_type = ( hdev - > esco_type & SCO_ESCO_MASK ) |
( hdev - > esco_type & EDR_ESCO_MASK ) ;
2013-08-19 14:24:03 +02:00
if ( hci_setup_sync ( conn , conn - > link - > handle ) )
goto unlock ;
2009-04-19 19:14:14 +02:00
}
/* fall through */
default :
2007-10-20 14:55:10 +02:00
conn - > state = BT_CLOSED ;
2009-04-19 19:14:14 +02:00
break ;
}
2007-10-20 14:55:10 +02:00
hci_proto_connect_cfm ( conn , ev - > status ) ;
if ( ev - > status )
hci_conn_del ( conn ) ;
unlock :
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
}
2013-10-15 10:31:12 -07:00
static inline size_t eir_get_length ( u8 * eir , size_t eir_len )
{
size_t parsed = 0 ;
while ( parsed < eir_len ) {
u8 field_len = eir [ 0 ] ;
if ( field_len = = 0 )
return parsed ;
parsed + = field_len + 1 ;
eir + = field_len + 1 ;
}
return eir_len ;
}
2012-05-23 04:04:18 -03:00
static void hci_extended_inquiry_result_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2007-10-20 13:33:56 +02:00
struct inquiry_data data ;
struct extended_inquiry_info * info = ( void * ) ( skb - > data + 1 ) ;
int num_rsp = * ( ( __u8 * ) skb - > data ) ;
2012-04-26 19:19:56 +05:30
size_t eir_len ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
BT_DBG ( " %s num_rsp %d " , hdev - > name , num_rsp ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
if ( ! num_rsp )
return ;
2005-04-16 15:20:36 -07:00
2012-03-21 00:03:38 -03:00
if ( test_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) )
return ;
2007-10-20 13:33:56 +02:00
hci_dev_lock ( hdev ) ;
2011-03-30 23:57:16 +03:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2012-02-23 00:38:59 +02:00
bool name_known , ssp ;
2012-01-04 13:31:59 +02:00
2007-10-20 13:33:56 +02:00
bacpy ( & data . bdaddr , & info - > bdaddr ) ;
2011-02-17 16:44:23 +01:00
data . pscan_rep_mode = info - > pscan_rep_mode ;
data . pscan_period_mode = info - > pscan_period_mode ;
data . pscan_mode = 0x00 ;
2007-10-20 13:33:56 +02:00
memcpy ( data . dev_class , info - > dev_class , 3 ) ;
2011-02-17 16:44:23 +01:00
data . clock_offset = info - > clock_offset ;
data . rssi = info - > rssi ;
2008-07-14 20:13:48 +02:00
data . ssp_mode = 0x01 ;
2012-01-04 13:31:59 +02:00
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-01-15 20:04:43 +02:00
name_known = eir_has_data_type ( info - > data ,
2012-03-08 01:25:00 -03:00
sizeof ( info - > data ) ,
EIR_NAME_COMPLETE ) ;
2012-01-04 13:31:59 +02:00
else
name_known = true ;
2012-02-23 00:38:59 +02:00
name_known = hci_inquiry_cache_update ( hdev , & data , name_known ,
2012-03-08 01:25:00 -03:00
& ssp ) ;
2012-04-26 19:19:56 +05:30
eir_len = eir_get_length ( info - > data , sizeof ( info - > data ) ) ;
2011-11-09 13:58:58 +02:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 01:25:00 -03:00
info - > dev_class , info - > rssi , ! name_known ,
2012-04-26 19:19:56 +05:30
ssp , info - > data , eir_len ) ;
2007-10-20 13:33:56 +02:00
}
hci_dev_unlock ( hdev ) ;
}
2005-04-16 15:20:36 -07:00
2012-06-08 23:31:13 +08:00
static void hci_key_refresh_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_ev_key_refresh_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x handle 0x%4.4x " , hdev - > name , ev - > status ,
2012-06-08 23:31:13 +08:00
__le16_to_cpu ( ev - > handle ) ) ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
if ( ! conn )
goto unlock ;
if ( ! ev - > status )
conn - > sec_level = conn - > pending_sec_level ;
clear_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ;
if ( ev - > status & & conn - > state = = BT_CONNECTED ) {
2013-01-30 11:50:56 -03:00
hci_disconnect ( conn , HCI_ERROR_AUTH_FAILURE ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2012-06-08 23:31:13 +08:00
goto unlock ;
}
if ( conn - > state = = BT_CONFIG ) {
if ( ! ev - > status )
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2012-06-08 23:31:13 +08:00
} else {
hci_auth_cfm ( conn , ev - > status ) ;
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_DISCONN_TIMEOUT ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2012-06-08 23:31:13 +08:00
}
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static u8 hci_get_auth_req ( struct hci_conn * conn )
2011-01-25 13:28:33 +02:00
{
/* If remote requests dedicated bonding follow that lead */
2013-06-28 10:56:28 +02:00
if ( conn - > remote_auth = = HCI_AT_DEDICATED_BONDING | |
conn - > remote_auth = = HCI_AT_DEDICATED_BONDING_MITM ) {
2011-01-25 13:28:33 +02:00
/* If both remote and local IO capabilities allow MITM
* protection then require it , otherwise don ' t */
2013-06-28 10:56:28 +02:00
if ( conn - > remote_cap = = HCI_IO_NO_INPUT_OUTPUT | |
conn - > io_capability = = HCI_IO_NO_INPUT_OUTPUT )
return HCI_AT_DEDICATED_BONDING ;
2011-01-25 13:28:33 +02:00
else
2013-06-28 10:56:28 +02:00
return HCI_AT_DEDICATED_BONDING_MITM ;
2011-01-25 13:28:33 +02:00
}
/* If remote requests no-bonding follow that lead */
2013-06-28 10:56:28 +02:00
if ( conn - > remote_auth = = HCI_AT_NO_BONDING | |
conn - > remote_auth = = HCI_AT_NO_BONDING_MITM )
2011-04-28 12:07:58 +02:00
return conn - > remote_auth | ( conn - > auth_type & 0x01 ) ;
2011-01-25 13:28:33 +02:00
return conn - > auth_type ;
}
2012-05-23 04:04:18 -03:00
static void hci_io_capa_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2008-07-14 20:13:48 +02:00
{
struct hci_ev_io_capa_request * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2011-01-04 15:40:05 +02:00
if ( ! conn )
goto unlock ;
hci_conn_hold ( conn ) ;
2012-01-08 23:11:15 +02:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-01-04 15:40:05 +02:00
goto unlock ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_PAIRABLE , & hdev - > dev_flags ) | |
2012-05-17 00:36:24 -03:00
( conn - > remote_auth & ~ 0x01 ) = = HCI_AT_NO_BONDING ) {
2011-01-25 13:28:33 +02:00
struct hci_cp_io_capability_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2012-01-16 13:34:29 +05:30
/* Change the IO capability from KeyboardDisplay
* to DisplayYesNo as it is not supported by BT spec . */
cp . capability = ( conn - > io_capability = = 0x04 ) ?
2013-06-28 10:56:29 +02:00
HCI_IO_DISPLAY_YESNO : conn - > io_capability ;
2011-04-28 11:29:04 -07:00
conn - > auth_type = hci_get_auth_req ( conn ) ;
cp . authentication = conn - > auth_type ;
2011-01-25 13:28:33 +02:00
2012-05-23 04:04:21 -03:00
if ( hci_find_remote_oob_data ( hdev , & conn - > dst ) & &
( conn - > out | | test_bit ( HCI_CONN_REMOTE_OOB , & conn - > flags ) ) )
2011-03-22 13:12:23 +01:00
cp . oob_data = 0x01 ;
else
cp . oob_data = 0x00 ;
2011-01-25 13:28:33 +02:00
hci_send_cmd ( hdev , HCI_OP_IO_CAPABILITY_REPLY ,
2012-05-17 00:36:24 -03:00
sizeof ( cp ) , & cp ) ;
2011-01-04 15:40:05 +02:00
} else {
struct hci_cp_io_capability_neg_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2011-11-07 14:20:25 +02:00
cp . reason = HCI_ERROR_PAIRING_NOT_ALLOWED ;
2008-07-14 20:13:48 +02:00
2011-01-04 15:40:05 +02:00
hci_send_cmd ( hdev , HCI_OP_IO_CAPABILITY_NEG_REPLY ,
2012-05-17 00:36:24 -03:00
sizeof ( cp ) , & cp ) ;
2011-01-04 15:40:05 +02:00
}
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_io_capa_reply_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-01-04 15:40:05 +02:00
{
struct hci_ev_io_capa_reply * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( ! conn )
goto unlock ;
conn - > remote_cap = ev - > capability ;
conn - > remote_auth = ev - > authentication ;
2012-01-16 06:47:28 +02:00
if ( ev - > oob_data )
set_bit ( HCI_CONN_REMOTE_OOB , & conn - > flags ) ;
2011-01-04 15:40:05 +02:00
unlock :
2008-07-14 20:13:48 +02:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_user_confirm_request_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-02-19 12:05:57 -03:00
{
struct hci_ev_user_confirm_req * ev = ( void * ) skb - > data ;
2011-04-28 11:28:56 -07:00
int loc_mitm , rem_mitm , confirm_hint = 0 ;
2011-04-28 11:28:53 -07:00
struct hci_conn * conn ;
2011-02-19 12:05:57 -03:00
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-04-28 11:28:53 -07:00
goto unlock ;
2011-02-19 12:05:57 -03:00
2011-04-28 11:28:53 -07:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( ! conn )
goto unlock ;
loc_mitm = ( conn - > auth_type & 0x01 ) ;
rem_mitm = ( conn - > remote_auth & 0x01 ) ;
/* If we require MITM but the remote device can't provide that
* ( it has NoInputNoOutput ) then reject the confirmation
* request . The only exception is when we ' re dedicated bonding
* initiators ( connect_cfm_cb set ) since then we always have the MITM
* bit set . */
2013-06-28 10:56:29 +02:00
if ( ! conn - > connect_cfm_cb & & loc_mitm & &
conn - > remote_cap = = HCI_IO_NO_INPUT_OUTPUT ) {
2011-04-28 11:28:53 -07:00
BT_DBG ( " Rejecting request: remote device can't provide MITM " ) ;
hci_send_cmd ( hdev , HCI_OP_USER_CONFIRM_NEG_REPLY ,
2012-05-17 00:36:24 -03:00
sizeof ( ev - > bdaddr ) , & ev - > bdaddr ) ;
2011-04-28 11:28:53 -07:00
goto unlock ;
}
/* If no side requires MITM protection; auto-accept */
2013-06-28 10:56:29 +02:00
if ( ( ! loc_mitm | | conn - > remote_cap = = HCI_IO_NO_INPUT_OUTPUT ) & &
( ! rem_mitm | | conn - > io_capability = = HCI_IO_NO_INPUT_OUTPUT ) ) {
2011-04-28 11:28:56 -07:00
/* If we're not the initiators request authorization to
* proceed from user space ( mgmt_user_confirm with
* confirm_hint set to 1 ) . */
2012-01-16 06:10:31 +02:00
if ( ! test_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ) {
2011-04-28 11:28:56 -07:00
BT_DBG ( " Confirming auto-accept as acceptor " ) ;
confirm_hint = 1 ;
goto confirm ;
}
2011-04-28 11:28:54 -07:00
BT_DBG ( " Auto-accept of user confirmation with %ums delay " ,
2012-05-17 00:36:24 -03:00
hdev - > auto_accept_delay ) ;
2011-04-28 11:28:54 -07:00
if ( hdev - > auto_accept_delay > 0 ) {
int delay = msecs_to_jiffies ( hdev - > auto_accept_delay ) ;
2013-10-16 18:11:39 +03:00
queue_delayed_work ( conn - > hdev - > workqueue ,
& conn - > auto_accept_work , delay ) ;
2011-04-28 11:28:54 -07:00
goto unlock ;
}
2011-04-28 11:28:53 -07:00
hci_send_cmd ( hdev , HCI_OP_USER_CONFIRM_REPLY ,
2012-05-17 00:36:24 -03:00
sizeof ( ev - > bdaddr ) , & ev - > bdaddr ) ;
2011-04-28 11:28:53 -07:00
goto unlock ;
}
2011-04-28 11:28:56 -07:00
confirm :
2012-02-09 15:26:12 +02:00
mgmt_user_confirm_request ( hdev , & ev - > bdaddr , ACL_LINK , 0 , ev - > passkey ,
2012-03-08 01:25:00 -03:00
confirm_hint ) ;
2011-04-28 11:28:53 -07:00
unlock :
2011-02-19 12:05:57 -03:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_user_passkey_request_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-11-23 08:28:34 -08:00
{
struct hci_ev_user_passkey_req * ev = ( void * ) skb - > data ;
BT_DBG ( " %s " , hdev - > name ) ;
2012-01-08 23:11:15 +02:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-02-09 15:26:12 +02:00
mgmt_user_passkey_request ( hdev , & ev - > bdaddr , ACL_LINK , 0 ) ;
2011-11-23 08:28:34 -08:00
}
2012-09-06 18:39:26 +03:00
static void hci_user_passkey_notify_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_ev_user_passkey_notify * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
BT_DBG ( " %s " , hdev - > name ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( ! conn )
return ;
conn - > passkey_notify = __le32_to_cpu ( ev - > passkey ) ;
conn - > passkey_entered = 0 ;
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_user_passkey_notify ( hdev , & conn - > dst , conn - > type ,
conn - > dst_type , conn - > passkey_notify ,
conn - > passkey_entered ) ;
}
static void hci_keypress_notify_evt ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_ev_keypress_notify * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
BT_DBG ( " %s " , hdev - > name ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( ! conn )
return ;
switch ( ev - > type ) {
case HCI_KEYPRESS_STARTED :
conn - > passkey_entered = 0 ;
return ;
case HCI_KEYPRESS_ENTERED :
conn - > passkey_entered + + ;
break ;
case HCI_KEYPRESS_ERASED :
conn - > passkey_entered - - ;
break ;
case HCI_KEYPRESS_CLEARED :
conn - > passkey_entered = 0 ;
break ;
case HCI_KEYPRESS_COMPLETED :
return ;
}
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_user_passkey_notify ( hdev , & conn - > dst , conn - > type ,
conn - > dst_type , conn - > passkey_notify ,
conn - > passkey_entered ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_simple_pair_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 20:13:48 +02:00
{
struct hci_ev_simple_pair_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2011-02-19 12:06:00 -03:00
if ( ! conn )
goto unlock ;
/* To avoid duplicate auth_failed events to user space we check
* the HCI_CONN_AUTH_PEND flag which will be set if we
* initiated the authentication . A traditional auth_complete
* event gets always produced as initiator and is also mapped to
* the mgmt_auth_failed event */
2012-08-09 09:52:29 +02:00
if ( ! test_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) & & ev - > status )
2012-02-09 16:07:29 +02:00
mgmt_auth_failed ( hdev , & conn - > dst , conn - > type , conn - > dst_type ,
2012-03-08 01:25:00 -03:00
ev - > status ) ;
2008-07-14 20:13:48 +02:00
2013-04-06 20:28:37 +02:00
hci_conn_drop ( conn ) ;
2011-02-19 12:06:00 -03:00
unlock :
2008-07-14 20:13:48 +02:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_remote_host_features_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 20:13:48 +02:00
{
struct hci_ev_remote_host_features * ev = ( void * ) skb - > data ;
struct inquiry_entry * ie ;
2013-04-17 15:00:51 +03:00
struct hci_conn * conn ;
2008-07-14 20:13:48 +02:00
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2013-04-17 15:00:51 +03:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
if ( conn )
memcpy ( conn - > features [ 1 ] , ev - > features , 8 ) ;
2010-11-22 13:21:37 +02:00
ie = hci_inquiry_cache_lookup ( hdev , & ev - > bdaddr ) ;
if ( ie )
2012-02-28 02:28:43 +02:00
ie - > data . ssp_mode = ( ev - > features [ 0 ] & LMP_HOST_SSP ) ;
2008-07-14 20:13:48 +02:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_remote_oob_data_request_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-03-22 13:12:22 +01:00
{
struct hci_ev_remote_oob_data_request * ev = ( void * ) skb - > data ;
struct oob_data * data ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2012-01-08 23:11:15 +02:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-04-06 13:01:59 +02:00
goto unlock ;
2011-03-22 13:12:22 +01:00
data = hci_find_remote_oob_data ( hdev , & ev - > bdaddr ) ;
if ( data ) {
2014-01-10 02:07:28 -08:00
if ( test_bit ( HCI_SC_ENABLED , & hdev - > dev_flags ) ) {
struct hci_cp_remote_oob_ext_data_reply cp ;
2011-03-22 13:12:22 +01:00
2014-01-10 02:07:28 -08:00
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
memcpy ( cp . hash192 , data - > hash192 , sizeof ( cp . hash192 ) ) ;
memcpy ( cp . randomizer192 , data - > randomizer192 ,
sizeof ( cp . randomizer192 ) ) ;
memcpy ( cp . hash256 , data - > hash256 , sizeof ( cp . hash256 ) ) ;
memcpy ( cp . randomizer256 , data - > randomizer256 ,
sizeof ( cp . randomizer256 ) ) ;
hci_send_cmd ( hdev , HCI_OP_REMOTE_OOB_EXT_DATA_REPLY ,
sizeof ( cp ) , & cp ) ;
} else {
struct hci_cp_remote_oob_data_reply cp ;
2011-03-22 13:12:22 +01:00
2014-01-10 02:07:28 -08:00
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
memcpy ( cp . hash , data - > hash192 , sizeof ( cp . hash ) ) ;
memcpy ( cp . randomizer , data - > randomizer192 ,
sizeof ( cp . randomizer ) ) ;
hci_send_cmd ( hdev , HCI_OP_REMOTE_OOB_DATA_REPLY ,
sizeof ( cp ) , & cp ) ;
}
2011-03-22 13:12:22 +01:00
} else {
struct hci_cp_remote_oob_data_neg_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2014-01-10 02:07:28 -08:00
hci_send_cmd ( hdev , HCI_OP_REMOTE_OOB_DATA_NEG_REPLY ,
sizeof ( cp ) , & cp ) ;
2011-03-22 13:12:22 +01:00
}
2011-04-06 13:01:59 +02:00
unlock :
2011-03-22 13:12:22 +01:00
hci_dev_unlock ( hdev ) ;
}
2012-10-25 15:20:44 +03:00
static void hci_phy_link_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_ev_phy_link_complete * ev = ( void * ) skb - > data ;
struct hci_conn * hcon , * bredr_hcon ;
BT_DBG ( " %s handle 0x%2.2x status 0x%2.2x " , hdev - > name , ev - > phy_handle ,
ev - > status ) ;
hci_dev_lock ( hdev ) ;
hcon = hci_conn_hash_lookup_handle ( hdev , ev - > phy_handle ) ;
if ( ! hcon ) {
hci_dev_unlock ( hdev ) ;
return ;
}
if ( ev - > status ) {
hci_conn_del ( hcon ) ;
hci_dev_unlock ( hdev ) ;
return ;
}
bredr_hcon = hcon - > amp_mgr - > l2cap_conn - > hcon ;
hcon - > state = BT_CONNECTED ;
bacpy ( & hcon - > dst , & bredr_hcon - > dst ) ;
hci_conn_hold ( hcon ) ;
hcon - > disc_timeout = HCI_DISCONN_TIMEOUT ;
2013-04-06 20:28:37 +02:00
hci_conn_drop ( hcon ) ;
2012-10-25 15:20:44 +03:00
hci_conn_add_sysfs ( hcon ) ;
2012-10-31 15:46:36 +02:00
amp_physical_cfm ( bredr_hcon , hcon ) ;
2012-10-25 15:20:44 +03:00
2012-10-31 15:46:36 +02:00
hci_dev_unlock ( hdev ) ;
2012-10-25 15:20:44 +03:00
}
2012-10-25 15:20:45 +03:00
static void hci_loglink_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_ev_logical_link_complete * ev = ( void * ) skb - > data ;
struct hci_conn * hcon ;
struct hci_chan * hchan ;
struct amp_mgr * mgr ;
BT_DBG ( " %s log_handle 0x%4.4x phy_handle 0x%2.2x status 0x%2.2x " ,
hdev - > name , le16_to_cpu ( ev - > handle ) , ev - > phy_handle ,
ev - > status ) ;
hcon = hci_conn_hash_lookup_handle ( hdev , ev - > phy_handle ) ;
if ( ! hcon )
return ;
/* Create AMP hchan */
hchan = hci_chan_create ( hcon ) ;
if ( ! hchan )
return ;
hchan - > handle = le16_to_cpu ( ev - > handle ) ;
BT_DBG ( " hcon %p mgr %p hchan %p " , hcon , hcon - > amp_mgr , hchan ) ;
mgr = hcon - > amp_mgr ;
if ( mgr & & mgr - > bredr_chan ) {
struct l2cap_chan * bredr_chan = mgr - > bredr_chan ;
l2cap_chan_lock ( bredr_chan ) ;
bredr_chan - > conn - > mtu = hdev - > block_mtu ;
l2cap_logical_cfm ( bredr_chan , hchan , 0 ) ;
hci_conn_hold ( hcon ) ;
l2cap_chan_unlock ( bredr_chan ) ;
}
}
2012-10-31 15:46:31 +02:00
static void hci_disconn_loglink_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_ev_disconn_logical_link_complete * ev = ( void * ) skb - > data ;
struct hci_chan * hchan ;
BT_DBG ( " %s log handle 0x%4.4x status 0x%2.2x " , hdev - > name ,
le16_to_cpu ( ev - > handle ) , ev - > status ) ;
if ( ev - > status )
return ;
hci_dev_lock ( hdev ) ;
hchan = hci_chan_lookup_handle ( hdev , le16_to_cpu ( ev - > handle ) ) ;
if ( ! hchan )
goto unlock ;
amp_destroy_logical_link ( hchan , ev - > reason ) ;
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-10-31 15:46:32 +02:00
static void hci_disconn_phylink_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
{
struct hci_ev_disconn_phy_link_complete * ev = ( void * ) skb - > data ;
struct hci_conn * hcon ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
if ( ev - > status )
return ;
hci_dev_lock ( hdev ) ;
hcon = hci_conn_hash_lookup_handle ( hdev , ev - > phy_handle ) ;
if ( hcon ) {
hcon - > state = BT_CLOSED ;
hci_conn_del ( hcon ) ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_le_conn_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-02-10 22:38:47 -03:00
{
struct hci_ev_le_conn_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2014-02-18 21:41:32 +02:00
struct smp_irk * irk ;
2011-02-10 22:38:47 -03:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2011-02-10 22:38:47 -03:00
hci_dev_lock ( hdev ) ;
2012-07-27 15:10:15 -03:00
conn = hci_conn_hash_lookup_state ( hdev , LE_LINK , BT_CONNECT ) ;
2011-02-10 22:38:50 -03:00
if ( ! conn ) {
conn = hci_conn_add ( hdev , LE_LINK , & ev - > bdaddr ) ;
if ( ! conn ) {
BT_ERR ( " No memory for new connection " ) ;
2012-07-27 15:10:10 -03:00
goto unlock ;
2011-02-10 22:38:50 -03:00
}
2011-05-31 14:20:54 -03:00
conn - > dst_type = ev - > bdaddr_type ;
2012-07-27 15:10:11 -03:00
2013-10-13 07:25:18 -07:00
/* The advertising parameters for own address type
* define which source address and source address
* type this connections has .
*/
if ( bacmp ( & conn - > src , BDADDR_ANY ) ) {
conn - > src_type = ADDR_LE_DEV_PUBLIC ;
} else {
bacpy ( & conn - > src , & hdev - > static_addr ) ;
conn - > src_type = ADDR_LE_DEV_RANDOM ;
}
2012-07-27 15:10:11 -03:00
if ( ev - > role = = LE_CONN_ROLE_MASTER ) {
conn - > out = true ;
conn - > link_mode | = HCI_LM_MASTER ;
}
2011-02-10 22:38:50 -03:00
}
2011-02-10 22:38:47 -03:00
2014-02-23 19:42:17 +02:00
/* Ensure that the hci_conn contains the identity address type
* regardless of which address the connection was made with .
*
* If the controller has a public BD_ADDR , then by default
* use that one . If this is a LE only controller without
* a public address , default to the static random address .
*
* For debugging purposes it is possible to force
* controllers with a public address to use the static
* random address instead .
*/
if ( test_bit ( HCI_FORCE_STATIC_ADDR , & hdev - > dev_flags ) | |
! bacmp ( & hdev - > bdaddr , BDADDR_ANY ) ) {
bacpy ( & conn - > src , & hdev - > static_addr ) ;
conn - > src_type = ADDR_LE_DEV_RANDOM ;
} else {
bacpy ( & conn - > src , & hdev - > bdaddr ) ;
conn - > src_type = ADDR_LE_DEV_PUBLIC ;
}
2014-02-18 15:13:43 -08:00
/* Lookup the identity address from the stored connection
* address and address type .
*
* When establishing connections to an identity address , the
* connection procedure will store the resolvable random
* address first . Now if it can be converted back into the
* identity address , start using the identity address from
* now on .
*/
irk = hci_get_irk ( hdev , & conn - > dst , conn - > dst_type ) ;
2014-02-18 21:41:32 +02:00
if ( irk ) {
bacpy ( & conn - > dst , & irk - > bdaddr ) ;
conn - > dst_type = irk - > addr_type ;
}
2012-07-27 15:10:16 -03:00
if ( ev - > status ) {
2014-02-26 20:21:41 -03:00
hci_le_conn_failed ( conn , ev - > status ) ;
2012-07-27 15:10:16 -03:00
goto unlock ;
}
2012-01-17 21:48:47 +02:00
if ( ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
2014-02-18 14:22:19 -08:00
mgmt_device_connected ( hdev , & conn - > dst , conn - > type ,
2012-03-08 01:25:00 -03:00
conn - > dst_type , 0 , NULL , 0 , NULL ) ;
2011-05-06 18:41:43 -03:00
2011-06-09 18:50:50 -03:00
conn - > sec_level = BT_SECURITY_LOW ;
2011-02-10 22:38:47 -03:00
conn - > handle = __le16_to_cpu ( ev - > handle ) ;
conn - > state = BT_CONNECTED ;
2013-12-11 17:05:37 +02:00
if ( test_bit ( HCI_6LOWPAN_ENABLED , & hdev - > dev_flags ) )
set_bit ( HCI_CONN_6LOWPAN , & conn - > flags ) ;
2011-02-10 22:38:47 -03:00
hci_conn_add_sysfs ( conn ) ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-05-23 04:04:18 -03:00
static void hci_le_adv_report_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-05-26 16:23:51 -03:00
{
2011-09-26 20:48:35 -03:00
u8 num_reports = skb - > data [ 0 ] ;
void * ptr = & skb - > data [ 1 ] ;
2012-01-10 18:20:50 -03:00
s8 rssi ;
2011-05-26 16:23:51 -03:00
2011-09-26 20:48:35 -03:00
while ( num_reports - - ) {
struct hci_ev_le_advertising_info * ev = ptr ;
2011-05-26 16:23:51 -03:00
2012-01-10 18:20:50 -03:00
rssi = ev - > data [ ev - > length ] ;
mgmt_device_found ( hdev , & ev - > bdaddr , LE_LINK , ev - > bdaddr_type ,
2012-03-08 01:25:00 -03:00
NULL , rssi , 0 , 1 , ev - > data , ev - > length ) ;
2012-01-10 18:20:50 -03:00
2011-09-26 20:48:35 -03:00
ptr + = sizeof ( * ev ) + ev - > length + 1 ;
2011-05-26 16:23:51 -03:00
}
}
2012-05-23 04:04:18 -03:00
static void hci_le_ltk_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-06-09 18:50:47 -03:00
{
struct hci_ev_le_ltk_req * ev = ( void * ) skb - > data ;
struct hci_cp_le_ltk_reply cp ;
2011-07-07 18:59:37 -03:00
struct hci_cp_le_ltk_neg_reply neg ;
2011-06-09 18:50:47 -03:00
struct hci_conn * conn ;
2012-02-02 21:08:01 -03:00
struct smp_ltk * ltk ;
2011-06-09 18:50:47 -03:00
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s handle 0x%4.4x " , hdev - > name , __le16_to_cpu ( ev - > handle ) ) ;
2011-06-09 18:50:47 -03:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2011-07-07 18:59:37 -03:00
if ( conn = = NULL )
goto not_found ;
2011-06-09 18:50:47 -03:00
2014-01-30 19:40:00 -08:00
ltk = hci_find_ltk ( hdev , ev - > ediv , ev - > random , conn - > out ) ;
2011-07-07 18:59:37 -03:00
if ( ltk = = NULL )
goto not_found ;
memcpy ( cp . ltk , ltk - > val , sizeof ( ltk - > val ) ) ;
2011-06-09 18:50:47 -03:00
cp . handle = cpu_to_le16 ( conn - > handle ) ;
2012-02-02 21:08:01 -03:00
if ( ltk - > authenticated )
2013-07-31 16:25:28 -03:00
conn - > pending_sec_level = BT_SECURITY_HIGH ;
else
conn - > pending_sec_level = BT_SECURITY_MEDIUM ;
2011-06-09 18:50:47 -03:00
2013-07-31 16:25:29 -03:00
conn - > enc_key_size = ltk - > enc_size ;
2011-06-09 18:50:47 -03:00
hci_send_cmd ( hdev , HCI_OP_LE_LTK_REPLY , sizeof ( cp ) , & cp ) ;
2012-02-02 21:08:01 -03:00
if ( ltk - > type & HCI_SMP_STK ) {
list_del ( & ltk - > list ) ;
kfree ( ltk ) ;
}
2011-06-09 18:50:47 -03:00
hci_dev_unlock ( hdev ) ;
2011-07-07 18:59:37 -03:00
return ;
not_found :
neg . handle = ev - > handle ;
hci_send_cmd ( hdev , HCI_OP_LE_LTK_NEG_REPLY , sizeof ( neg ) , & neg ) ;
hci_dev_unlock ( hdev ) ;
2011-06-09 18:50:47 -03:00
}
2012-05-23 04:04:18 -03:00
static void hci_le_meta_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-02-10 22:38:47 -03:00
{
struct hci_ev_le_meta * le_ev = ( void * ) skb - > data ;
skb_pull ( skb , sizeof ( * le_ev ) ) ;
switch ( le_ev - > subevent ) {
case HCI_EV_LE_CONN_COMPLETE :
hci_le_conn_complete_evt ( hdev , skb ) ;
break ;
2011-05-26 16:23:51 -03:00
case HCI_EV_LE_ADVERTISING_REPORT :
hci_le_adv_report_evt ( hdev , skb ) ;
break ;
2011-06-09 18:50:47 -03:00
case HCI_EV_LE_LTK_REQ :
hci_le_ltk_request_evt ( hdev , skb ) ;
break ;
2011-02-10 22:38:47 -03:00
default :
break ;
}
}
2012-09-27 17:26:22 +03:00
static void hci_chan_selected_evt ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_ev_channel_selected * ev = ( void * ) skb - > data ;
struct hci_conn * hcon ;
BT_DBG ( " %s handle 0x%2.2x " , hdev - > name , ev - > phy_handle ) ;
skb_pull ( skb , sizeof ( * ev ) ) ;
hcon = hci_conn_hash_lookup_handle ( hdev , ev - > phy_handle ) ;
if ( ! hcon )
return ;
amp_read_loc_assoc_final_data ( hdev , hcon ) ;
}
2007-10-20 13:33:56 +02:00
void hci_event_packet ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_event_hdr * hdr = ( void * ) skb - > data ;
__u8 event = hdr - > evt ;
2013-04-02 13:34:31 +03:00
hci_dev_lock ( hdev ) ;
/* Received events are (currently) only needed when a request is
* ongoing so avoid unnecessary memory allocation .
*/
if ( hdev - > req_status = = HCI_REQ_PEND ) {
kfree_skb ( hdev - > recv_evt ) ;
hdev - > recv_evt = skb_clone ( skb , GFP_KERNEL ) ;
}
hci_dev_unlock ( hdev ) ;
2007-10-20 13:33:56 +02:00
skb_pull ( skb , HCI_EVENT_HDR_SIZE ) ;
2013-04-03 21:50:29 +03:00
if ( hdev - > sent_cmd & & bt_cb ( hdev - > sent_cmd ) - > req . event = = event ) {
2013-10-07 18:19:16 +02:00
struct hci_command_hdr * cmd_hdr = ( void * ) hdev - > sent_cmd - > data ;
u16 opcode = __le16_to_cpu ( cmd_hdr - > opcode ) ;
2013-04-03 21:50:29 +03:00
hci_req_cmd_complete ( hdev , opcode , 0 ) ;
}
2007-10-20 13:33:56 +02:00
switch ( event ) {
2005-04-16 15:20:36 -07:00
case HCI_EV_INQUIRY_COMPLETE :
hci_inquiry_complete_evt ( hdev , skb ) ;
break ;
case HCI_EV_INQUIRY_RESULT :
hci_inquiry_result_evt ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_EV_CONN_COMPLETE :
hci_conn_complete_evt ( hdev , skb ) ;
2005-09-13 01:32:25 +02:00
break ;
2005-04-16 15:20:36 -07:00
case HCI_EV_CONN_REQUEST :
hci_conn_request_evt ( hdev , skb ) ;
break ;
case HCI_EV_DISCONN_COMPLETE :
hci_disconn_complete_evt ( hdev , skb ) ;
break ;
case HCI_EV_AUTH_COMPLETE :
hci_auth_complete_evt ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_EV_REMOTE_NAME :
hci_remote_name_evt ( hdev , skb ) ;
break ;
2005-04-16 15:20:36 -07:00
case HCI_EV_ENCRYPT_CHANGE :
hci_encrypt_change_evt ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_EV_CHANGE_LINK_KEY_COMPLETE :
hci_change_link_key_complete_evt ( hdev , skb ) ;
break ;
case HCI_EV_REMOTE_FEATURES :
hci_remote_features_evt ( hdev , skb ) ;
break ;
case HCI_EV_CMD_COMPLETE :
hci_cmd_complete_evt ( hdev , skb ) ;
break ;
case HCI_EV_CMD_STATUS :
hci_cmd_status_evt ( hdev , skb ) ;
break ;
case HCI_EV_ROLE_CHANGE :
hci_role_change_evt ( hdev , skb ) ;
break ;
case HCI_EV_NUM_COMP_PKTS :
hci_num_comp_pkts_evt ( hdev , skb ) ;
break ;
case HCI_EV_MODE_CHANGE :
hci_mode_change_evt ( hdev , skb ) ;
2005-04-16 15:20:36 -07:00
break ;
case HCI_EV_PIN_CODE_REQ :
hci_pin_code_request_evt ( hdev , skb ) ;
break ;
case HCI_EV_LINK_KEY_REQ :
hci_link_key_request_evt ( hdev , skb ) ;
break ;
case HCI_EV_LINK_KEY_NOTIFY :
hci_link_key_notify_evt ( hdev , skb ) ;
break ;
case HCI_EV_CLOCK_OFFSET :
hci_clock_offset_evt ( hdev , skb ) ;
break ;
2008-07-14 20:13:46 +02:00
case HCI_EV_PKT_TYPE_CHANGE :
hci_pkt_type_change_evt ( hdev , skb ) ;
break ;
2005-08-09 20:28:02 -07:00
case HCI_EV_PSCAN_REP_MODE :
hci_pscan_rep_mode_evt ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
case HCI_EV_INQUIRY_RESULT_WITH_RSSI :
hci_inquiry_result_with_rssi_evt ( hdev , skb ) ;
2006-07-03 10:02:33 +02:00
break ;
2007-10-20 13:33:56 +02:00
case HCI_EV_REMOTE_EXT_FEATURES :
hci_remote_ext_features_evt ( hdev , skb ) ;
2005-04-16 15:20:36 -07:00
break ;
2007-10-20 13:33:56 +02:00
case HCI_EV_SYNC_CONN_COMPLETE :
hci_sync_conn_complete_evt ( hdev , skb ) ;
break ;
2005-04-16 15:20:36 -07:00
2007-10-20 13:33:56 +02:00
case HCI_EV_EXTENDED_INQUIRY_RESULT :
hci_extended_inquiry_result_evt ( hdev , skb ) ;
break ;
2005-04-16 15:20:36 -07:00
2012-06-08 23:31:13 +08:00
case HCI_EV_KEY_REFRESH_COMPLETE :
hci_key_refresh_complete_evt ( hdev , skb ) ;
break ;
2008-07-14 20:13:48 +02:00
case HCI_EV_IO_CAPA_REQUEST :
hci_io_capa_request_evt ( hdev , skb ) ;
break ;
2011-01-04 15:40:05 +02:00
case HCI_EV_IO_CAPA_REPLY :
hci_io_capa_reply_evt ( hdev , skb ) ;
break ;
2011-02-19 12:05:57 -03:00
case HCI_EV_USER_CONFIRM_REQUEST :
hci_user_confirm_request_evt ( hdev , skb ) ;
break ;
2011-11-23 08:28:34 -08:00
case HCI_EV_USER_PASSKEY_REQUEST :
hci_user_passkey_request_evt ( hdev , skb ) ;
break ;
2012-09-06 18:39:26 +03:00
case HCI_EV_USER_PASSKEY_NOTIFY :
hci_user_passkey_notify_evt ( hdev , skb ) ;
break ;
case HCI_EV_KEYPRESS_NOTIFY :
hci_keypress_notify_evt ( hdev , skb ) ;
break ;
2008-07-14 20:13:48 +02:00
case HCI_EV_SIMPLE_PAIR_COMPLETE :
hci_simple_pair_complete_evt ( hdev , skb ) ;
break ;
2008-07-14 20:13:48 +02:00
case HCI_EV_REMOTE_HOST_FEATURES :
hci_remote_host_features_evt ( hdev , skb ) ;
break ;
2011-02-10 22:38:47 -03:00
case HCI_EV_LE_META :
hci_le_meta_evt ( hdev , skb ) ;
break ;
2012-09-27 17:26:22 +03:00
case HCI_EV_CHANNEL_SELECTED :
hci_chan_selected_evt ( hdev , skb ) ;
break ;
2011-03-22 13:12:22 +01:00
case HCI_EV_REMOTE_OOB_DATA_REQUEST :
hci_remote_oob_data_request_evt ( hdev , skb ) ;
break ;
2012-10-25 15:20:44 +03:00
case HCI_EV_PHY_LINK_COMPLETE :
hci_phy_link_complete_evt ( hdev , skb ) ;
break ;
2012-10-25 15:20:45 +03:00
case HCI_EV_LOGICAL_LINK_COMPLETE :
hci_loglink_complete_evt ( hdev , skb ) ;
break ;
2012-10-31 15:46:31 +02:00
case HCI_EV_DISCONN_LOGICAL_LINK_COMPLETE :
hci_disconn_loglink_complete_evt ( hdev , skb ) ;
break ;
2012-10-31 15:46:32 +02:00
case HCI_EV_DISCONN_PHY_LINK_COMPLETE :
hci_disconn_phylink_complete_evt ( hdev , skb ) ;
break ;
2012-01-04 12:41:58 +02:00
case HCI_EV_NUM_COMP_BLOCKS :
hci_num_comp_blocks_evt ( hdev , skb ) ;
break ;
2007-10-20 13:33:56 +02:00
default :
2012-07-11 14:32:43 +03:00
BT_DBG ( " %s event 0x%2.2x " , hdev - > name , event ) ;
2005-04-16 15:20:36 -07:00
break ;
}
kfree_skb ( skb ) ;
hdev - > stat . evt_rx + + ;
}