2007-02-09 17:24:33 +03:00
/*
2005-04-17 02:20:36 +04:00
BlueZ - Bluetooth protocol stack for Linux
2010-05-28 19:53:46 +04:00
Copyright ( c ) 2000 - 2001 , 2010 , Code Aurora Forum . All rights reserved .
2005-04-17 02:20:36 +04: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 17:24:33 +03: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-17 02:20:36 +04:00
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
2007-02-09 17:24:33 +03:00
ALL LIABILITY , INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS ,
COPYRIGHTS , TRADEMARKS OR OTHER RIGHTS , RELATING TO USE OF THIS
2005-04-17 02:20:36 +04:00
SOFTWARE IS DISCLAIMED .
*/
/* Bluetooth HCI event handling. */
2012-05-23 11:04:22 +04:00
# include <linux/export.h>
2005-04-17 02:20:36 +04:00
# include <asm/unaligned.h>
# include <net/bluetooth/bluetooth.h>
# include <net/bluetooth/hci_core.h>
2012-08-09 11:52:30 +04:00
# include <net/bluetooth/mgmt.h>
2012-09-27 18:26:08 +04:00
# include <net/bluetooth/a2mp.h>
2012-09-27 18:26:09 +04:00
# include <net/bluetooth/amp.h>
2005-04-17 02:20:36 +04:00
/* Handle HCI Event packets */
2007-10-20 15:33:56 +04:00
static void hci_cc_inquiry_cancel ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2011-11-10 00:14:26 +04:00
if ( status ) {
hci_dev_lock ( hdev ) ;
mgmt_stop_discovery_failed ( hdev , status ) ;
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
return ;
2011-11-10 00:14:26 +04:00
}
2005-04-17 02:20:36 +04:00
2011-11-04 21:16:53 +04:00
clear_bit ( HCI_INQUIRY , & hdev - > flags ) ;
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-04 16:23:45 +04:00
hci_discovery_set_state ( hdev , DISCOVERY_STOPPED ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2006-11-19 00:14:22 +03:00
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_INQUIRY_CANCEL , status ) ;
2007-10-20 15:33:56 +04:00
hci_conn_check_pending ( hdev ) ;
}
2006-11-19 00:14:22 +03:00
2012-03-21 07:03:35 +04:00
static void hci_cc_periodic_inq ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2012-03-21 07:03:36 +04:00
if ( status )
return ;
set_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) ;
2012-03-21 07:03:35 +04:00
}
2007-10-20 15:33:56 +04:00
static void hci_cc_exit_periodic_inq ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2006-11-19 00:14:22 +03:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-11-19 00:14:22 +03:00
2007-10-20 15:33:56 +04:00
if ( status )
return ;
2005-04-17 02:20:36 +04:00
2012-03-21 07:03:36 +04:00
clear_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) ;
2007-10-20 15:33:56 +04:00
hci_conn_check_pending ( hdev ) ;
}
2012-05-17 07:36:24 +04:00
static void hci_cc_remote_name_req_cancel ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 15:33:56 +04: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-17 02:20:36 +04:00
}
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-14 22:13:47 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2008-07-14 22:13:47 +04: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 15:33:56 +04:00
static void hci_cc_write_link_policy ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_rp_write_link_policy * rp = ( void * ) skb - > data ;
2005-04-17 02:20:36 +04:00
struct hci_conn * conn ;
2006-07-03 12:02:33 +04:00
void * sent ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( rp - > status )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_LINK_POLICY ) ;
if ( ! sent )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( rp - > handle ) ) ;
2008-07-14 22:13:47 +04:00
if ( conn )
2008-05-03 03:25:46 +04:00
conn - > link_policy = get_unaligned_le16 ( sent + 2 ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
}
2005-04-17 02:20:36 +04:00
2012-05-17 07:36:24 +04:00
static void hci_cc_read_def_link_policy ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 22:13:47 +04:00
{
struct hci_rp_read_def_link_policy * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2008-07-14 22:13:47 +04:00
if ( rp - > status )
return ;
hdev - > link_policy = __le16_to_cpu ( rp - > policy ) ;
}
2012-05-17 07:36:24 +04:00
static void hci_cc_write_def_link_policy ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 22:13:47 +04:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
void * sent ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 22:13:47 +04: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 ) ;
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_WRITE_DEF_LINK_POLICY , status ) ;
2008-07-14 22:13:47 +04:00
}
2007-10-20 15:33:56 +04:00
static void hci_cc_reset ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2006-07-03 12:02:33 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 12:02:33 +04:00
2011-03-16 21:36:29 +03:00
clear_bit ( HCI_RESET , & hdev - > flags ) ;
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_RESET , status ) ;
2011-11-26 03:53:38 +04:00
2012-02-21 19:55:47 +04:00
/* Reset all non-persistent flags */
2012-03-21 07:03:36 +04:00
hdev - > dev_flags & = ~ ( BIT ( HCI_LE_SCAN ) | BIT ( HCI_PENDING_CLASS ) |
BIT ( HCI_PERIODIC_INQ ) ) ;
2012-02-23 18:50:05 +04:00
hdev - > discovery . state = DISCOVERY_STOPPED ;
2007-10-20 15:33:56 +04:00
}
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04: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 12:02:33 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_LOCAL_NAME ) ;
if ( ! sent )
return ;
2006-07-03 12:02:33 +04:00
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-02-22 20:17:32 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_set_local_name_complete ( hdev , sent , status ) ;
2012-02-22 23:06:55 +04:00
else if ( ! status )
memcpy ( hdev - > dev_name , sent , HCI_MAX_NAME_LENGTH ) ;
2012-02-22 20:17:32 +04:00
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2012-02-24 15:47:56 +04:00
hci_req_complete ( hdev , HCI_OP_WRITE_LOCAL_NAME , status ) ;
2007-10-20 15:33:56 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 15:33:56 +04:00
if ( rp - > status )
return ;
2012-02-22 22:14:22 +04:00
if ( test_bit ( HCI_SETUP , & hdev - > dev_flags ) )
memcpy ( hdev - > dev_name , rp - > name , HCI_MAX_NAME_LENGTH ) ;
2007-10-20 15:33:56 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 15:33:56 +04: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-17 02:20:36 +04:00
}
2007-10-20 15:33:56 +04:00
2012-02-17 01:56:27 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
mgmt_auth_enable_complete ( hdev , status ) ;
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_WRITE_AUTH_ENABLE , status ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-20 15:33:56 +04:00
static void hci_cc_write_encrypt_mode ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2005-04-17 02:20:36 +04:00
void * sent ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_ENCRYPT_MODE ) ;
if ( ! sent )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( ! status ) {
__u8 param = * ( ( __u8 * ) sent ) ;
if ( param )
set_bit ( HCI_ENCRYPT , & hdev - > flags ) ;
else
clear_bit ( HCI_ENCRYPT , & hdev - > flags ) ;
}
2005-04-17 02:20:36 +04:00
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_WRITE_ENCRYPT_MODE , status ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
static void hci_cc_write_scan_enable ( struct hci_dev * hdev , struct sk_buff * skb )
{
2011-11-04 02:17:45 +04:00
__u8 param , status = * ( ( __u8 * ) skb - > data ) ;
int old_pscan , old_iscan ;
2007-10-20 15:33:56 +04:00
void * sent ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_SCAN_ENABLE ) ;
if ( ! sent )
return ;
2005-04-17 02:20:36 +04:00
2011-11-04 02:17:45 +04:00
param = * ( ( __u8 * ) sent ) ;
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-08-09 11:52:29 +04:00
if ( status ) {
2011-11-08 22:40:14 +04:00
mgmt_write_scan_failed ( hdev , param , status ) ;
2011-11-08 00:16:03 +04:00
hdev - > discov_timeout = 0 ;
goto done ;
}
2011-11-04 02:17:45 +04: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 22:40:14 +04:00
mgmt_discoverable ( hdev , 1 ) ;
2011-11-08 00:16:02 +04:00
if ( hdev - > discov_timeout > 0 ) {
int to = msecs_to_jiffies ( hdev - > discov_timeout * 1000 ) ;
queue_delayed_work ( hdev - > workqueue , & hdev - > discov_off ,
2012-05-17 07:36:24 +04:00
to ) ;
2011-11-08 00:16:02 +04:00
}
2011-11-04 02:17:45 +04:00
} else if ( old_iscan )
2011-11-08 22:40:14 +04:00
mgmt_discoverable ( hdev , 0 ) ;
2011-11-04 02:17:45 +04:00
if ( param & SCAN_PAGE ) {
set_bit ( HCI_PSCAN , & hdev - > flags ) ;
if ( ! old_pscan )
2011-11-08 22:40:14 +04:00
mgmt_connectable ( hdev , 1 ) ;
2011-11-04 02:17:45 +04:00
} else if ( old_pscan )
2011-11-08 22:40:14 +04:00
mgmt_connectable ( hdev , 0 ) ;
2005-04-17 02:20:36 +04:00
2011-11-04 02:17:45 +04:00
done :
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_WRITE_SCAN_ENABLE , status ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04: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-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( rp - > status )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
memcpy ( hdev - > dev_class , rp - > dev_class , 3 ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s class 0x%.2x%.2x%.2x " , hdev - > name ,
2012-05-17 07:36:24 +04:00
hdev - > dev_class [ 2 ] , hdev - > dev_class [ 1 ] , hdev - > dev_class [ 0 ] ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04: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-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_CLASS_OF_DEV ) ;
if ( ! sent )
return ;
2005-04-17 02:20:36 +04:00
2012-02-22 21:38:01 +04: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 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 15:33:56 +04:00
if ( rp - > status )
return ;
setting = __le16_to_cpu ( rp - > voice_setting ) ;
2008-07-14 22:13:47 +04:00
if ( hdev - > voice_setting = = setting )
2007-10-20 15:33:56 +04:00
return ;
hdev - > voice_setting = setting ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s voice setting 0x%4.4x " , hdev - > name , setting ) ;
2007-10-20 15:33:56 +04:00
2011-12-15 04:58:44 +04:00
if ( hdev - > notify )
2007-10-20 15:33:56 +04:00
hdev - > notify ( hdev , HCI_NOTIFY_VOICE_SETTING ) ;
}
2012-05-23 11:04:21 +04:00
static void hci_cc_write_voice_setting ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2008-07-14 22:13:47 +04:00
__u16 setting ;
2007-10-20 15:33:56 +04:00
void * sent ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2008-07-14 22:13:47 +04:00
if ( status )
return ;
2007-10-20 15:33:56 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_VOICE_SETTING ) ;
if ( ! sent )
return ;
2005-04-17 02:20:36 +04:00
2008-07-14 22:13:47 +04:00
setting = get_unaligned_le16 ( sent ) ;
2005-04-17 02:20:36 +04:00
2008-07-14 22:13:47 +04:00
if ( hdev - > voice_setting = = setting )
return ;
hdev - > voice_setting = setting ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s voice setting 0x%4.4x " , hdev - > name , setting ) ;
2005-04-17 02:20:36 +04:00
2011-12-15 04:58:44 +04:00
if ( hdev - > notify )
2008-07-14 22:13:47 +04:00
hdev - > notify ( hdev , HCI_NOTIFY_VOICE_SETTING ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-20 15:33:56 +04:00
static void hci_cc_host_buffer_size ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_HOST_BUFFER_SIZE , status ) ;
2007-10-20 15:33:56 +04:00
}
2006-09-23 11:57:20 +04:00
2008-07-14 22:13:48 +04:00
static void hci_cc_write_ssp_mode ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
void * sent ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 22:13:48 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_SSP_MODE ) ;
if ( ! sent )
return ;
2012-02-17 02:56:28 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-02-22 14:38:31 +04:00
mgmt_ssp_enable_complete ( hdev , * ( ( u8 * ) sent ) , status ) ;
else if ( ! status ) {
if ( * ( ( u8 * ) sent ) )
set_bit ( HCI_SSP_ENABLED , & hdev - > dev_flags ) ;
else
clear_bit ( HCI_SSP_ENABLED , & hdev - > dev_flags ) ;
}
2008-07-14 22:13:48 +04:00
}
2011-01-25 02:19:58 +03:00
static u8 hci_get_inquiry_mode ( struct hci_dev * hdev )
{
2012-10-24 22:12:01 +04:00
if ( lmp_ext_inq_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
return 2 ;
2012-10-24 22:12:01 +04:00
if ( lmp_inq_rssi_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
return 1 ;
if ( hdev - > manufacturer = = 11 & & hdev - > hci_rev = = 0x00 & &
2012-05-17 07:36:24 +04:00
hdev - > lmp_subver = = 0x0757 )
2011-01-25 02:19:58 +03:00
return 1 ;
if ( hdev - > manufacturer = = 15 ) {
if ( hdev - > hci_rev = = 0x03 & & hdev - > lmp_subver = = 0x6963 )
return 1 ;
if ( hdev - > hci_rev = = 0x09 & & hdev - > lmp_subver = = 0x6963 )
return 1 ;
if ( hdev - > hci_rev = = 0x00 & & hdev - > lmp_subver = = 0x6965 )
return 1 ;
}
if ( hdev - > manufacturer = = 31 & & hdev - > hci_rev = = 0x2005 & &
2012-05-17 07:36:24 +04:00
hdev - > lmp_subver = = 0x1805 )
2011-01-25 02:19:58 +03:00
return 1 ;
return 0 ;
}
static void hci_setup_inquiry_mode ( struct hci_dev * hdev )
{
u8 mode ;
mode = hci_get_inquiry_mode ( hdev ) ;
hci_send_cmd ( hdev , HCI_OP_WRITE_INQUIRY_MODE , 1 , & mode ) ;
}
static void hci_setup_event_mask ( struct hci_dev * hdev )
{
/* The second byte is 0xff instead of 0x9f (two reserved bits
* disabled ) since a Broadcom 1.2 dongle doesn ' t respond to the
* command otherwise */
u8 events [ 8 ] = { 0xff , 0xff , 0xfb , 0xff , 0x00 , 0x00 , 0x00 , 0x00 } ;
2011-05-27 12:16:21 +04:00
/* CSR 1.1 dongles does not accept any bitfield so don't try to set
* any event mask for pre 1.2 devices */
2011-12-01 16:33:28 +04:00
if ( hdev - > hci_ver < BLUETOOTH_VER_1_2 )
2011-05-27 12:16:21 +04:00
return ;
2012-10-19 21:57:45 +04:00
if ( lmp_bredr_capable ( hdev ) ) {
events [ 4 ] | = 0x01 ; /* Flow Specification Complete */
events [ 4 ] | = 0x02 ; /* Inquiry Result with RSSI */
events [ 4 ] | = 0x04 ; /* Read Remote Extended Features Complete */
events [ 5 ] | = 0x08 ; /* Synchronous Connection Complete */
events [ 5 ] | = 0x10 ; /* Synchronous Connection Changed */
}
2011-01-25 02:19:58 +03:00
2012-10-24 22:12:01 +04:00
if ( lmp_inq_rssi_capable ( hdev ) )
2012-04-26 10:47:46 +04:00
events [ 4 ] | = 0x02 ; /* Inquiry Result with RSSI */
2011-01-25 02:19:58 +03:00
2012-07-24 22:03:52 +04:00
if ( lmp_sniffsubr_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
events [ 5 ] | = 0x20 ; /* Sniff Subrating */
2012-10-24 22:12:01 +04:00
if ( lmp_pause_enc_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
events [ 5 ] | = 0x80 ; /* Encryption Key Refresh Complete */
2012-10-24 22:12:01 +04:00
if ( lmp_ext_inq_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
events [ 5 ] | = 0x40 ; /* Extended Inquiry Result */
2012-07-24 22:03:53 +04:00
if ( lmp_no_flush_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
events [ 7 ] | = 0x01 ; /* Enhanced Flush Complete */
2012-10-24 22:12:01 +04:00
if ( lmp_lsto_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
events [ 6 ] | = 0x80 ; /* Link Supervision Timeout Changed */
2012-07-24 22:03:48 +04:00
if ( lmp_ssp_capable ( hdev ) ) {
2011-01-25 02:19:58 +03:00
events [ 6 ] | = 0x01 ; /* IO Capability Request */
events [ 6 ] | = 0x02 ; /* IO Capability Response */
events [ 6 ] | = 0x04 ; /* User Confirmation Request */
events [ 6 ] | = 0x08 ; /* User Passkey Request */
events [ 6 ] | = 0x10 ; /* Remote OOB Data Request */
events [ 6 ] | = 0x20 ; /* Simple Pairing Complete */
events [ 7 ] | = 0x04 ; /* User Passkey Notification */
events [ 7 ] | = 0x08 ; /* Keypress Notification */
events [ 7 ] | = 0x10 ; /* Remote Host Supported
* Features Notification */
}
2012-07-24 22:03:47 +04:00
if ( lmp_le_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
events [ 7 ] | = 0x20 ; /* LE Meta-Event */
hci_send_cmd ( hdev , HCI_OP_SET_EVENT_MASK , sizeof ( events ) , events ) ;
2012-10-19 21:57:47 +04:00
if ( lmp_le_capable ( hdev ) ) {
memset ( events , 0 , sizeof ( events ) ) ;
events [ 0 ] = 0x1f ;
hci_send_cmd ( hdev , HCI_OP_LE_SET_EVENT_MASK ,
sizeof ( events ) , events ) ;
}
2011-01-25 02:19:58 +03:00
}
2012-10-24 17:18:41 +04:00
static void bredr_setup ( struct hci_dev * hdev )
2012-10-19 21:57:45 +04:00
{
struct hci_cp_delete_stored_link_key cp ;
__le16 param ;
__u8 flt_type ;
/* Read Buffer Size (ACL mtu, max pkt, etc.) */
hci_send_cmd ( hdev , HCI_OP_READ_BUFFER_SIZE , 0 , NULL ) ;
/* Read Class of Device */
hci_send_cmd ( hdev , HCI_OP_READ_CLASS_OF_DEV , 0 , NULL ) ;
/* Read Local Name */
hci_send_cmd ( hdev , HCI_OP_READ_LOCAL_NAME , 0 , NULL ) ;
/* Read Voice Setting */
hci_send_cmd ( hdev , HCI_OP_READ_VOICE_SETTING , 0 , NULL ) ;
/* Clear Event Filters */
flt_type = HCI_FLT_CLEAR_ALL ;
hci_send_cmd ( hdev , HCI_OP_SET_EVENT_FLT , 1 , & flt_type ) ;
/* Connection accept timeout ~20 secs */
param = __constant_cpu_to_le16 ( 0x7d00 ) ;
hci_send_cmd ( hdev , HCI_OP_WRITE_CA_TIMEOUT , 2 , & param ) ;
bacpy ( & cp . bdaddr , BDADDR_ANY ) ;
cp . delete_all = 1 ;
hci_send_cmd ( hdev , HCI_OP_DELETE_STORED_LINK_KEY , sizeof ( cp ) , & cp ) ;
}
2012-10-24 17:18:41 +04:00
static void le_setup ( struct hci_dev * hdev )
2012-10-19 21:57:45 +04:00
{
/* Read LE Buffer Size */
hci_send_cmd ( hdev , HCI_OP_LE_READ_BUFFER_SIZE , 0 , NULL ) ;
2012-10-19 21:57:49 +04:00
/* Read LE Advertising Channel TX Power */
hci_send_cmd ( hdev , HCI_OP_LE_READ_ADV_TX_POWER , 0 , NULL ) ;
2012-10-19 21:57:45 +04:00
}
2011-01-25 02:19:58 +03:00
static void hci_setup ( struct hci_dev * hdev )
{
2011-12-19 18:31:27 +04:00
if ( hdev - > dev_type ! = HCI_BREDR )
return ;
2012-10-19 21:57:45 +04:00
/* Read BD Address */
hci_send_cmd ( hdev , HCI_OP_READ_BD_ADDR , 0 , NULL ) ;
if ( lmp_bredr_capable ( hdev ) )
2012-10-24 17:18:41 +04:00
bredr_setup ( hdev ) ;
2012-10-19 21:57:45 +04:00
if ( lmp_le_capable ( hdev ) )
2012-10-24 17:18:41 +04:00
le_setup ( hdev ) ;
2012-10-19 21:57:45 +04:00
2011-01-25 02:19:58 +03:00
hci_setup_event_mask ( hdev ) ;
2011-12-01 16:33:27 +04:00
if ( hdev - > hci_ver > BLUETOOTH_VER_1_1 )
2011-01-25 02:19:58 +03:00
hci_send_cmd ( hdev , HCI_OP_READ_LOCAL_COMMANDS , 0 , NULL ) ;
2012-05-24 10:36:37 +04:00
if ( lmp_ssp_capable ( hdev ) ) {
2012-02-22 17:47:48 +04:00
if ( test_bit ( HCI_SSP_ENABLED , & hdev - > dev_flags ) ) {
u8 mode = 0x01 ;
hci_send_cmd ( hdev , HCI_OP_WRITE_SSP_MODE ,
2012-03-08 08:25:00 +04:00
sizeof ( mode ) , & mode ) ;
2012-02-22 17:47:48 +04:00
} else {
struct hci_cp_write_eir cp ;
memset ( hdev - > eir , 0 , sizeof ( hdev - > eir ) ) ;
memset ( & cp , 0 , sizeof ( cp ) ) ;
hci_send_cmd ( hdev , HCI_OP_WRITE_EIR , sizeof ( cp ) , & cp ) ;
}
2011-01-25 02:19:58 +03:00
}
2012-10-24 22:12:01 +04:00
if ( lmp_inq_rssi_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
hci_setup_inquiry_mode ( hdev ) ;
2012-10-24 22:12:01 +04:00
if ( lmp_inq_tx_pwr_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
hci_send_cmd ( hdev , HCI_OP_READ_INQ_RSP_TX_POWER , 0 , NULL ) ;
2011-07-01 02:20:52 +04:00
2012-10-24 22:12:01 +04:00
if ( lmp_ext_feat_capable ( hdev ) ) {
2011-07-01 02:20:52 +04:00
struct hci_cp_read_local_ext_features cp ;
cp . page = 0x01 ;
2012-03-08 08:25:00 +04:00
hci_send_cmd ( hdev , HCI_OP_READ_LOCAL_EXT_FEATURES , sizeof ( cp ) ,
& cp ) ;
2011-07-01 02:20:52 +04:00
}
2011-07-01 02:20:54 +04:00
2012-02-22 13:58:37 +04:00
if ( test_bit ( HCI_LINK_SECURITY , & hdev - > dev_flags ) ) {
u8 enable = 1 ;
2012-03-08 08:25:00 +04:00
hci_send_cmd ( hdev , HCI_OP_WRITE_AUTH_ENABLE , sizeof ( enable ) ,
& enable ) ;
2012-02-22 13:58:37 +04:00
}
2011-01-25 02:19:58 +03:00
}
2007-10-20 15:33:56 +04: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 11:57:20 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2006-09-23 11:57:20 +04:00
2007-10-20 15:33:56 +04:00
if ( rp - > status )
2012-02-24 14:45:44 +04:00
goto done ;
2006-09-23 11:57:20 +04:00
2007-10-20 15:33:56 +04:00
hdev - > hci_ver = rp - > hci_ver ;
2008-07-14 22:13:47 +04:00
hdev - > hci_rev = __le16_to_cpu ( rp - > hci_rev ) ;
2011-01-25 02:19:58 +03:00
hdev - > lmp_ver = rp - > lmp_ver ;
2008-07-14 22:13:47 +04:00
hdev - > manufacturer = __le16_to_cpu ( rp - > manufacturer ) ;
2011-01-25 02:19:58 +03:00
hdev - > lmp_subver = __le16_to_cpu ( rp - > lmp_subver ) ;
2006-09-23 11:57:20 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s manufacturer 0x%4.4x hci ver %d:%d " , hdev - > name ,
2012-05-17 07:36:24 +04:00
hdev - > manufacturer , hdev - > hci_ver , hdev - > hci_rev ) ;
2011-01-25 02:19:58 +03:00
if ( test_bit ( HCI_INIT , & hdev - > flags ) )
hci_setup ( hdev ) ;
2012-02-24 14:45:44 +04:00
done :
hci_req_complete ( hdev , HCI_OP_READ_LOCAL_VERSION , rp - > status ) ;
2011-01-25 02:19:58 +03:00
}
static void hci_setup_link_policy ( struct hci_dev * hdev )
{
2012-03-12 17:59:32 +04:00
struct hci_cp_write_def_link_policy cp ;
2011-01-25 02:19:58 +03:00
u16 link_policy = 0 ;
2012-07-24 22:03:50 +04:00
if ( lmp_rswitch_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
link_policy | = HCI_LP_RSWITCH ;
2012-10-24 22:12:01 +04:00
if ( lmp_hold_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
link_policy | = HCI_LP_HOLD ;
2012-07-24 22:03:51 +04:00
if ( lmp_sniff_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
link_policy | = HCI_LP_SNIFF ;
2012-10-24 22:12:01 +04:00
if ( lmp_park_capable ( hdev ) )
2011-01-25 02:19:58 +03:00
link_policy | = HCI_LP_PARK ;
2012-03-12 17:59:32 +04:00
cp . policy = cpu_to_le16 ( link_policy ) ;
hci_send_cmd ( hdev , HCI_OP_WRITE_DEF_LINK_POLICY , sizeof ( cp ) , & cp ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2012-05-23 11:04:21 +04:00
static void hci_cc_read_local_commands ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct hci_rp_read_local_commands * rp = ( void * ) skb - > data ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( rp - > status )
2011-01-25 02:19:58 +03:00
goto done ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
memcpy ( hdev - > commands , rp - > commands , sizeof ( hdev - > commands ) ) ;
2011-01-25 02:19:58 +03:00
if ( test_bit ( HCI_INIT , & hdev - > flags ) & & ( hdev - > commands [ 5 ] & 0x10 ) )
hci_setup_link_policy ( hdev ) ;
done :
hci_req_complete ( hdev , HCI_OP_READ_LOCAL_COMMANDS , rp - > status ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2012-05-23 11:04:21 +04:00
static void hci_cc_read_local_features ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct hci_rp_read_local_features * rp = ( void * ) skb - > data ;
2007-07-11 11:51:55 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( rp - > status )
return ;
2007-07-11 11:51:55 +04:00
2007-10-20 15:33:56 +04:00
memcpy ( hdev - > features , rp - > features , 8 ) ;
2007-07-11 11:51:55 +04:00
2007-10-20 15:33:56 +04:00
/* Adjust default settings according to features
* supported by device . */
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( hdev - > features [ 0 ] & LMP_3SLOT )
hdev - > pkt_type | = ( HCI_DM3 | HCI_DH3 ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( hdev - > features [ 0 ] & LMP_5SLOT )
hdev - > pkt_type | = ( HCI_DM5 | HCI_DH5 ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( hdev - > features [ 1 ] & LMP_HV2 ) {
hdev - > pkt_type | = ( HCI_HV2 ) ;
hdev - > esco_type | = ( ESCO_HV2 ) ;
}
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( hdev - > features [ 1 ] & LMP_HV3 ) {
hdev - > pkt_type | = ( HCI_HV3 ) ;
hdev - > esco_type | = ( ESCO_HV3 ) ;
}
2005-04-17 02:20:36 +04:00
2012-07-24 22:03:49 +04:00
if ( lmp_esco_capable ( hdev ) )
2007-10-20 15:33:56 +04:00
hdev - > esco_type | = ( ESCO_EV3 ) ;
2006-07-03 12:02:29 +04:00
2007-10-20 15:33:56 +04:00
if ( hdev - > features [ 4 ] & LMP_EV4 )
hdev - > esco_type | = ( ESCO_EV4 ) ;
2006-07-03 12:02:29 +04:00
2007-10-20 15:33:56 +04:00
if ( hdev - > features [ 4 ] & LMP_EV5 )
hdev - > esco_type | = ( ESCO_EV5 ) ;
2005-04-17 02:20:36 +04:00
2009-02-06 11:13:37 +03:00
if ( hdev - > features [ 5 ] & LMP_EDR_ESCO_2M )
hdev - > esco_type | = ( ESCO_2EV3 ) ;
if ( hdev - > features [ 5 ] & LMP_EDR_ESCO_3M )
hdev - > esco_type | = ( ESCO_3EV3 ) ;
if ( hdev - > features [ 5 ] & LMP_EDR_3S_ESCO )
hdev - > esco_type | = ( ESCO_2EV5 | ESCO_3EV5 ) ;
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s features 0x%.2x%.2x%.2x%.2x%.2x%.2x%.2x%.2x " , hdev - > name ,
2012-05-17 07:36:24 +04:00
hdev - > features [ 0 ] , hdev - > features [ 1 ] ,
hdev - > features [ 2 ] , hdev - > features [ 3 ] ,
hdev - > features [ 4 ] , hdev - > features [ 5 ] ,
hdev - > features [ 6 ] , hdev - > features [ 7 ] ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2012-02-28 03:07:22 +04:00
static void hci_set_le_support ( struct hci_dev * hdev )
{
struct hci_cp_write_le_host_supported cp ;
memset ( & cp , 0 , sizeof ( cp ) ) ;
2012-05-03 09:12:31 +04:00
if ( test_bit ( HCI_LE_ENABLED , & hdev - > dev_flags ) ) {
2012-02-28 03:07:22 +04:00
cp . le = 1 ;
2012-10-24 22:12:01 +04:00
cp . simul = ! ! lmp_le_br_capable ( hdev ) ;
2012-02-28 03:07:22 +04:00
}
2012-10-24 22:12:01 +04:00
if ( cp . le ! = ! ! lmp_host_le_capable ( hdev ) )
2012-03-08 08:25:00 +04:00
hci_send_cmd ( hdev , HCI_OP_WRITE_LE_HOST_SUPPORTED , sizeof ( cp ) ,
& cp ) ;
2012-02-28 03:07:22 +04:00
}
2011-07-01 02:20:52 +04:00
static void hci_cc_read_local_ext_features ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-07-01 02:20:52 +04:00
{
struct hci_rp_read_local_ext_features * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-07-01 02:20:52 +04:00
if ( rp - > status )
2012-02-28 03:07:22 +04:00
goto done ;
2011-07-01 02:20:52 +04:00
2011-12-30 17:34:04 +04:00
switch ( rp - > page ) {
case 0 :
memcpy ( hdev - > features , rp - > features , 8 ) ;
break ;
case 1 :
memcpy ( hdev - > host_features , rp - > features , 8 ) ;
break ;
}
2011-07-01 02:20:52 +04:00
2012-07-24 22:03:47 +04:00
if ( test_bit ( HCI_INIT , & hdev - > flags ) & & lmp_le_capable ( hdev ) )
2012-02-28 03:07:22 +04:00
hci_set_le_support ( hdev ) ;
done :
2011-07-01 02:20:52 +04:00
hci_req_complete ( hdev , HCI_OP_READ_LOCAL_EXT_FEATURES , rp - > status ) ;
}
2011-11-24 16:52:02 +04:00
static void hci_cc_read_flow_control_mode ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-11-24 16:52:02 +04:00
{
struct hci_rp_read_flow_control_mode * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-11-24 16:52:02 +04:00
if ( rp - > status )
return ;
hdev - > flow_ctl_mode = rp - > mode ;
hci_req_complete ( hdev , HCI_OP_READ_FLOW_CONTROL_MODE , rp - > status ) ;
}
2007-10-20 15:33:56 +04: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-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( rp - > status )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04: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-17 02:20:36 +04:00
}
2007-10-20 15:33:56 +04:00
hdev - > acl_cnt = hdev - > acl_pkts ;
hdev - > sco_cnt = hdev - > sco_pkts ;
2012-05-17 07:36:24 +04: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 15:33:56 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2007-10-20 15:33:56 +04:00
if ( ! rp - > status )
bacpy ( & hdev - > bdaddr , & rp - > bdaddr ) ;
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_READ_BD_ADDR , rp - > status ) ;
}
2011-12-07 17:56:51 +04:00
static void hci_cc_read_data_block_size ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-12-07 17:56:51 +04:00
{
struct hci_rp_read_data_block_size * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-12-07 17:56:51 +04: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 07:36:24 +04:00
hdev - > block_cnt , hdev - > block_len ) ;
2011-12-07 17:56:51 +04:00
hci_req_complete ( hdev , HCI_OP_READ_DATA_BLOCK_SIZE , rp - > status ) ;
}
2010-12-22 00:01:27 +03:00
static void hci_cc_write_ca_timeout ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_WRITE_CA_TIMEOUT , status ) ;
2007-10-20 15:33:56 +04:00
}
2011-10-12 11:53:57 +04:00
static void hci_cc_read_local_amp_info ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-10-12 11:53:57 +04:00
{
struct hci_rp_read_local_amp_info * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-10-12 11:53:57 +04:00
if ( rp - > status )
2012-09-27 18:26:08 +04:00
goto a2mp_rsp ;
2011-10-12 11:53:57 +04: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 ) ;
hci_req_complete ( hdev , HCI_OP_READ_LOCAL_AMP_INFO , rp - > status ) ;
2012-09-27 18:26:08 +04:00
a2mp_rsp :
a2mp_send_getinfo_rsp ( hdev ) ;
2011-10-12 11:53:57 +04:00
}
2012-09-27 18:26:09 +04: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 15:44:23 +04:00
BT_DBG ( " frag_len %zu rem_len %zu " , frag_len , rem_len ) ;
2012-09-27 18:26:09 +04: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 18:26:22 +04:00
a2mp_send_create_phy_link_req ( hdev , rp - > status ) ;
2012-09-27 18:26:09 +04:00
}
2011-01-10 14:44:55 +03:00
static void hci_cc_delete_stored_link_key ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-01-10 14:44:55 +03:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-01-10 14:44:55 +03:00
hci_req_complete ( hdev , HCI_OP_DELETE_STORED_LINK_KEY , status ) ;
}
2011-01-25 02:19:58 +03:00
static void hci_cc_set_event_mask ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-01-25 02:19:58 +03:00
hci_req_complete ( hdev , HCI_OP_SET_EVENT_MASK , status ) ;
}
static void hci_cc_write_inquiry_mode ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-01-25 02:19:58 +03:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-01-25 02:19:58 +03:00
hci_req_complete ( hdev , HCI_OP_WRITE_INQUIRY_MODE , status ) ;
}
static void hci_cc_read_inq_rsp_tx_power ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-01-25 02:19:58 +03:00
{
2012-03-12 06:27:21 +04:00
struct hci_rp_read_inq_rsp_tx_power * rp = ( void * ) skb - > data ;
2011-01-25 02:19:58 +03:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2012-03-12 06:27:21 +04:00
if ( ! rp - > status )
hdev - > inq_tx_power = rp - > tx_power ;
2011-01-25 02:19:58 +03:00
2012-03-12 06:27:21 +04:00
hci_req_complete ( hdev , HCI_OP_READ_INQ_RSP_TX_POWER , rp - > status ) ;
2011-01-25 02:19:58 +03:00
}
static void hci_cc_set_event_flt ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-01-25 02:19:58 +03:00
hci_req_complete ( hdev , HCI_OP_SET_EVENT_FLT , status ) ;
}
2011-01-22 07:10:07 +03: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-01-22 07:10:07 +03:00
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-08 22:40:14 +04:00
mgmt_pin_code_reply_complete ( hdev , & rp - > bdaddr , rp - > status ) ;
2011-01-22 07:10:07 +03:00
2012-08-09 11:52:29 +04:00
if ( rp - > status )
2011-11-08 22:40:16 +04:00
goto unlock ;
2011-01-22 07:10:07 +03:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_PIN_CODE_REPLY ) ;
if ( ! cp )
2011-11-08 22:40:16 +04:00
goto unlock ;
2011-01-22 07:10:07 +03:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & cp - > bdaddr ) ;
if ( conn )
conn - > pin_length = cp - > pin_len ;
2011-11-08 22:40:16 +04:00
unlock :
hci_dev_unlock ( hdev ) ;
2011-01-22 07:10:07 +03: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-01-22 07:10:07 +03:00
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-08 22:40:14 +04:00
mgmt_pin_code_neg_reply_complete ( hdev , & rp - > bdaddr ,
2012-05-17 07:36:24 +04:00
rp - > status ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2011-01-22 07:10:07 +03:00
}
2011-11-08 22:40:16 +04:00
2011-02-11 04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-02-11 04: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 ) ;
hci_req_complete ( hdev , HCI_OP_LE_READ_BUFFER_SIZE , rp - > status ) ;
}
2011-01-22 07:10:07 +03:00
2012-10-19 21:57:49 +04: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 ) ;
if ( ! rp - > status )
hdev - > adv_tx_power = rp - > tx_power ;
hci_req_complete ( hdev , HCI_OP_LE_READ_ADV_TX_POWER , rp - > status ) ;
}
2012-10-19 21:57:47 +04:00
static void hci_cc_le_set_event_mask ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
hci_req_complete ( hdev , HCI_OP_LE_SET_EVENT_MASK , status ) ;
}
2011-02-19 18: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-02-19 18:05:57 +03:00
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-03-08 08:25:00 +04:00
mgmt_user_confirm_reply_complete ( hdev , & rp - > bdaddr , ACL_LINK , 0 ,
rp - > status ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2011-02-19 18:05:57 +03:00
}
static void hci_cc_user_confirm_neg_reply ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-02-19 18:05:57 +03:00
{
struct hci_rp_user_confirm_reply * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-02-19 18:05:57 +03:00
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-08 22:40:14 +04:00
mgmt_user_confirm_neg_reply_complete ( hdev , & rp - > bdaddr ,
2012-03-08 08:25:00 +04:00
ACL_LINK , 0 , rp - > status ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2011-02-19 18:05:57 +03:00
}
2011-11-23 20:28:34 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-11-23 20:28:34 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-02-09 17:26:12 +04:00
mgmt_user_passkey_reply_complete ( hdev , & rp - > bdaddr , ACL_LINK ,
2012-03-08 08:25:00 +04:00
0 , rp - > status ) ;
2011-11-23 20:28:34 +04:00
hci_dev_unlock ( hdev ) ;
}
static void hci_cc_user_passkey_neg_reply ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-11-23 20:28:34 +04:00
{
struct hci_rp_user_confirm_reply * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-11-23 20:28:34 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-23 20:28:34 +04:00
mgmt_user_passkey_neg_reply_complete ( hdev , & rp - > bdaddr ,
2012-03-08 08:25:00 +04:00
ACL_LINK , 0 , rp - > status ) ;
2011-11-23 20:28:34 +04:00
hci_dev_unlock ( hdev ) ;
}
2011-03-22 15:12:21 +03:00
static void hci_cc_read_local_oob_data_reply ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-03-22 15:12:21 +03:00
{
struct hci_rp_read_local_oob_data * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-03-22 15:12:21 +03:00
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2011-11-08 22:40:14 +04:00
mgmt_read_local_oob_data_reply_complete ( hdev , rp - > hash ,
2011-03-22 15:12:21 +03:00
rp - > randomizer , rp - > status ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2011-03-22 15:12:21 +03:00
}
2011-12-02 16:13:31 +04:00
static void hci_cc_le_set_scan_param ( struct hci_dev * hdev , struct sk_buff * skb )
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2012-02-04 00:47:59 +04:00
hci_req_complete ( hdev , HCI_OP_LE_SET_SCAN_PARAM , status ) ;
2012-02-04 00:48:01 +04:00
if ( status ) {
hci_dev_lock ( hdev ) ;
mgmt_start_discovery_failed ( hdev , status ) ;
hci_dev_unlock ( hdev ) ;
return ;
}
2011-12-02 16:13:31 +04:00
}
2011-05-26 23:23:52 +04:00
static void hci_cc_le_set_scan_enable ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct sk_buff * skb )
2011-05-26 23:23:52 +04:00
{
struct hci_cp_le_set_scan_enable * cp ;
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-05-26 23:23:52 +04:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_LE_SET_SCAN_ENABLE ) ;
if ( ! cp )
return ;
2011-12-19 18:14:18 +04:00
switch ( cp - > enable ) {
case LE_SCANNING_ENABLED :
2012-02-04 00:47:59 +04:00
hci_req_complete ( hdev , HCI_OP_LE_SET_SCAN_ENABLE , status ) ;
2012-02-04 00:48:01 +04:00
if ( status ) {
hci_dev_lock ( hdev ) ;
mgmt_start_discovery_failed ( hdev , status ) ;
hci_dev_unlock ( hdev ) ;
2012-02-04 00:47:59 +04:00
return ;
2012-02-04 00:48:01 +04:00
}
2012-02-04 00:47:59 +04:00
2011-11-26 03:53:38 +04:00
set_bit ( HCI_LE_SCAN , & hdev - > dev_flags ) ;
2011-09-10 01:56:24 +04:00
hci_dev_lock ( hdev ) ;
2012-02-18 03:39:37 +04:00
hci_discovery_set_state ( hdev , DISCOVERY_FINDING ) ;
2011-09-10 01:56:24 +04:00
hci_dev_unlock ( hdev ) ;
2011-12-19 18:14:18 +04:00
break ;
case LE_SCANNING_DISABLED :
2012-03-15 23:52:08 +04:00
if ( status ) {
hci_dev_lock ( hdev ) ;
mgmt_stop_discovery_failed ( hdev , status ) ;
hci_dev_unlock ( hdev ) ;
2012-02-04 00:47:59 +04:00
return ;
2012-03-15 23:52:08 +04:00
}
2012-02-04 00:47:59 +04:00
2011-11-26 03:53:38 +04:00
clear_bit ( HCI_LE_SCAN , & hdev - > dev_flags ) ;
2012-03-07 02:37:06 +04:00
if ( hdev - > discovery . type = = DISCOV_TYPE_INTERLEAVED & &
hdev - > discovery . state = = DISCOVERY_FINDING ) {
2012-02-18 03:39:38 +04:00
mgmt_interleaved_discovery ( hdev ) ;
} else {
hci_dev_lock ( hdev ) ;
hci_discovery_set_state ( hdev , DISCOVERY_STOPPED ) ;
hci_dev_unlock ( hdev ) ;
}
2011-12-19 18:14:18 +04:00
break ;
default :
BT_ERR ( " Used reserved LE_Scan_Enable param %d " , cp - > enable ) ;
break ;
2011-05-26 23:23:53 +04:00
}
2011-05-26 23:23:52 +04:00
}
2011-06-10 01:50:47 +04:00
static void hci_cc_le_ltk_reply ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_le_ltk_reply * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-06-10 01:50:47 +04:00
if ( rp - > status )
return ;
hci_req_complete ( hdev , HCI_OP_LE_LTK_REPLY , rp - > status ) ;
}
static void hci_cc_le_ltk_neg_reply ( struct hci_dev * hdev , struct sk_buff * skb )
{
struct hci_rp_le_ltk_neg_reply * rp = ( void * ) skb - > data ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , rp - > status ) ;
2011-06-10 01:50:47 +04:00
if ( rp - > status )
return ;
hci_req_complete ( hdev , HCI_OP_LE_LTK_NEG_REPLY , rp - > status ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_cc_write_le_host_supported ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-07-01 02:20:53 +04:00
{
2012-02-22 18:37:11 +04:00
struct hci_cp_write_le_host_supported * sent ;
2011-07-01 02:20:53 +04:00
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-07-01 02:20:53 +04:00
2012-02-22 18:37:11 +04:00
sent = hci_sent_cmd_data ( hdev , HCI_OP_WRITE_LE_HOST_SUPPORTED ) ;
2012-02-28 03:07:22 +04:00
if ( ! sent )
2011-07-01 02:20:53 +04:00
return ;
2012-02-28 03:07:22 +04:00
if ( ! status ) {
if ( sent - > le )
hdev - > host_features [ 0 ] | = LMP_HOST_LE ;
else
hdev - > host_features [ 0 ] & = ~ LMP_HOST_LE ;
2012-10-24 22:11:59 +04:00
if ( sent - > simul )
hdev - > host_features [ 0 ] | = LMP_HOST_LE_BREDR ;
else
hdev - > host_features [ 0 ] & = ~ LMP_HOST_LE_BREDR ;
2012-02-28 03:07:22 +04:00
}
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) & &
2012-05-17 07:36:24 +04:00
! test_bit ( HCI_INIT , & hdev - > flags ) )
2012-02-28 03:07:22 +04:00
mgmt_le_enable_complete ( hdev , sent - > le , status ) ;
hci_req_complete ( hdev , HCI_OP_WRITE_LE_HOST_SUPPORTED , status ) ;
2011-07-01 02:20:53 +04:00
}
2012-09-27 18:26:20 +04: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 11:04:18 +04:00
static void hci_cs_inquiry ( struct hci_dev * hdev , __u8 status )
2007-10-20 15:33:56 +04:00
{
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 15:33:56 +04:00
if ( status ) {
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_INQUIRY , status ) ;
2007-10-20 15:33:56 +04:00
hci_conn_check_pending ( hdev ) ;
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-11-10 00:14:25 +04:00
mgmt_start_discovery_failed ( hdev , status ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2011-04-27 18:29:57 +04:00
return ;
}
2011-11-04 21:16:53 +04:00
set_bit ( HCI_INQUIRY , & hdev - > flags ) ;
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-02-18 03:39:37 +04:00
hci_discovery_set_state ( hdev , DISCOVERY_FINDING ) ;
2011-11-08 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_cs_create_conn ( struct hci_dev * hdev , __u8 status )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_cp_create_conn * cp ;
2005-04-17 02:20:36 +04:00
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 15:33:56 +04:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_CREATE_CONN ) ;
2005-04-17 02:20:36 +04:00
if ( ! cp )
return ;
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & cp - > bdaddr ) ;
2012-09-25 13:49:43 +04:00
BT_DBG ( " %s bdaddr %pMR hcon %p " , hdev - > name , & cp - > bdaddr , conn ) ;
2005-04-17 02:20:36 +04:00
if ( status ) {
if ( conn & & conn - > state = = BT_CONNECT ) {
2006-10-15 19:30:56 +04: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-17 02:20:36 +04:00
}
} else {
if ( ! conn ) {
conn = hci_conn_add ( hdev , ACL_LINK , & cp - > bdaddr ) ;
if ( conn ) {
2012-01-16 11:49:58 +04:00
conn - > out = true ;
2005-04-17 02:20:36 +04:00
conn - > link_mode | = HCI_LM_MASTER ;
} else
2010-07-18 22:13:37 +04:00
BT_ERR ( " No memory for new connection " ) ;
2005-04-17 02:20:36 +04:00
}
}
hci_dev_unlock ( hdev ) ;
}
2007-10-20 15:33:56 +04:00
static void hci_cs_add_sco ( struct hci_dev * hdev , __u8 status )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_cp_add_sco * cp ;
struct hci_conn * acl , * sco ;
__u16 handle ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 16:55:10 +04:00
2007-10-20 15:33:56 +04:00
if ( ! status )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_ADD_SCO ) ;
if ( ! cp )
return ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
handle = __le16_to_cpu ( cp - > handle ) ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s handle 0x%4.4x " , hdev - > name , handle ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
acl = hci_conn_hash_lookup_handle ( hdev , handle ) ;
2011-01-11 18:20:20 +03:00
if ( acl ) {
sco = acl - > link ;
if ( sco ) {
sco - > state = BT_CLOSED ;
2005-04-17 02:20:36 +04:00
2011-01-11 18:20:20 +03:00
hci_proto_connect_cfm ( sco , status ) ;
hci_conn_del ( sco ) ;
}
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
}
2005-04-17 02:20:36 +04:00
2008-07-14 22:13:49 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 22:13:49 +04: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 ) ;
hci_conn_put ( conn ) ;
}
}
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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 22:13:49 +04: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 ) ;
hci_conn_put ( conn ) ;
}
}
hci_dev_unlock ( hdev ) ;
}
2010-11-18 23:22:29 +03:00
static int hci_outgoing_auth_needed ( struct hci_dev * hdev ,
2012-05-17 07:36:24 +04:00
struct hci_conn * conn )
2010-11-18 23:22:28 +03:00
{
if ( conn - > state ! = BT_CONFIG | | ! conn - > out )
return 0 ;
2011-01-19 09:36:52 +03:00
if ( conn - > pending_sec_level = = BT_SECURITY_SDP )
2010-11-18 23:22:28 +03:00
return 0 ;
/* Only request authentication for SSP connections or non-SSP
2011-09-02 21:51:20 +04:00
* devices with sec_level HIGH or if MITM protection is requested */
2012-05-17 07:36:24 +04:00
if ( ! hci_conn_ssp_enabled ( conn ) & & ! ( conn - > auth_type & 0x01 ) & &
conn - > pending_sec_level ! = BT_SECURITY_HIGH )
2010-11-18 23:22:28 +03:00
return 0 ;
return 1 ;
}
2012-05-23 11:04:18 +04:00
static int hci_resolve_name ( struct hci_dev * hdev ,
2012-03-08 08:25:00 +04:00
struct inquiry_entry * e )
2012-01-04 17:44:20 +04: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 23:48:47 +04:00
static bool hci_resolve_next_name ( struct hci_dev * hdev )
2012-01-04 17:44:20 +04:00
{
struct discovery_state * discov = & hdev - > discovery ;
struct inquiry_entry * e ;
2012-01-17 23:48:47 +04:00
if ( list_empty ( & discov - > resolve ) )
return false ;
e = hci_inquiry_cache_lookup_resolve ( hdev , BDADDR_ANY , NAME_NEEDED ) ;
2012-07-19 11:26:09 +04:00
if ( ! e )
return false ;
2012-01-17 23:48:47 +04: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 08:25:00 +04:00
bdaddr_t * bdaddr , u8 * name , u8 name_len )
2012-01-17 23:48:47 +04: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 08:25:00 +04:00
mgmt_device_connected ( hdev , bdaddr , ACL_LINK , 0x00 , 0 , name ,
name_len , conn - > dev_class ) ;
2012-01-17 23:48:47 +04:00
if ( discov - > state = = DISCOVERY_STOPPED )
return ;
2012-01-04 17:44:20 +04: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 11:26:10 +04: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 17:44:20 +04:00
e - > name_state = NAME_KNOWN ;
2012-07-19 11:26:10 +04:00
mgmt_remote_name ( hdev , bdaddr , ACL_LINK , 0x00 ,
e - > data . rssi , name , name_len ) ;
2012-07-19 11:26:11 +04:00
} else {
e - > name_state = NAME_NOT_KNOWN ;
2012-01-04 17:44:20 +04:00
}
2012-01-17 23:48:47 +04:00
if ( hci_resolve_next_name ( hdev ) )
2012-01-04 17:44:20 +04:00
return ;
discov_complete :
hci_discovery_set_state ( hdev , DISCOVERY_STOPPED ) ;
}
2007-10-20 15:33:56 +04:00
static void hci_cs_remote_name_req ( struct hci_dev * hdev , __u8 status )
{
2010-11-18 23:22:29 +03:00
struct hci_cp_remote_name_req * cp ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2010-11-18 23:22:29 +03: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 23:48:47 +04:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & cp - > bdaddr ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-01-17 23:48:47 +04:00
hci_check_pending_name ( hdev , conn , & cp - > bdaddr , NULL , 0 ) ;
2012-01-04 17:44:20 +04:00
2011-04-28 22:28:55 +04:00
if ( ! conn )
goto unlock ;
if ( ! hci_outgoing_auth_needed ( hdev , conn ) )
goto unlock ;
2012-01-16 08:10:31 +04:00
if ( ! test_and_set_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ) {
2010-11-18 23:22:29 +03: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 22:28:55 +04:00
unlock :
2010-11-18 23:22:29 +03:00
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
}
2005-04-17 02:20:36 +04:00
2008-07-14 22:13:49 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 22:13:49 +04: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 ) ;
hci_conn_put ( conn ) ;
}
}
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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2008-07-14 22:13:49 +04: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 ) ;
hci_conn_put ( conn ) ;
}
}
hci_dev_unlock ( hdev ) ;
}
2007-10-20 15:33:56 +04:00
static void hci_cs_setup_sync_conn ( struct hci_dev * hdev , __u8 status )
{
2007-10-20 16:55:10 +04:00
struct hci_cp_setup_sync_conn * cp ;
struct hci_conn * acl , * sco ;
__u16 handle ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2007-10-20 16:55:10 +04: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 15:32:43 +04:00
BT_DBG ( " %s handle 0x%4.4x " , hdev - > name , handle ) ;
2007-10-20 16:55:10 +04:00
hci_dev_lock ( hdev ) ;
acl = hci_conn_hash_lookup_handle ( hdev , handle ) ;
2011-01-11 18:20:20 +03:00
if ( acl ) {
sco = acl - > link ;
if ( sco ) {
sco - > state = BT_CLOSED ;
2007-10-20 16:55:10 +04:00
2011-01-11 18:20:20 +03:00
hci_proto_connect_cfm ( sco , status ) ;
hci_conn_del ( sco ) ;
}
2007-10-20 16:55:10 +04:00
}
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-20 15:33:56 +04:00
static void hci_cs_sniff_mode ( struct hci_dev * hdev , __u8 status )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_cp_sniff_mode * cp ;
struct hci_conn * conn ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
if ( ! status )
return ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_SNIFF_MODE ) ;
if ( ! cp )
return ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
2010-07-26 18:06:00 +04:00
if ( conn ) {
2012-01-16 08:10:31 +04:00
clear_bit ( HCI_CONN_MODE_CHANGE_PEND , & conn - > flags ) ;
2006-07-03 12:02:33 +04:00
2012-01-16 08:10:31 +04:00
if ( test_and_clear_bit ( HCI_CONN_SCO_SETUP_PEND , & conn - > flags ) )
2010-07-26 18:06:00 +04:00
hci_sco_setup ( conn , status ) ;
}
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
}
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04: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 12:02:33 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
if ( ! status )
return ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
cp = hci_sent_cmd_data ( hdev , HCI_OP_EXIT_SNIFF_MODE ) ;
if ( ! cp )
return ;
2006-07-03 12:02:33 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( cp - > handle ) ) ;
2010-07-26 18:06:00 +04:00
if ( conn ) {
2012-01-16 08:10:31 +04:00
clear_bit ( HCI_CONN_MODE_CHANGE_PEND , & conn - > flags ) ;
2005-04-17 02:20:36 +04:00
2012-01-16 08:10:31 +04:00
if ( test_and_clear_bit ( HCI_CONN_SCO_SETUP_PEND , & conn - > flags ) )
2010-07-26 18:06:00 +04:00
hci_sco_setup ( conn , status ) ;
}
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2012-02-09 16:27:38 +04: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 08:25:00 +04:00
conn - > dst_type , status ) ;
2012-02-09 16:27:38 +04:00
hci_dev_unlock ( hdev ) ;
}
2011-02-11 04:38:47 +03:00
static void hci_cs_le_create_conn ( struct hci_dev * hdev , __u8 status )
{
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-02-11 04:38:47 +03:00
2012-07-27 22:10:13 +04:00
if ( status ) {
hci_dev_lock ( hdev ) ;
2011-02-11 04:38:47 +03:00
2012-07-27 22:10:14 +04:00
conn = hci_conn_hash_lookup_state ( hdev , LE_LINK , BT_CONNECT ) ;
2012-07-27 22:10:13 +04:00
if ( ! conn ) {
hci_dev_unlock ( hdev ) ;
return ;
}
2011-02-11 04:38:47 +03:00
2012-09-25 13:49:43 +04:00
BT_DBG ( " %s bdaddr %pMR conn %p " , hdev - > name , & conn - > dst , conn ) ;
2011-02-11 04:38:47 +03:00
2012-07-27 22:10:13 +04:00
conn - > state = BT_CLOSED ;
2012-07-27 22:10:14 +04:00
mgmt_connect_failed ( hdev , & conn - > dst , conn - > type ,
2012-07-27 22:10:13 +04:00
conn - > dst_type , status ) ;
hci_proto_connect_cfm ( conn , status ) ;
hci_conn_del ( conn ) ;
2011-02-11 04:38:47 +03:00
2012-07-27 22:10:13 +04:00
hci_dev_unlock ( hdev ) ;
}
2011-02-11 04:38:47 +03:00
}
2011-06-10 01:50:47 +04:00
static void hci_cs_le_start_enc ( struct hci_dev * hdev , u8 status )
{
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2011-06-10 01:50:47 +04:00
}
2012-09-27 18:26:19 +04:00
static void hci_cs_create_phylink ( struct hci_dev * hdev , u8 status )
{
2012-09-27 18:26:20 +04:00
struct hci_cp_create_phy_link * cp ;
2012-09-27 18:26:19 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2012-09-27 18:26:20 +04:00
if ( status )
return ;
cp = hci_sent_cmd_data ( hdev , HCI_OP_CREATE_PHY_LINK ) ;
if ( ! cp )
return ;
amp_write_remote_assoc ( hdev , cp - > phy_handle ) ;
2012-09-27 18:26:19 +04:00
}
2012-09-27 18:26:24 +04: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 11:04:18 +04:00
static void hci_inquiry_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
__u8 status = * ( ( __u8 * ) skb - > data ) ;
2012-01-04 17:44:20 +04:00
struct discovery_state * discov = & hdev - > discovery ;
struct inquiry_entry * e ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , status ) ;
2005-04-17 02:20:36 +04:00
2010-12-22 00:01:27 +03:00
hci_req_complete ( hdev , HCI_OP_INQUIRY , status ) ;
2006-11-19 00:14:22 +03:00
2007-10-20 15:33:56 +04:00
hci_conn_check_pending ( hdev ) ;
2011-11-04 21:16:53 +04:00
if ( ! test_and_clear_bit ( HCI_INQUIRY , & hdev - > flags ) )
return ;
2012-01-09 01:11:15 +04:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-01-04 17:44:20 +04:00
return ;
2011-11-08 22:40:16 +04:00
hci_dev_lock ( hdev ) ;
2012-01-04 17:44:20 +04:00
2012-02-18 03:39:37 +04:00
if ( discov - > state ! = DISCOVERY_FINDING )
2012-01-04 17:44:20 +04: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 22:40:16 +04:00
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_inquiry_result_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2005-08-10 07:27:49 +04:00
struct inquiry_data data ;
2007-10-20 15:33:56 +04:00
struct inquiry_info * info = ( void * ) ( skb - > data + 1 ) ;
2005-04-17 02:20:36 +04:00
int num_rsp = * ( ( __u8 * ) skb - > data ) ;
BT_DBG ( " %s num_rsp %d " , hdev - > name , num_rsp ) ;
2005-08-10 07:27:49 +04:00
if ( ! num_rsp )
return ;
2012-03-21 07:03:38 +04:00
if ( test_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) )
return ;
2005-04-17 02:20:36 +04:00
hci_dev_lock ( hdev ) ;
2005-08-10 07:27:49 +04:00
2011-03-31 00:57:16 +04:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2012-02-23 02:38:59 +04:00
bool name_known , ssp ;
2012-01-04 15:39:52 +04:00
2005-04-17 02:20:36 +04: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 22:13:48 +04:00
data . ssp_mode = 0x00 ;
2012-01-04 15:39:52 +04:00
2012-02-23 02:38:59 +04:00
name_known = hci_inquiry_cache_update ( hdev , & data , false , & ssp ) ;
2011-11-09 15:58:58 +04:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 08:25:00 +04:00
info - > dev_class , 0 , ! name_known , ssp , NULL ,
0 ) ;
2005-04-17 02:20:36 +04:00
}
2005-08-10 07:27:49 +04:00
2005-04-17 02:20:36 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_conn_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_conn_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2005-04-17 02:20:36 +04: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 21:30:03 +04: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-17 02:20:36 +04:00
if ( ! ev - > status ) {
conn - > handle = __le16_to_cpu ( ev - > handle ) ;
2008-07-14 22:13:49 +04:00
if ( conn - > type = = ACL_LINK ) {
conn - > state = BT_CONFIG ;
hci_conn_hold ( conn ) ;
2012-07-19 16:46:08 +04: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 22:13:49 +04:00
} else
conn - > state = BT_CONNECTED ;
2005-04-17 02:20:36 +04:00
2009-08-23 01:19:26 +04:00
hci_conn_hold_device ( conn ) ;
2008-07-14 22:13:51 +04:00
hci_conn_add_sysfs ( conn ) ;
2005-04-17 02:20:36 +04: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 12:02:33 +04:00
/* Get remote features */
if ( conn - > type = = ACL_LINK ) {
struct hci_cp_read_remote_features cp ;
cp . handle = ev - > handle ;
2008-07-14 22:13:49 +04:00
hci_send_cmd ( hdev , HCI_OP_READ_REMOTE_FEATURES ,
2012-03-08 08:25:00 +04:00
sizeof ( cp ) , & cp ) ;
2006-07-03 12:02:33 +04:00
}
2005-04-17 02:20:36 +04:00
/* Set packet type for incoming connection */
2011-12-01 16:33:27 +04:00
if ( ! conn - > out & & hdev - > hci_ver < BLUETOOTH_VER_2_0 ) {
2005-04-17 02:20:36 +04:00
struct hci_cp_change_conn_ptype cp ;
cp . handle = ev - > handle ;
2008-07-14 22:13:46 +04:00
cp . pkt_type = cpu_to_le16 ( conn - > pkt_type ) ;
2012-03-08 08:25:00 +04:00
hci_send_cmd ( hdev , HCI_OP_CHANGE_CONN_PTYPE , sizeof ( cp ) ,
& cp ) ;
2005-04-17 02:20:36 +04:00
}
2011-01-22 07:09:08 +03:00
} else {
2005-04-17 02:20:36 +04:00
conn - > state = BT_CLOSED ;
2011-01-22 07:09:08 +03:00
if ( conn - > type = = ACL_LINK )
2011-11-08 22:40:14 +04:00
mgmt_connect_failed ( hdev , & ev - > bdaddr , conn - > type ,
2012-03-08 08:25:00 +04:00
conn - > dst_type , ev - > status ) ;
2011-01-22 07:09:08 +03:00
}
2005-04-17 02:20:36 +04:00
2010-07-26 18:06:00 +04:00
if ( conn - > type = = ACL_LINK )
hci_sco_setup ( conn , ev - > status ) ;
2005-04-17 02:20:36 +04:00
2008-07-14 22:13:49 +04:00
if ( ev - > status ) {
hci_proto_connect_cfm ( conn , ev - > status ) ;
2005-04-17 02:20:36 +04:00
hci_conn_del ( conn ) ;
2009-01-15 23:57:03 +03:00
} else if ( ev - > link_type ! = ACL_LINK )
hci_proto_connect_cfm ( conn , ev - > status ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
unlock :
2005-04-17 02:20:36 +04:00
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
hci_conn_check_pending ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_conn_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_conn_request * ev = ( void * ) skb - > data ;
int mask = hdev - > link_mode ;
2005-04-17 02:20:36 +04:00
2012-09-25 13:49:43 +04:00
BT_DBG ( " %s bdaddr %pMR type 0x%x " , hdev - > name , & ev - > bdaddr ,
2012-05-17 07:36:24 +04:00
ev - > link_type ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
mask | = hci_proto_connect_ind ( hdev , & ev - > bdaddr , ev - > link_type ) ;
2005-04-17 02:20:36 +04:00
2011-02-17 18:44:23 +03:00
if ( ( mask & HCI_LM_ACCEPT ) & &
2012-05-17 07:36:24 +04:00
! hci_blacklist_lookup ( hdev , & ev - > bdaddr ) ) {
2007-10-20 15:33:56 +04:00
/* Connection accepted */
2008-07-14 22:13:47 +04:00
struct inquiry_entry * ie ;
2005-04-17 02:20:36 +04:00
struct hci_conn * conn ;
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
2007-10-20 16:55:10 +04:00
2010-11-22 14:21:37 +03:00
ie = hci_inquiry_cache_lookup ( hdev , & ev - > bdaddr ) ;
if ( ie )
2008-07-14 22:13:47 +04:00
memcpy ( ie - > data . dev_class , ev - > dev_class , 3 ) ;
2012-05-23 11:04:21 +04:00
conn = hci_conn_hash_lookup_ba ( hdev , ev - > link_type ,
& ev - > bdaddr ) ;
2007-10-20 15:33:56 +04:00
if ( ! conn ) {
2010-11-22 14:21:37 +03:00
conn = hci_conn_add ( hdev , ev - > link_type , & ev - > bdaddr ) ;
if ( ! conn ) {
2010-07-18 22:13:37 +04:00
BT_ERR ( " No memory for new connection " ) ;
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
return ;
2005-04-17 02:20:36 +04:00
}
}
2007-10-20 16:55:10 +04:00
2007-10-20 15:33:56 +04:00
memcpy ( conn - > dev_class , ev - > dev_class , 3 ) ;
conn - > state = BT_CONNECT ;
2007-10-20 16:55:10 +04:00
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 16:55:10 +04:00
if ( ev - > link_type = = ACL_LINK | | ! lmp_esco_capable ( hdev ) ) {
struct hci_cp_accept_conn_req cp ;
2005-04-17 02:20:36 +04:00
2007-10-20 16:55:10 +04: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 08:25:00 +04:00
hci_send_cmd ( hdev , HCI_OP_ACCEPT_CONN_REQ , sizeof ( cp ) ,
& cp ) ;
2007-10-20 16:55:10 +04:00
} else {
struct hci_cp_accept_sync_conn_req cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2008-07-14 22:13:46 +04:00
cp . pkt_type = cpu_to_le16 ( conn - > pkt_type ) ;
2007-10-20 16:55:10 +04:00
2012-05-25 12:38:27 +04: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 16:55:10 +04:00
cp . content_format = cpu_to_le16 ( hdev - > voice_setting ) ;
cp . retrans_effort = 0xff ;
2005-04-17 02:20:36 +04:00
2007-10-20 16:55:10 +04:00
hci_send_cmd ( hdev , HCI_OP_ACCEPT_SYNC_CONN_REQ ,
2012-03-08 08:25:00 +04:00
sizeof ( cp ) , & cp ) ;
2007-10-20 16:55:10 +04:00
}
2007-10-20 15:33:56 +04:00
} else {
/* Connection rejected */
struct hci_cp_reject_conn_req cp ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2011-11-07 16:20:25 +04:00
cp . reason = HCI_ERROR_REJ_BAD_ADDR ;
2007-10-20 15:33:56 +04:00
hci_send_cmd ( hdev , HCI_OP_REJECT_CONN_REQ , sizeof ( cp ) , & cp ) ;
2005-04-17 02:20:36 +04:00
}
}
2012-08-09 11:52:30 +04: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 11:04:18 +04:00
static void hci_disconn_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2006-07-03 12:02:33 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_disconn_complete * ev = ( void * ) skb - > data ;
2006-07-03 12:02:33 +04:00
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2006-07-03 12:02:33 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2011-01-20 13:34:39 +03:00
if ( ! conn )
goto unlock ;
2008-07-14 22:13:51 +04:00
2011-11-10 17:54:39 +04:00
if ( ev - > status = = 0 )
conn - > state = BT_CLOSED ;
2006-07-03 12:02:33 +04:00
2012-01-17 23:48:47 +04:00
if ( test_and_clear_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) & &
2012-05-17 07:36:24 +04:00
( conn - > type = = ACL_LINK | | conn - > type = = LE_LINK ) ) {
2012-08-09 11:52:30 +04:00
if ( ev - > status ) {
2012-02-09 16:27:38 +04:00
mgmt_disconnect_failed ( hdev , & conn - > dst , conn - > type ,
2012-05-17 07:36:24 +04:00
conn - > dst_type , ev - > status ) ;
2012-08-09 11:52:30 +04:00
} else {
u8 reason = hci_to_mgmt_reason ( ev - > reason ) ;
2012-01-15 20:11:07 +04:00
mgmt_device_disconnected ( hdev , & conn - > dst , conn - > type ,
2012-08-09 11:52:30 +04:00
conn - > dst_type , reason ) ;
}
2011-11-10 17:54:39 +04:00
}
2011-01-20 13:34:39 +03:00
2011-11-10 17:54:39 +04:00
if ( ev - > status = = 0 ) {
2012-04-16 13:14:44 +04:00
if ( conn - > type = = ACL_LINK & & conn - > flush_key )
hci_remove_link_key ( hdev , & conn - > dst ) ;
2011-11-10 17:54:39 +04:00
hci_proto_disconn_cfm ( conn , ev - > reason ) ;
hci_conn_del ( conn ) ;
}
2011-01-20 13:34:39 +03:00
unlock :
2006-07-03 12:02:33 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_auth_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_auth_complete * ev = ( void * ) skb - > data ;
2006-07-03 12:02:33 +04:00
struct hci_conn * conn ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-17 02:20:36 +04:00
hci_dev_lock ( hdev ) ;
2006-07-03 12:02:33 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2011-05-31 17:49:26 +04:00
if ( ! conn )
goto unlock ;
if ( ! ev - > status ) {
2012-01-18 23:33:12 +04:00
if ( ! hci_conn_ssp_enabled ( conn ) & &
2012-05-17 07:36:24 +04:00
test_bit ( HCI_CONN_REAUTH_PEND , & conn - > flags ) ) {
2011-05-31 17:49:26 +04:00
BT_INFO ( " re-auth of legacy device is not possible. " ) ;
2011-02-19 18:06:00 +03:00
} else {
2011-05-31 17:49:26 +04:00
conn - > link_mode | = HCI_LM_AUTH ;
conn - > sec_level = conn - > pending_sec_level ;
2011-02-19 18:06:00 +03:00
}
2011-05-31 17:49:26 +04:00
} else {
2012-02-09 18:07:29 +04:00
mgmt_auth_failed ( hdev , & conn - > dst , conn - > type , conn - > dst_type ,
2012-03-08 08:25:00 +04:00
ev - > status ) ;
2011-05-31 17:49:26 +04:00
}
2005-04-17 02:20:36 +04:00
2012-01-16 08:10:31 +04:00
clear_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ;
clear_bit ( HCI_CONN_REAUTH_PEND , & conn - > flags ) ;
2005-04-17 02:20:36 +04:00
2011-05-31 17:49:26 +04:00
if ( conn - > state = = BT_CONFIG ) {
2012-01-18 23:33:12 +04:00
if ( ! ev - > status & & hci_conn_ssp_enabled ( conn ) ) {
2011-05-31 17:49:26 +04: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 07:36:24 +04:00
& cp ) ;
2009-04-26 22:01:22 +04:00
} else {
2011-05-31 17:49:26 +04:00
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
2009-04-26 22:01:22 +04:00
hci_conn_put ( conn ) ;
}
2011-05-31 17:49:26 +04:00
} else {
hci_auth_cfm ( conn , ev - > status ) ;
2009-04-26 22:01:22 +04:00
2011-05-31 17:49:26 +04:00
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_DISCONN_TIMEOUT ;
hci_conn_put ( conn ) ;
}
2012-01-16 08:10:31 +04:00
if ( test_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ) {
2011-05-31 17:49:26 +04: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 07:36:24 +04:00
& cp ) ;
2011-05-31 17:49:26 +04:00
} else {
2012-01-16 08:10:31 +04:00
clear_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ;
2011-05-31 17:49:26 +04:00
hci_encrypt_cfm ( conn , ev - > status , 0x00 ) ;
2005-04-17 02:20:36 +04:00
}
}
2011-05-31 17:49:26 +04:00
unlock :
2005-04-17 02:20:36 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_remote_name_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2010-11-18 23:22:29 +03:00
struct hci_ev_remote_name * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
hci_conn_check_pending ( hdev ) ;
2010-11-18 23:22:29 +03:00
hci_dev_lock ( hdev ) ;
2012-01-17 23:48:47 +04:00
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2012-01-04 17:44:20 +04:00
2012-01-17 23:48:47 +04:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
goto check_auth ;
2011-03-30 14:18:12 +04:00
2012-01-17 23:48:47 +04:00
if ( ev - > status = = 0 )
hci_check_pending_name ( hdev , conn , & ev - > bdaddr , ev - > name ,
2012-03-08 08:25:00 +04:00
strnlen ( ev - > name , HCI_MAX_NAME_LENGTH ) ) ;
2012-01-17 23:48:47 +04:00
else
hci_check_pending_name ( hdev , conn , & ev - > bdaddr , NULL , 0 ) ;
check_auth :
2011-04-28 22:28:55 +04:00
if ( ! conn )
goto unlock ;
if ( ! hci_outgoing_auth_needed ( hdev , conn ) )
goto unlock ;
2012-01-16 08:10:31 +04:00
if ( ! test_and_set_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ) {
2010-11-18 23:22:29 +03: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 22:28:55 +04:00
unlock :
2010-11-18 23:22:29 +03:00
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_encrypt_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct hci_ev_encrypt_change * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-17 02:20:36 +04:00
hci_dev_lock ( hdev ) ;
2006-07-03 12:02:33 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2005-04-17 02:20:36 +04:00
if ( conn ) {
if ( ! ev - > status ) {
2008-07-14 22:13:45 +04:00
if ( ev - > encrypt ) {
/* Encryption implies authentication */
conn - > link_mode | = HCI_LM_AUTH ;
2005-04-17 02:20:36 +04:00
conn - > link_mode | = HCI_LM_ENCRYPT ;
2011-06-10 01:50:53 +04:00
conn - > sec_level = conn - > pending_sec_level ;
2008-07-14 22:13:45 +04:00
} else
2005-04-17 02:20:36 +04:00
conn - > link_mode & = ~ HCI_LM_ENCRYPT ;
}
2012-01-16 08:10:31 +04:00
clear_bit ( HCI_CONN_ENCRYPT_PEND , & conn - > flags ) ;
2005-04-17 02:20:36 +04:00
2012-05-13 10:20:07 +04:00
if ( ev - > status & & conn - > state = = BT_CONNECTED ) {
2012-05-16 19:17:12 +04:00
hci_acl_disconn ( conn , HCI_ERROR_AUTH_FAILURE ) ;
2012-05-13 10:20:07 +04:00
hci_conn_put ( conn ) ;
goto unlock ;
}
2008-07-14 22:13:49 +04:00
if ( conn - > state = = BT_CONFIG ) {
if ( ! ev - > status )
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
hci_conn_put ( conn ) ;
} else
hci_encrypt_cfm ( conn , ev - > status , ev - > encrypt ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-13 10:20:07 +04:00
unlock :
2005-04-17 02:20:36 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_change_link_key_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_change_link_key_complete * ev = ( void * ) skb - > data ;
2006-07-03 12:02:33 +04:00
struct hci_conn * conn ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-17 02:20:36 +04:00
hci_dev_lock ( hdev ) ;
2006-07-03 12:02:33 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2005-04-17 02:20:36 +04:00
if ( conn ) {
if ( ! ev - > status )
conn - > link_mode | = HCI_LM_SECURE ;
2012-01-16 08:10:31 +04:00
clear_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ;
2005-04-17 02:20:36 +04:00
hci_key_change_cfm ( conn , ev - > status ) ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_remote_features_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_remote_features * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2010-11-10 18:11:51 +03:00
if ( ! conn )
goto unlock ;
2008-07-14 22:13:49 +04:00
2010-11-10 18:11:51 +03:00
if ( ! ev - > status )
memcpy ( conn - > features , ev - > features , 8 ) ;
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 07:36:24 +04:00
sizeof ( cp ) , & cp ) ;
2010-11-18 23:22:28 +03:00
goto unlock ;
}
2012-05-12 23:11:50 +04:00
if ( ! ev - > status & & ! test_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) ) {
2010-11-18 23:22:29 +03: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 23:48:47 +04:00
} else if ( ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
mgmt_device_connected ( hdev , & conn - > dst , conn - > type ,
2012-03-08 08:25:00 +04:00
conn - > dst_type , 0 , NULL , 0 ,
conn - > dev_class ) ;
2010-11-18 23:22:28 +03:00
2010-11-18 23:22:29 +03:00
if ( ! hci_outgoing_auth_needed ( hdev , conn ) ) {
2010-11-10 18:11:51 +03:00
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
hci_conn_put ( conn ) ;
2008-07-14 22:13:49 +04:00
}
2007-10-20 15:33:56 +04:00
2010-11-10 18:11:51 +03:00
unlock :
2007-10-20 15:33:56 +04:00
hci_dev_unlock ( hdev ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_remote_version_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_qos_setup_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2005-04-17 02:20:36 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_cmd_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct hci_ev_cmd_complete * ev = ( void * ) skb - > data ;
__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 07:03:35 +04:00
case HCI_OP_PERIODIC_INQ :
hci_cc_periodic_inq ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04: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 22:13:47 +04:00
case HCI_OP_READ_LINK_POLICY :
hci_cc_read_link_policy ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04:00
case HCI_OP_WRITE_LINK_POLICY :
hci_cc_write_link_policy ( hdev , skb ) ;
break ;
2008-07-14 22:13:47 +04: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 15:33:56 +04: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 ;
case HCI_OP_HOST_BUFFER_SIZE :
hci_cc_host_buffer_size ( hdev , skb ) ;
break ;
2008-07-14 22:13:48 +04:00
case HCI_OP_WRITE_SSP_MODE :
hci_cc_write_ssp_mode ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04: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-07-01 02:20:52 +04:00
case HCI_OP_READ_LOCAL_EXT_FEATURES :
hci_cc_read_local_ext_features ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04: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 ;
2011-12-07 17:56:51 +04:00
case HCI_OP_READ_DATA_BLOCK_SIZE :
hci_cc_read_data_block_size ( hdev , skb ) ;
break ;
2010-12-22 00:01:27 +03:00
case HCI_OP_WRITE_CA_TIMEOUT :
hci_cc_write_ca_timeout ( hdev , skb ) ;
break ;
2011-11-24 16:52:02 +04:00
case HCI_OP_READ_FLOW_CONTROL_MODE :
hci_cc_read_flow_control_mode ( hdev , skb ) ;
break ;
2011-10-12 11:53:57 +04:00
case HCI_OP_READ_LOCAL_AMP_INFO :
hci_cc_read_local_amp_info ( hdev , skb ) ;
break ;
2012-09-27 18:26:09 +04:00
case HCI_OP_READ_LOCAL_AMP_ASSOC :
hci_cc_read_local_amp_assoc ( hdev , skb ) ;
break ;
2011-01-10 14:44:55 +03:00
case HCI_OP_DELETE_STORED_LINK_KEY :
hci_cc_delete_stored_link_key ( hdev , skb ) ;
break ;
2011-01-25 02:19:58 +03:00
case HCI_OP_SET_EVENT_MASK :
hci_cc_set_event_mask ( hdev , skb ) ;
break ;
case HCI_OP_WRITE_INQUIRY_MODE :
hci_cc_write_inquiry_mode ( hdev , skb ) ;
break ;
case HCI_OP_READ_INQ_RSP_TX_POWER :
hci_cc_read_inq_rsp_tx_power ( hdev , skb ) ;
break ;
case HCI_OP_SET_EVENT_FLT :
hci_cc_set_event_flt ( hdev , skb ) ;
break ;
2011-01-22 07:10:07 +03: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 15:12:21 +03:00
case HCI_OP_READ_LOCAL_OOB_DATA :
hci_cc_read_local_oob_data_reply ( hdev , skb ) ;
break ;
2011-02-11 04:38:48 +03:00
case HCI_OP_LE_READ_BUFFER_SIZE :
hci_cc_le_read_buffer_size ( hdev , skb ) ;
break ;
2012-10-19 21:57:49 +04:00
case HCI_OP_LE_READ_ADV_TX_POWER :
hci_cc_le_read_adv_tx_power ( hdev , skb ) ;
break ;
2012-10-19 21:57:47 +04:00
case HCI_OP_LE_SET_EVENT_MASK :
hci_cc_le_set_event_mask ( hdev , skb ) ;
break ;
2011-02-19 18: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 20:28:34 +04: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 14:32:42 +04:00
break ;
2011-12-02 16:13:31 +04:00
case HCI_OP_LE_SET_SCAN_PARAM :
hci_cc_le_set_scan_param ( hdev , skb ) ;
2011-11-23 20:28:34 +04:00
break ;
2011-05-26 23:23:52 +04:00
case HCI_OP_LE_SET_SCAN_ENABLE :
hci_cc_le_set_scan_enable ( hdev , skb ) ;
break ;
2011-06-10 01:50:47 +04:00
case HCI_OP_LE_LTK_REPLY :
hci_cc_le_ltk_reply ( hdev , skb ) ;
break ;
case HCI_OP_LE_LTK_NEG_REPLY :
hci_cc_le_ltk_neg_reply ( hdev , skb ) ;
break ;
2011-07-01 02:20:53 +04:00
case HCI_OP_WRITE_LE_HOST_SUPPORTED :
hci_cc_write_le_host_supported ( hdev , skb ) ;
break ;
2012-09-27 18:26:20 +04:00
case HCI_OP_WRITE_REMOTE_AMP_ASSOC :
hci_cc_write_remote_amp_assoc ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04:00
default :
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s opcode 0x%4.4x " , hdev - > name , opcode ) ;
2007-10-20 15:33:56 +04:00
break ;
}
2011-02-16 17:32:41 +03:00
if ( ev - > opcode ! = HCI_OP_NOP )
del_timer ( & hdev - > cmd_timer ) ;
2007-10-20 15:33:56 +04:00
if ( ev - > ncmd ) {
atomic_set ( & hdev - > cmd_cnt , 1 ) ;
if ( ! skb_queue_empty ( & hdev - > cmd_q ) )
2011-12-15 05:53:47 +04:00
queue_work ( hdev - > workqueue , & hdev - > cmd_work ) ;
2007-10-20 15:33:56 +04:00
}
}
2012-05-23 11:04:18 +04:00
static void hci_cmd_status_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04: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 22:13:49 +04: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 15:33:56 +04:00
case HCI_OP_REMOTE_NAME_REQ :
hci_cs_remote_name_req ( hdev , ev - > status ) ;
break ;
2008-07-14 22:13:49 +04: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 15:33:56 +04: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 13:40:27 +03:00
case HCI_OP_DISCONNECT :
2012-02-09 16:27:38 +04:00
hci_cs_disconnect ( hdev , ev - > status ) ;
2011-01-20 13:40:27 +03:00
break ;
2011-02-11 04:38:47 +03:00
case HCI_OP_LE_CREATE_CONN :
hci_cs_le_create_conn ( hdev , ev - > status ) ;
break ;
2011-06-10 01:50:47 +04:00
case HCI_OP_LE_START_ENC :
hci_cs_le_start_enc ( hdev , ev - > status ) ;
break ;
2012-09-27 18:26:19 +04:00
case HCI_OP_CREATE_PHY_LINK :
hci_cs_create_phylink ( hdev , ev - > status ) ;
break ;
2012-09-27 18:26:24 +04:00
case HCI_OP_ACCEPT_PHY_LINK :
hci_cs_accept_phylink ( hdev , ev - > status ) ;
break ;
2007-10-20 15:33:56 +04:00
default :
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s opcode 0x%4.4x " , hdev - > name , opcode ) ;
2007-10-20 15:33:56 +04:00
break ;
}
2011-02-16 17:32:41 +03:00
if ( ev - > opcode ! = HCI_OP_NOP )
del_timer ( & hdev - > cmd_timer ) ;
2011-03-16 21:36:29 +03:00
if ( ev - > ncmd & & ! test_bit ( HCI_RESET , & hdev - > flags ) ) {
2007-10-20 15:33:56 +04:00
atomic_set ( & hdev - > cmd_cnt , 1 ) ;
if ( ! skb_queue_empty ( & hdev - > cmd_q ) )
2011-12-15 05:53:47 +04:00
queue_work ( hdev - > workqueue , & hdev - > cmd_work ) ;
2007-10-20 15:33:56 +04:00
}
}
2012-05-23 11:04:18 +04:00
static void hci_role_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct hci_ev_role_change * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2007-10-20 15:33:56 +04: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 08:10:31 +04:00
clear_bit ( HCI_CONN_RSWITCH_PEND , & conn - > flags ) ;
2007-10-20 15:33:56 +04:00
hci_role_switch_cfm ( conn , ev - > status , ev - > role ) ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_num_comp_pkts_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct hci_ev_num_comp_pkts * ev = ( void * ) skb - > data ;
int i ;
2011-12-19 18:31:29 +04: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 14:07:47 +04:00
if ( skb - > len < sizeof ( * ev ) | | skb - > len < sizeof ( * ev ) +
2012-05-17 07:36:24 +04:00
ev - > num_hndl * sizeof ( struct hci_comp_pkts_info ) ) {
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s bad parameters " , hdev - > name ) ;
return ;
}
2011-12-30 14:07:47 +04:00
BT_DBG ( " %s num_hndl %d " , hdev - > name , ev - > num_hndl ) ;
2011-12-19 18:31:30 +04:00
for ( i = 0 ; i < ev - > num_hndl ; i + + ) {
struct hci_comp_pkts_info * info = & ev - > handles [ i ] ;
2007-10-20 15:33:56 +04:00
struct hci_conn * conn ;
__u16 handle , count ;
2011-12-19 18:31:30 +04:00
handle = __le16_to_cpu ( info - > handle ) ;
count = __le16_to_cpu ( info - > count ) ;
2007-10-20 15:33:56 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , handle ) ;
2011-12-07 17:56:52 +04: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 17:58:25 +03:00
hdev - > acl_cnt + = count ;
if ( hdev - > acl_cnt > hdev - > acl_pkts )
2007-10-20 15:33:56 +04:00
hdev - > acl_cnt = hdev - > acl_pkts ;
}
2011-12-07 17:56:52 +04: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 15:33:56 +04:00
}
}
2011-12-15 06:50:02 +04:00
queue_work ( hdev - > workqueue , & hdev - > tx_work ) ;
2007-10-20 15:33:56 +04:00
}
2012-10-10 18:38:29 +04: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 11:04:18 +04:00
static void hci_num_comp_blocks_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2012-01-04 14:41:58 +04: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 07:36:24 +04:00
ev - > num_hndl * sizeof ( struct hci_comp_blocks_info ) ) {
2012-01-04 14:41:58 +04: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 07:36:24 +04:00
ev - > num_hndl ) ;
2012-01-04 14:41:58 +04:00
for ( i = 0 ; i < ev - > num_hndl ; i + + ) {
struct hci_comp_blocks_info * info = & ev - > handles [ i ] ;
2012-10-10 18:38:29 +04:00
struct hci_conn * conn = NULL ;
2012-01-04 14:41:58 +04:00
__u16 handle , block_count ;
handle = __le16_to_cpu ( info - > handle ) ;
block_count = __le16_to_cpu ( info - > blocks ) ;
2012-10-10 18:38:29 +04:00
conn = __hci_conn_lookup_handle ( hdev , handle ) ;
2012-01-04 14:41:58 +04:00
if ( ! conn )
continue ;
conn - > sent - = block_count ;
switch ( conn - > type ) {
case ACL_LINK :
2012-10-10 18:38:30 +04:00
case AMP_LINK :
2012-01-04 14:41:58 +04: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 11:04:18 +04:00
static void hci_mode_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2006-07-03 12:02:33 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_mode_change * ev = ( void * ) skb - > data ;
2006-07-03 12:02:33 +04:00
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2006-07-03 12:02:33 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2007-10-20 15:33:56 +04:00
if ( conn ) {
conn - > mode = ev - > mode ;
conn - > interval = __le16_to_cpu ( ev - > interval ) ;
2012-05-23 11:04:21 +04:00
if ( ! test_and_clear_bit ( HCI_CONN_MODE_CHANGE_PEND ,
& conn - > flags ) ) {
2007-10-20 15:33:56 +04:00
if ( conn - > mode = = HCI_CM_ACTIVE )
2012-01-16 08:47:28 +04:00
set_bit ( HCI_CONN_POWER_SAVE , & conn - > flags ) ;
2007-10-20 15:33:56 +04:00
else
2012-01-16 08:47:28 +04:00
clear_bit ( HCI_CONN_POWER_SAVE , & conn - > flags ) ;
2007-10-20 15:33:56 +04:00
}
2010-07-26 18:06:00 +04:00
2012-01-16 08:10:31 +04:00
if ( test_and_clear_bit ( HCI_CONN_SCO_SETUP_PEND , & conn - > flags ) )
2010-07-26 18:06:00 +04:00
hci_sco_setup ( conn , ev - > status ) ;
2006-07-03 12:02:33 +04:00
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_pin_code_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
2009-04-26 22:01:22 +04:00
struct hci_ev_pin_code_req * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2009-04-26 22:01:22 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ACL_LINK , & ev - > bdaddr ) ;
2011-09-23 12:01:30 +04:00
if ( ! conn )
goto unlock ;
if ( conn - > state = = BT_CONNECTED ) {
2009-04-26 22:01:22 +04:00
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_PAIRING_TIMEOUT ;
hci_conn_put ( conn ) ;
}
2012-01-09 01:11:15 +04:00
if ( ! test_bit ( HCI_PAIRABLE , & hdev - > dev_flags ) )
2011-01-04 16:40:05 +03:00
hci_send_cmd ( hdev , HCI_OP_PIN_CODE_NEG_REPLY ,
2012-05-17 07:36:24 +04:00
sizeof ( ev - > bdaddr ) , & ev - > bdaddr ) ;
2012-01-09 01:11:15 +04:00
else if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) ) {
2011-04-28 14:07:59 +04:00
u8 secure ;
if ( conn - > pending_sec_level = = BT_SECURITY_HIGH )
secure = 1 ;
else
secure = 0 ;
2011-11-08 22:40:14 +04:00
mgmt_pin_code_request ( hdev , & ev - > bdaddr , secure ) ;
2011-04-28 14:07:59 +04:00
}
2011-01-22 07:10:07 +03:00
2011-09-23 12:01:30 +04:00
unlock :
2009-04-26 22:01:22 +04:00
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_link_key_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
2011-01-17 15:41:05 +03: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 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2011-01-17 15:41:05 +03:00
2012-01-09 01:11:15 +04:00
if ( ! test_bit ( HCI_LINK_KEYS , & hdev - > dev_flags ) )
2011-01-17 15:41:05 +03:00
return ;
hci_dev_lock ( hdev ) ;
key = hci_find_link_key ( hdev , & ev - > bdaddr ) ;
if ( ! key ) {
2012-09-25 13:49:43 +04:00
BT_DBG ( " %s link key not found for %pMR " , hdev - > name ,
& ev - > bdaddr ) ;
2011-01-17 15:41:05 +03:00
goto not_found ;
}
2012-09-25 13:49:43 +04:00
BT_DBG ( " %s found key type %u for %pMR " , hdev - > name , key - > type ,
& ev - > bdaddr ) ;
2011-01-17 15:41:05 +03:00
2012-01-09 01:11:15 +04:00
if ( ! test_bit ( HCI_DEBUG_KEYS , & hdev - > dev_flags ) & &
2012-05-17 07:36:24 +04:00
key - > type = = HCI_LK_DEBUG_COMBINATION ) {
2011-01-17 15:41:05 +03: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 14:07:56 +04:00
if ( conn ) {
if ( key - > type = = HCI_LK_UNAUTH_COMBINATION & &
2012-05-17 07:36:24 +04:00
conn - > auth_type ! = 0xff & & ( conn - > auth_type & 0x01 ) ) {
2011-04-28 14:07:56 +04:00
BT_DBG ( " %s ignoring unauthenticated key " , hdev - > name ) ;
goto not_found ;
}
2011-01-17 15:41:05 +03:00
2011-04-28 14:07:56 +04:00
if ( key - > type = = HCI_LK_COMBINATION & & key - > pin_len < 16 & &
2012-05-17 07:36:24 +04:00
conn - > pending_sec_level = = BT_SECURITY_HIGH ) {
2012-05-23 11:04:21 +04:00
BT_DBG ( " %s ignoring key unauthenticated for high security " ,
hdev - > name ) ;
2011-04-28 14:07:56 +04:00
goto not_found ;
}
conn - > key_type = key - > type ;
conn - > pin_length = key - > pin_len ;
2011-01-17 15:41:05 +03:00
}
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2012-05-23 12:31:20 +04:00
memcpy ( cp . link_key , key - > val , HCI_LINK_KEY_SIZE ) ;
2011-01-17 15:41:05 +03: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 15:33:56 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_link_key_notify_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
2009-04-26 22:01:22 +04:00
struct hci_ev_link_key_notify * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2011-01-17 15:41:05 +03:00
u8 pin_len = 0 ;
2009-04-26 22:01:22 +04:00
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2009-04-26 22:01:22 +04: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 07:10:07 +03:00
pin_len = conn - > pin_length ;
2011-04-28 14:07:55 +04:00
if ( ev - > key_type ! = HCI_LK_CHANGED_COMBINATION )
conn - > key_type = ev - > key_type ;
2009-04-26 22:01:22 +04:00
hci_conn_put ( conn ) ;
}
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_LINK_KEYS , & hdev - > dev_flags ) )
2011-04-28 22:28:59 +04:00
hci_add_link_key ( hdev , conn , 1 , & ev - > bdaddr , ev - > link_key ,
2012-05-17 07:36:24 +04:00
ev - > key_type , pin_len ) ;
2011-01-17 15:41:05 +03:00
2009-04-26 22:01:22 +04:00
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_clock_offset_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_clock_offset * ev = ( void * ) skb - > data ;
2006-07-03 12:02:33 +04:00
struct hci_conn * conn ;
2005-04-17 02:20:36 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2005-04-17 02:20:36 +04:00
hci_dev_lock ( hdev ) ;
2006-07-03 12:02:33 +04:00
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2005-04-17 02:20:36 +04:00
if ( conn & & ! ev - > status ) {
struct inquiry_entry * ie ;
2010-11-22 14:21:37 +03:00
ie = hci_inquiry_cache_lookup ( hdev , & conn - > dst ) ;
if ( ie ) {
2005-04-17 02:20:36 +04:00
ie - > data . clock_offset = ev - > clock_offset ;
ie - > timestamp = jiffies ;
}
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_pkt_type_change_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2008-07-14 22:13:46 +04:00
{
struct hci_ev_pkt_type_change * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2008-07-14 22:13:46 +04: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 11:04:18 +04:00
static void hci_pscan_rep_mode_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2005-08-10 07:28:02 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_pscan_rep_mode * ev = ( void * ) skb - > data ;
2005-08-10 07:28:02 +04:00
struct inquiry_entry * ie ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2010-11-22 14:21:37 +03:00
ie = hci_inquiry_cache_lookup ( hdev , & ev - > bdaddr ) ;
if ( ie ) {
2005-08-10 07:28:02 +04:00
ie - > data . pscan_rep_mode = ev - > pscan_rep_mode ;
ie - > timestamp = jiffies ;
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_inquiry_result_with_rssi_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
struct inquiry_data data ;
int num_rsp = * ( ( __u8 * ) skb - > data ) ;
2012-02-23 02:38:59 +04:00
bool name_known , ssp ;
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s num_rsp %d " , hdev - > name , num_rsp ) ;
if ( ! num_rsp )
return ;
2012-03-21 07:03:38 +04:00
if ( test_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) )
return ;
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
if ( ( skb - > len - 1 ) / num_rsp ! = sizeof ( struct inquiry_info_with_rssi ) ) {
2011-02-17 18:44:23 +03:00
struct inquiry_info_with_rssi_and_pscan_mode * info ;
info = ( void * ) ( skb - > data + 1 ) ;
2007-10-20 15:33:56 +04:00
2011-03-31 00:57:16 +04:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2007-10-20 15:33:56 +04: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 22:13:48 +04:00
data . ssp_mode = 0x00 ;
2012-01-04 15:39:52 +04:00
name_known = hci_inquiry_cache_update ( hdev , & data ,
2012-03-08 08:25:00 +04:00
false , & ssp ) ;
2011-11-09 15:58:58 +04:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 08:25:00 +04:00
info - > dev_class , info - > rssi ,
! name_known , ssp , NULL , 0 ) ;
2007-10-20 15:33:56 +04:00
}
} else {
struct inquiry_info_with_rssi * info = ( void * ) ( skb - > data + 1 ) ;
2011-03-31 00:57:16 +04:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2007-10-20 15:33:56 +04: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 22:13:48 +04:00
data . ssp_mode = 0x00 ;
2012-01-04 15:39:52 +04:00
name_known = hci_inquiry_cache_update ( hdev , & data ,
2012-03-08 08:25:00 +04:00
false , & ssp ) ;
2011-11-09 15:58:58 +04:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 08:25:00 +04:00
info - > dev_class , info - > rssi ,
! name_known , ssp , NULL , 0 ) ;
2007-10-20 15:33:56 +04:00
}
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_remote_ext_features_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
2008-07-14 22:13:48 +04:00
struct hci_ev_remote_ext_features * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s " , hdev - > name ) ;
2008-07-14 22:13:48 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2010-11-10 18:11:51 +03:00
if ( ! conn )
goto unlock ;
2008-07-14 22:13:48 +04:00
2010-11-10 18:11:51 +03:00
if ( ! ev - > status & & ev - > page = = 0x01 ) {
struct inquiry_entry * ie ;
2008-07-14 22:13:48 +04:00
2010-11-22 14:21:37 +03:00
ie = hci_inquiry_cache_lookup ( hdev , & conn - > dst ) ;
if ( ie )
2012-02-28 04:28:43 +04:00
ie - > data . ssp_mode = ( ev - > features [ 0 ] & LMP_HOST_SSP ) ;
2008-07-14 22:13:49 +04:00
2012-02-28 04:28:43 +04:00
if ( ev - > features [ 0 ] & LMP_HOST_SSP )
2012-01-16 08:47:28 +04:00
set_bit ( HCI_CONN_SSP_ENABLED , & conn - > flags ) ;
2010-11-10 18:11:51 +03:00
}
if ( conn - > state ! = BT_CONFIG )
goto unlock ;
2012-05-12 23:11:50 +04:00
if ( ! ev - > status & & ! test_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) ) {
2010-11-18 23:22:29 +03: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 23:48:47 +04:00
} else if ( ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
mgmt_device_connected ( hdev , & conn - > dst , conn - > type ,
2012-03-08 08:25:00 +04:00
conn - > dst_type , 0 , NULL , 0 ,
conn - > dev_class ) ;
2010-11-18 23:22:28 +03:00
2010-11-18 23:22:29 +03:00
if ( ! hci_outgoing_auth_needed ( hdev , conn ) ) {
2010-11-10 18:11:51 +03:00
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
hci_conn_put ( conn ) ;
2008-07-14 22:13:48 +04:00
}
2010-11-10 18:11:51 +03:00
unlock :
2008-07-14 22:13:48 +04:00
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_sync_conn_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
2007-10-20 16:55:10 +04:00
struct hci_ev_sync_conn_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2007-10-20 16:55:10 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_ba ( hdev , ev - > link_type , & ev - > bdaddr ) ;
2008-07-14 22:13:46 +04: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 16:55:10 +04:00
2009-04-19 21:14:14 +04:00
switch ( ev - > status ) {
case 0x00 :
2007-10-20 16:55:10 +04:00
conn - > handle = __le16_to_cpu ( ev - > handle ) ;
conn - > state = BT_CONNECTED ;
2008-07-14 22:13:51 +04:00
2009-08-23 01:19:26 +04:00
hci_conn_hold_device ( conn ) ;
2008-07-14 22:13:51 +04:00
hci_conn_add_sysfs ( conn ) ;
2009-04-19 21:14:14 +04:00
break ;
2010-02-16 19:29:44 +03:00
case 0x11 : /* Unsupported Feature or Parameter Value */
2009-04-19 21:14:14 +04:00
case 0x1c : /* SCO interval rejected */
2010-02-03 22:42:26 +03:00
case 0x1a : /* Unsupported Remote Feature */
2009-04-19 21:14:14 +04:00
case 0x1f : /* Unspecified error */
if ( conn - > out & & conn - > attempt < 2 ) {
conn - > pkt_type = ( hdev - > esco_type & SCO_ESCO_MASK ) |
( hdev - > esco_type & EDR_ESCO_MASK ) ;
hci_setup_sync ( conn , conn - > link - > handle ) ;
goto unlock ;
}
/* fall through */
default :
2007-10-20 16:55:10 +04:00
conn - > state = BT_CLOSED ;
2009-04-19 21:14:14 +04:00
break ;
}
2007-10-20 16:55:10 +04:00
hci_proto_connect_cfm ( conn , ev - > status ) ;
if ( ev - > status )
hci_conn_del ( conn ) ;
unlock :
hci_dev_unlock ( hdev ) ;
2007-10-20 15:33:56 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_sync_conn_changed_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2007-10-20 15:33:56 +04:00
{
BT_DBG ( " %s " , hdev - > name ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_sniff_subrate_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2006-07-03 12:02:33 +04:00
{
2007-10-20 15:33:56 +04:00
struct hci_ev_sniff_subrate * ev = ( void * ) skb - > data ;
2006-07-03 12:02:33 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2006-07-03 12:02:33 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_extended_inquiry_result_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2005-04-17 02:20:36 +04:00
{
2007-10-20 15:33:56 +04:00
struct inquiry_data data ;
struct extended_inquiry_info * info = ( void * ) ( skb - > data + 1 ) ;
int num_rsp = * ( ( __u8 * ) skb - > data ) ;
2012-04-26 17:49:56 +04:00
size_t eir_len ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
BT_DBG ( " %s num_rsp %d " , hdev - > name , num_rsp ) ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
if ( ! num_rsp )
return ;
2005-04-17 02:20:36 +04:00
2012-03-21 07:03:38 +04:00
if ( test_bit ( HCI_PERIODIC_INQ , & hdev - > dev_flags ) )
return ;
2007-10-20 15:33:56 +04:00
hci_dev_lock ( hdev ) ;
2011-03-31 00:57:16 +04:00
for ( ; num_rsp ; num_rsp - - , info + + ) {
2012-02-23 02:38:59 +04:00
bool name_known , ssp ;
2012-01-04 15:31:59 +04:00
2007-10-20 15:33:56 +04:00
bacpy ( & data . bdaddr , & info - > bdaddr ) ;
2011-02-17 18:44:23 +03:00
data . pscan_rep_mode = info - > pscan_rep_mode ;
data . pscan_period_mode = info - > pscan_period_mode ;
data . pscan_mode = 0x00 ;
2007-10-20 15:33:56 +04:00
memcpy ( data . dev_class , info - > dev_class , 3 ) ;
2011-02-17 18:44:23 +03:00
data . clock_offset = info - > clock_offset ;
data . rssi = info - > rssi ;
2008-07-14 22:13:48 +04:00
data . ssp_mode = 0x01 ;
2012-01-04 15:31:59 +04:00
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-01-15 22:04:43 +04:00
name_known = eir_has_data_type ( info - > data ,
2012-03-08 08:25:00 +04:00
sizeof ( info - > data ) ,
EIR_NAME_COMPLETE ) ;
2012-01-04 15:31:59 +04:00
else
name_known = true ;
2012-02-23 02:38:59 +04:00
name_known = hci_inquiry_cache_update ( hdev , & data , name_known ,
2012-03-08 08:25:00 +04:00
& ssp ) ;
2012-04-26 17:49:56 +04:00
eir_len = eir_get_length ( info - > data , sizeof ( info - > data ) ) ;
2011-11-09 15:58:58 +04:00
mgmt_device_found ( hdev , & info - > bdaddr , ACL_LINK , 0x00 ,
2012-03-08 08:25:00 +04:00
info - > dev_class , info - > rssi , ! name_known ,
2012-04-26 17:49:56 +04:00
ssp , info - > data , eir_len ) ;
2007-10-20 15:33:56 +04:00
}
hci_dev_unlock ( hdev ) ;
}
2005-04-17 02:20:36 +04:00
2012-06-08 19:31:13 +04: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 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x handle 0x%4.4x " , hdev - > name , ev - > status ,
2012-06-08 19:31:13 +04: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 ) {
hci_acl_disconn ( conn , HCI_ERROR_AUTH_FAILURE ) ;
hci_conn_put ( conn ) ;
goto unlock ;
}
if ( conn - > state = = BT_CONFIG ) {
if ( ! ev - > status )
conn - > state = BT_CONNECTED ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
hci_conn_put ( conn ) ;
} else {
hci_auth_cfm ( conn , ev - > status ) ;
hci_conn_hold ( conn ) ;
conn - > disc_timeout = HCI_DISCONN_TIMEOUT ;
hci_conn_put ( conn ) ;
}
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static u8 hci_get_auth_req ( struct hci_conn * conn )
2011-01-25 14:28:33 +03:00
{
/* If remote requests dedicated bonding follow that lead */
if ( conn - > remote_auth = = 0x02 | | conn - > remote_auth = = 0x03 ) {
/* If both remote and local IO capabilities allow MITM
* protection then require it , otherwise don ' t */
if ( conn - > remote_cap = = 0x03 | | conn - > io_capability = = 0x03 )
return 0x02 ;
else
return 0x03 ;
}
/* If remote requests no-bonding follow that lead */
if ( conn - > remote_auth = = 0x00 | | conn - > remote_auth = = 0x01 )
2011-04-28 14:07:58 +04:00
return conn - > remote_auth | ( conn - > auth_type & 0x01 ) ;
2011-01-25 14:28:33 +03:00
return conn - > auth_type ;
}
2012-05-23 11:04:18 +04:00
static void hci_io_capa_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2008-07-14 22:13:48 +04: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 16:40:05 +03:00
if ( ! conn )
goto unlock ;
hci_conn_hold ( conn ) ;
2012-01-09 01:11:15 +04:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-01-04 16:40:05 +03:00
goto unlock ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_PAIRABLE , & hdev - > dev_flags ) | |
2012-05-17 07:36:24 +04:00
( conn - > remote_auth & ~ 0x01 ) = = HCI_AT_NO_BONDING ) {
2011-01-25 14:28:33 +03:00
struct hci_cp_io_capability_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2012-01-16 12:04:29 +04:00
/* Change the IO capability from KeyboardDisplay
* to DisplayYesNo as it is not supported by BT spec . */
cp . capability = ( conn - > io_capability = = 0x04 ) ?
0x01 : conn - > io_capability ;
2011-04-28 22:29:04 +04:00
conn - > auth_type = hci_get_auth_req ( conn ) ;
cp . authentication = conn - > auth_type ;
2011-01-25 14:28:33 +03:00
2012-05-23 11:04:21 +04:00
if ( hci_find_remote_oob_data ( hdev , & conn - > dst ) & &
( conn - > out | | test_bit ( HCI_CONN_REMOTE_OOB , & conn - > flags ) ) )
2011-03-22 15:12:23 +03:00
cp . oob_data = 0x01 ;
else
cp . oob_data = 0x00 ;
2011-01-25 14:28:33 +03:00
hci_send_cmd ( hdev , HCI_OP_IO_CAPABILITY_REPLY ,
2012-05-17 07:36:24 +04:00
sizeof ( cp ) , & cp ) ;
2011-01-04 16:40:05 +03:00
} else {
struct hci_cp_io_capability_neg_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
2011-11-07 16:20:25 +04:00
cp . reason = HCI_ERROR_PAIRING_NOT_ALLOWED ;
2008-07-14 22:13:48 +04:00
2011-01-04 16:40:05 +03:00
hci_send_cmd ( hdev , HCI_OP_IO_CAPABILITY_NEG_REPLY ,
2012-05-17 07:36:24 +04:00
sizeof ( cp ) , & cp ) ;
2011-01-04 16:40:05 +03:00
}
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_io_capa_reply_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-01-04 16:40:05 +03: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 08:47:28 +04:00
if ( ev - > oob_data )
set_bit ( HCI_CONN_REMOTE_OOB , & conn - > flags ) ;
2011-01-04 16:40:05 +03:00
unlock :
2008-07-14 22:13:48 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_user_confirm_request_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-02-19 18:05:57 +03:00
{
struct hci_ev_user_confirm_req * ev = ( void * ) skb - > data ;
2011-04-28 22:28:56 +04:00
int loc_mitm , rem_mitm , confirm_hint = 0 ;
2011-04-28 22:28:53 +04:00
struct hci_conn * conn ;
2011-02-19 18:05:57 +03:00
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2012-01-09 01:11:15 +04:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-04-28 22:28:53 +04:00
goto unlock ;
2011-02-19 18:05:57 +03:00
2011-04-28 22:28:53 +04: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 . */
if ( ! conn - > connect_cfm_cb & & loc_mitm & & conn - > remote_cap = = 0x03 ) {
BT_DBG ( " Rejecting request: remote device can't provide MITM " ) ;
hci_send_cmd ( hdev , HCI_OP_USER_CONFIRM_NEG_REPLY ,
2012-05-17 07:36:24 +04:00
sizeof ( ev - > bdaddr ) , & ev - > bdaddr ) ;
2011-04-28 22:28:53 +04:00
goto unlock ;
}
/* If no side requires MITM protection; auto-accept */
if ( ( ! loc_mitm | | conn - > remote_cap = = 0x03 ) & &
2012-05-17 07:36:24 +04:00
( ! rem_mitm | | conn - > io_capability = = 0x03 ) ) {
2011-04-28 22:28:56 +04: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 08:10:31 +04:00
if ( ! test_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) ) {
2011-04-28 22:28:56 +04:00
BT_DBG ( " Confirming auto-accept as acceptor " ) ;
confirm_hint = 1 ;
goto confirm ;
}
2011-04-28 22:28:54 +04:00
BT_DBG ( " Auto-accept of user confirmation with %ums delay " ,
2012-05-17 07:36:24 +04:00
hdev - > auto_accept_delay ) ;
2011-04-28 22:28:54 +04:00
if ( hdev - > auto_accept_delay > 0 ) {
int delay = msecs_to_jiffies ( hdev - > auto_accept_delay ) ;
mod_timer ( & conn - > auto_accept_timer , jiffies + delay ) ;
goto unlock ;
}
2011-04-28 22:28:53 +04:00
hci_send_cmd ( hdev , HCI_OP_USER_CONFIRM_REPLY ,
2012-05-17 07:36:24 +04:00
sizeof ( ev - > bdaddr ) , & ev - > bdaddr ) ;
2011-04-28 22:28:53 +04:00
goto unlock ;
}
2011-04-28 22:28:56 +04:00
confirm :
2012-02-09 17:26:12 +04:00
mgmt_user_confirm_request ( hdev , & ev - > bdaddr , ACL_LINK , 0 , ev - > passkey ,
2012-03-08 08:25:00 +04:00
confirm_hint ) ;
2011-04-28 22:28:53 +04:00
unlock :
2011-02-19 18:05:57 +03:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_user_passkey_request_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-11-23 20:28:34 +04:00
{
struct hci_ev_user_passkey_req * ev = ( void * ) skb - > data ;
BT_DBG ( " %s " , hdev - > name ) ;
2012-01-09 01:11:15 +04:00
if ( test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2012-02-09 17:26:12 +04:00
mgmt_user_passkey_request ( hdev , & ev - > bdaddr , ACL_LINK , 0 ) ;
2011-11-23 20:28:34 +04:00
}
2012-09-06 19:39:26 +04: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 11:04:18 +04:00
static void hci_simple_pair_complete_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 22:13:48 +04: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 18: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 11:52:29 +04:00
if ( ! test_bit ( HCI_CONN_AUTH_PEND , & conn - > flags ) & & ev - > status )
2012-02-09 18:07:29 +04:00
mgmt_auth_failed ( hdev , & conn - > dst , conn - > type , conn - > dst_type ,
2012-03-08 08:25:00 +04:00
ev - > status ) ;
2008-07-14 22:13:48 +04:00
2011-02-19 18:06:00 +03:00
hci_conn_put ( conn ) ;
unlock :
2008-07-14 22:13:48 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_remote_host_features_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2008-07-14 22:13:48 +04:00
{
struct hci_ev_remote_host_features * ev = ( void * ) skb - > data ;
struct inquiry_entry * ie ;
BT_DBG ( " %s " , hdev - > name ) ;
hci_dev_lock ( hdev ) ;
2010-11-22 14:21:37 +03:00
ie = hci_inquiry_cache_lookup ( hdev , & ev - > bdaddr ) ;
if ( ie )
2012-02-28 04:28:43 +04:00
ie - > data . ssp_mode = ( ev - > features [ 0 ] & LMP_HOST_SSP ) ;
2008-07-14 22:13:48 +04:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_remote_oob_data_request_evt ( struct hci_dev * hdev ,
struct sk_buff * skb )
2011-03-22 15:12:22 +03: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-09 01:11:15 +04:00
if ( ! test_bit ( HCI_MGMT , & hdev - > dev_flags ) )
2011-04-06 15:01:59 +04:00
goto unlock ;
2011-03-22 15:12:22 +03:00
data = hci_find_remote_oob_data ( hdev , & ev - > bdaddr ) ;
if ( data ) {
struct hci_cp_remote_oob_data_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
memcpy ( cp . hash , data - > hash , sizeof ( cp . hash ) ) ;
memcpy ( cp . randomizer , data - > randomizer , sizeof ( cp . randomizer ) ) ;
hci_send_cmd ( hdev , HCI_OP_REMOTE_OOB_DATA_REPLY , sizeof ( cp ) ,
2012-05-17 07:36:24 +04:00
& cp ) ;
2011-03-22 15:12:22 +03:00
} else {
struct hci_cp_remote_oob_data_neg_reply cp ;
bacpy ( & cp . bdaddr , & ev - > bdaddr ) ;
hci_send_cmd ( hdev , HCI_OP_REMOTE_OOB_DATA_NEG_REPLY , sizeof ( cp ) ,
2012-05-17 07:36:24 +04:00
& cp ) ;
2011-03-22 15:12:22 +03:00
}
2011-04-06 15:01:59 +04:00
unlock :
2011-03-22 15:12:22 +03:00
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_le_conn_complete_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-02-11 04:38:47 +03:00
{
struct hci_ev_le_conn_complete * ev = ( void * ) skb - > data ;
struct hci_conn * conn ;
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s status 0x%2.2x " , hdev - > name , ev - > status ) ;
2011-02-11 04:38:47 +03:00
hci_dev_lock ( hdev ) ;
2012-07-27 22:10:15 +04:00
conn = hci_conn_hash_lookup_state ( hdev , LE_LINK , BT_CONNECT ) ;
2011-02-11 04: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 22:10:10 +04:00
goto unlock ;
2011-02-11 04:38:50 +03:00
}
2011-05-31 21:20:54 +04:00
conn - > dst_type = ev - > bdaddr_type ;
2012-07-27 22:10:11 +04:00
if ( ev - > role = = LE_CONN_ROLE_MASTER ) {
conn - > out = true ;
conn - > link_mode | = HCI_LM_MASTER ;
}
2011-02-11 04:38:50 +03:00
}
2011-02-11 04:38:47 +03:00
2012-07-27 22:10:16 +04:00
if ( ev - > status ) {
mgmt_connect_failed ( hdev , & conn - > dst , conn - > type ,
conn - > dst_type , ev - > status ) ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
conn - > state = BT_CLOSED ;
hci_conn_del ( conn ) ;
goto unlock ;
}
2012-01-17 23:48:47 +04:00
if ( ! test_and_set_bit ( HCI_CONN_MGMT_CONNECTED , & conn - > flags ) )
mgmt_device_connected ( hdev , & ev - > bdaddr , conn - > type ,
2012-03-08 08:25:00 +04:00
conn - > dst_type , 0 , NULL , 0 , NULL ) ;
2011-05-07 01:41:43 +04:00
2011-06-10 01:50:50 +04:00
conn - > sec_level = BT_SECURITY_LOW ;
2011-02-11 04:38:47 +03:00
conn - > handle = __le16_to_cpu ( ev - > handle ) ;
conn - > state = BT_CONNECTED ;
hci_conn_hold_device ( conn ) ;
hci_conn_add_sysfs ( conn ) ;
hci_proto_connect_cfm ( conn , ev - > status ) ;
unlock :
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_le_adv_report_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-05-26 23:23:51 +04:00
{
2011-09-27 03:48:35 +04:00
u8 num_reports = skb - > data [ 0 ] ;
void * ptr = & skb - > data [ 1 ] ;
2012-01-11 01:20:50 +04:00
s8 rssi ;
2011-05-26 23:23:51 +04:00
hci_dev_lock ( hdev ) ;
2011-09-27 03:48:35 +04:00
while ( num_reports - - ) {
struct hci_ev_le_advertising_info * ev = ptr ;
2011-05-26 23:23:51 +04:00
2012-01-11 01:20:50 +04:00
rssi = ev - > data [ ev - > length ] ;
mgmt_device_found ( hdev , & ev - > bdaddr , LE_LINK , ev - > bdaddr_type ,
2012-03-08 08:25:00 +04:00
NULL , rssi , 0 , 1 , ev - > data , ev - > length ) ;
2012-01-11 01:20:50 +04:00
2011-09-27 03:48:35 +04:00
ptr + = sizeof ( * ev ) + ev - > length + 1 ;
2011-05-26 23:23:51 +04:00
}
hci_dev_unlock ( hdev ) ;
}
2012-05-23 11:04:18 +04:00
static void hci_le_ltk_request_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-06-10 01:50:47 +04:00
{
struct hci_ev_le_ltk_req * ev = ( void * ) skb - > data ;
struct hci_cp_le_ltk_reply cp ;
2011-07-08 01:59:37 +04:00
struct hci_cp_le_ltk_neg_reply neg ;
2011-06-10 01:50:47 +04:00
struct hci_conn * conn ;
2012-02-03 04:08:01 +04:00
struct smp_ltk * ltk ;
2011-06-10 01:50:47 +04:00
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s handle 0x%4.4x " , hdev - > name , __le16_to_cpu ( ev - > handle ) ) ;
2011-06-10 01:50:47 +04:00
hci_dev_lock ( hdev ) ;
conn = hci_conn_hash_lookup_handle ( hdev , __le16_to_cpu ( ev - > handle ) ) ;
2011-07-08 01:59:37 +04:00
if ( conn = = NULL )
goto not_found ;
2011-06-10 01:50:47 +04:00
2011-07-08 01:59:37 +04:00
ltk = hci_find_ltk ( hdev , ev - > ediv , ev - > random ) ;
if ( ltk = = NULL )
goto not_found ;
memcpy ( cp . ltk , ltk - > val , sizeof ( ltk - > val ) ) ;
2011-06-10 01:50:47 +04:00
cp . handle = cpu_to_le16 ( conn - > handle ) ;
2012-02-03 04:08:01 +04:00
if ( ltk - > authenticated )
conn - > sec_level = BT_SECURITY_HIGH ;
2011-06-10 01:50:47 +04:00
hci_send_cmd ( hdev , HCI_OP_LE_LTK_REPLY , sizeof ( cp ) , & cp ) ;
2012-02-03 04:08:01 +04:00
if ( ltk - > type & HCI_SMP_STK ) {
list_del ( & ltk - > list ) ;
kfree ( ltk ) ;
}
2011-06-10 01:50:47 +04:00
hci_dev_unlock ( hdev ) ;
2011-07-08 01:59:37 +04: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-10 01:50:47 +04:00
}
2012-05-23 11:04:18 +04:00
static void hci_le_meta_evt ( struct hci_dev * hdev , struct sk_buff * skb )
2011-02-11 04: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 23:23:51 +04:00
case HCI_EV_LE_ADVERTISING_REPORT :
hci_le_adv_report_evt ( hdev , skb ) ;
break ;
2011-06-10 01:50:47 +04:00
case HCI_EV_LE_LTK_REQ :
hci_le_ltk_request_evt ( hdev , skb ) ;
break ;
2011-02-11 04:38:47 +03:00
default :
break ;
}
}
2012-09-27 18:26:22 +04: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 15:33:56 +04: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 ;
skb_pull ( skb , HCI_EVENT_HDR_SIZE ) ;
switch ( event ) {
2005-04-17 02:20:36 +04: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 15:33:56 +04:00
case HCI_EV_CONN_COMPLETE :
hci_conn_complete_evt ( hdev , skb ) ;
2005-09-13 03:32:25 +04:00
break ;
2005-04-17 02:20:36 +04: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 15:33:56 +04:00
case HCI_EV_REMOTE_NAME :
hci_remote_name_evt ( hdev , skb ) ;
break ;
2005-04-17 02:20:36 +04:00
case HCI_EV_ENCRYPT_CHANGE :
hci_encrypt_change_evt ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04: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_REMOTE_VERSION :
hci_remote_version_evt ( hdev , skb ) ;
break ;
case HCI_EV_QOS_SETUP_COMPLETE :
hci_qos_setup_complete_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-17 02:20:36 +04: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 22:13:46 +04:00
case HCI_EV_PKT_TYPE_CHANGE :
hci_pkt_type_change_evt ( hdev , skb ) ;
break ;
2005-08-10 07:28:02 +04:00
case HCI_EV_PSCAN_REP_MODE :
hci_pscan_rep_mode_evt ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04:00
case HCI_EV_INQUIRY_RESULT_WITH_RSSI :
hci_inquiry_result_with_rssi_evt ( hdev , skb ) ;
2006-07-03 12:02:33 +04:00
break ;
2007-10-20 15:33:56 +04:00
case HCI_EV_REMOTE_EXT_FEATURES :
hci_remote_ext_features_evt ( hdev , skb ) ;
2005-04-17 02:20:36 +04:00
break ;
2007-10-20 15:33:56 +04:00
case HCI_EV_SYNC_CONN_COMPLETE :
hci_sync_conn_complete_evt ( hdev , skb ) ;
break ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
case HCI_EV_SYNC_CONN_CHANGED :
hci_sync_conn_changed_evt ( hdev , skb ) ;
break ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
case HCI_EV_SNIFF_SUBRATE :
hci_sniff_subrate_evt ( hdev , skb ) ;
break ;
2005-04-17 02:20:36 +04:00
2007-10-20 15:33:56 +04:00
case HCI_EV_EXTENDED_INQUIRY_RESULT :
hci_extended_inquiry_result_evt ( hdev , skb ) ;
break ;
2005-04-17 02:20:36 +04:00
2012-06-08 19:31:13 +04:00
case HCI_EV_KEY_REFRESH_COMPLETE :
hci_key_refresh_complete_evt ( hdev , skb ) ;
break ;
2008-07-14 22:13:48 +04:00
case HCI_EV_IO_CAPA_REQUEST :
hci_io_capa_request_evt ( hdev , skb ) ;
break ;
2011-01-04 16:40:05 +03:00
case HCI_EV_IO_CAPA_REPLY :
hci_io_capa_reply_evt ( hdev , skb ) ;
break ;
2011-02-19 18:05:57 +03:00
case HCI_EV_USER_CONFIRM_REQUEST :
hci_user_confirm_request_evt ( hdev , skb ) ;
break ;
2011-11-23 20:28:34 +04:00
case HCI_EV_USER_PASSKEY_REQUEST :
hci_user_passkey_request_evt ( hdev , skb ) ;
break ;
2012-09-06 19:39:26 +04: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 22:13:48 +04:00
case HCI_EV_SIMPLE_PAIR_COMPLETE :
hci_simple_pair_complete_evt ( hdev , skb ) ;
break ;
2008-07-14 22:13:48 +04:00
case HCI_EV_REMOTE_HOST_FEATURES :
hci_remote_host_features_evt ( hdev , skb ) ;
break ;
2011-02-11 04:38:47 +03:00
case HCI_EV_LE_META :
hci_le_meta_evt ( hdev , skb ) ;
break ;
2012-09-27 18:26:22 +04:00
case HCI_EV_CHANNEL_SELECTED :
hci_chan_selected_evt ( hdev , skb ) ;
break ;
2011-03-22 15:12:22 +03:00
case HCI_EV_REMOTE_OOB_DATA_REQUEST :
hci_remote_oob_data_request_evt ( hdev , skb ) ;
break ;
2012-01-04 14:41:58 +04:00
case HCI_EV_NUM_COMP_BLOCKS :
hci_num_comp_blocks_evt ( hdev , skb ) ;
break ;
2007-10-20 15:33:56 +04:00
default :
2012-07-11 15:32:43 +04:00
BT_DBG ( " %s event 0x%2.2x " , hdev - > name , event ) ;
2005-04-17 02:20:36 +04:00
break ;
}
kfree_skb ( skb ) ;
hdev - > stat . evt_rx + + ;
}