diff options
Diffstat (limited to 'net/bluetooth/mgmt.c')
| -rw-r--r-- | net/bluetooth/mgmt.c | 402 | 
1 files changed, 391 insertions, 11 deletions
| diff --git a/net/bluetooth/mgmt.c b/net/bluetooth/mgmt.c index 8a80d48d89c4..3bdc8f3ca259 100644 --- a/net/bluetooth/mgmt.c +++ b/net/bluetooth/mgmt.c @@ -617,6 +617,127 @@ static int read_config_info(struct sock *sk, struct hci_dev *hdev,  				 &rp, sizeof(rp));  } +static u32 get_supported_phys(struct hci_dev *hdev) +{ +	u32 supported_phys = 0; + +	if (lmp_bredr_capable(hdev)) { +		supported_phys |= MGMT_PHY_BR_1M_1SLOT; + +		if (hdev->features[0][0] & LMP_3SLOT) +			supported_phys |= MGMT_PHY_BR_1M_3SLOT; + +		if (hdev->features[0][0] & LMP_5SLOT) +			supported_phys |= MGMT_PHY_BR_1M_5SLOT; + +		if (lmp_edr_2m_capable(hdev)) { +			supported_phys |= MGMT_PHY_EDR_2M_1SLOT; + +			if (lmp_edr_3slot_capable(hdev)) +				supported_phys |= MGMT_PHY_EDR_2M_3SLOT; + +			if (lmp_edr_5slot_capable(hdev)) +				supported_phys |= MGMT_PHY_EDR_2M_5SLOT; + +			if (lmp_edr_3m_capable(hdev)) { +				supported_phys |= MGMT_PHY_EDR_3M_1SLOT; + +				if (lmp_edr_3slot_capable(hdev)) +					supported_phys |= MGMT_PHY_EDR_3M_3SLOT; + +				if (lmp_edr_5slot_capable(hdev)) +					supported_phys |= MGMT_PHY_EDR_3M_5SLOT; +			} +		} +	} + +	if (lmp_le_capable(hdev)) { +		supported_phys |= MGMT_PHY_LE_1M_TX; +		supported_phys |= MGMT_PHY_LE_1M_RX; + +		if (hdev->le_features[1] & HCI_LE_PHY_2M) { +			supported_phys |= MGMT_PHY_LE_2M_TX; +			supported_phys |= MGMT_PHY_LE_2M_RX; +		} + +		if (hdev->le_features[1] & HCI_LE_PHY_CODED) { +			supported_phys |= MGMT_PHY_LE_CODED_TX; +			supported_phys |= MGMT_PHY_LE_CODED_RX; +		} +	} + +	return supported_phys; +} + +static u32 get_selected_phys(struct hci_dev *hdev) +{ +	u32 selected_phys = 0; + +	if (lmp_bredr_capable(hdev)) { +		selected_phys |= MGMT_PHY_BR_1M_1SLOT; + +		if (hdev->pkt_type & (HCI_DM3 | HCI_DH3)) +			selected_phys |= MGMT_PHY_BR_1M_3SLOT; + +		if (hdev->pkt_type & (HCI_DM5 | HCI_DH5)) +			selected_phys |= MGMT_PHY_BR_1M_5SLOT; + +		if (lmp_edr_2m_capable(hdev)) { +			if (!(hdev->pkt_type & HCI_2DH1)) +				selected_phys |= MGMT_PHY_EDR_2M_1SLOT; + +			if (lmp_edr_3slot_capable(hdev) && +			    !(hdev->pkt_type & HCI_2DH3)) +				selected_phys |= MGMT_PHY_EDR_2M_3SLOT; + +			if (lmp_edr_5slot_capable(hdev) && +			    !(hdev->pkt_type & HCI_2DH5)) +				selected_phys |= MGMT_PHY_EDR_2M_5SLOT; + +			if (lmp_edr_3m_capable(hdev)) { +				if (!(hdev->pkt_type & HCI_3DH1)) +					selected_phys |= MGMT_PHY_EDR_3M_1SLOT; + +				if (lmp_edr_3slot_capable(hdev) && +				    !(hdev->pkt_type & HCI_3DH3)) +					selected_phys |= MGMT_PHY_EDR_3M_3SLOT; + +				if (lmp_edr_5slot_capable(hdev) && +				    !(hdev->pkt_type & HCI_3DH5)) +					selected_phys |= MGMT_PHY_EDR_3M_5SLOT; +			} +		} +	} + +	if (lmp_le_capable(hdev)) { +		if (hdev->le_tx_def_phys & HCI_LE_SET_PHY_1M) +			selected_phys |= MGMT_PHY_LE_1M_TX; + +		if (hdev->le_rx_def_phys & HCI_LE_SET_PHY_1M) +			selected_phys |= MGMT_PHY_LE_1M_RX; + +		if (hdev->le_tx_def_phys & HCI_LE_SET_PHY_2M) +			selected_phys |= MGMT_PHY_LE_2M_TX; + +		if (hdev->le_rx_def_phys & HCI_LE_SET_PHY_2M) +			selected_phys |= MGMT_PHY_LE_2M_RX; + +		if (hdev->le_tx_def_phys & HCI_LE_SET_PHY_CODED) +			selected_phys |= MGMT_PHY_LE_CODED_TX; + +		if (hdev->le_rx_def_phys & HCI_LE_SET_PHY_CODED) +			selected_phys |= MGMT_PHY_LE_CODED_RX; +	} + +	return selected_phys; +} + +static u32 get_configurable_phys(struct hci_dev *hdev) +{ +	return (get_supported_phys(hdev) & ~MGMT_PHY_BR_1M_1SLOT & +		~MGMT_PHY_LE_1M_TX & ~MGMT_PHY_LE_1M_RX); +} +  static u32 get_supported_settings(struct hci_dev *hdev)  {  	u32 settings = 0; @@ -654,6 +775,8 @@ static u32 get_supported_settings(struct hci_dev *hdev)  	    hdev->set_bdaddr)  		settings |= MGMT_SETTING_CONFIGURATION; +	settings |= MGMT_SETTING_PHY_CONFIGURATION; +  	return settings;  } @@ -817,7 +940,10 @@ static void rpa_expired(struct work_struct *work)  	 * function.  	 */  	hci_req_init(&req, hdev); -	__hci_req_enable_advertising(&req); +	if (ext_adv_capable(hdev)) +		__hci_req_start_ext_adv(&req, hdev->cur_adv_instance); +	else +		__hci_req_enable_advertising(&req);  	hci_req_run(&req, NULL);  } @@ -1721,10 +1847,17 @@ static void le_enable_complete(struct hci_dev *hdev, u8 status, u16 opcode)  	 */  	if (hci_dev_test_flag(hdev, HCI_LE_ENABLED)) {  		struct hci_request req; -  		hci_req_init(&req, hdev); -		__hci_req_update_adv_data(&req, 0x00); -		__hci_req_update_scan_rsp_data(&req, 0x00); +		if (ext_adv_capable(hdev)) { +			int err; + +			err = __hci_req_setup_ext_adv_instance(&req, 0x00); +			if (!err) +				__hci_req_update_scan_rsp_data(&req, 0x00); +		} else { +			__hci_req_update_adv_data(&req, 0x00); +			__hci_req_update_scan_rsp_data(&req, 0x00); +		}  		hci_req_run(&req, NULL);  		hci_update_background_scan(hdev);  	} @@ -1823,6 +1956,9 @@ static int set_le(struct sock *sk, struct hci_dev *hdev, void *data, u16 len)  	} else {  		if (hci_dev_test_flag(hdev, HCI_LE_ADV))  			__hci_req_disable_advertising(&req); + +		if (ext_adv_capable(hdev)) +			__hci_req_clear_ext_adv_sets(&req);  	}  	hci_req_add(&req, HCI_OP_WRITE_LE_HOST_SUPPORTED, sizeof(hci_cp), @@ -3184,6 +3320,225 @@ static int set_appearance(struct sock *sk, struct hci_dev *hdev, void *data,  	return err;  } +static int get_phy_configuration(struct sock *sk, struct hci_dev *hdev, +				 void *data, u16 len) +{ +	struct mgmt_rp_get_phy_confguration rp; + +	BT_DBG("sock %p %s", sk, hdev->name); + +	hci_dev_lock(hdev); + +	memset(&rp, 0, sizeof(rp)); + +	rp.supported_phys = cpu_to_le32(get_supported_phys(hdev)); +	rp.selected_phys = cpu_to_le32(get_selected_phys(hdev)); +	rp.configurable_phys = cpu_to_le32(get_configurable_phys(hdev)); + +	hci_dev_unlock(hdev); + +	return mgmt_cmd_complete(sk, hdev->id, MGMT_OP_GET_PHY_CONFIGURATION, 0, +				 &rp, sizeof(rp)); +} + +int mgmt_phy_configuration_changed(struct hci_dev *hdev, struct sock *skip) +{ +	struct mgmt_ev_phy_configuration_changed ev; + +	memset(&ev, 0, sizeof(ev)); + +	ev.selected_phys = cpu_to_le32(get_selected_phys(hdev)); + +	return mgmt_event(MGMT_EV_PHY_CONFIGURATION_CHANGED, hdev, &ev, +			  sizeof(ev), skip); +} + +static void set_default_phy_complete(struct hci_dev *hdev, u8 status, +				     u16 opcode, struct sk_buff *skb) +{ +	struct mgmt_pending_cmd *cmd; + +	BT_DBG("status 0x%02x", status); + +	hci_dev_lock(hdev); + +	cmd = pending_find(MGMT_OP_SET_PHY_CONFIGURATION, hdev); +	if (!cmd) +		goto unlock; + +	if (status) { +		mgmt_cmd_status(cmd->sk, hdev->id, +				MGMT_OP_SET_PHY_CONFIGURATION, +				mgmt_status(status)); +	} else { +		mgmt_cmd_complete(cmd->sk, hdev->id, +				  MGMT_OP_SET_PHY_CONFIGURATION, 0, +				  NULL, 0); + +		mgmt_phy_configuration_changed(hdev, cmd->sk); +	} + +	mgmt_pending_remove(cmd); + +unlock: +	hci_dev_unlock(hdev); +} + +static int set_phy_configuration(struct sock *sk, struct hci_dev *hdev, +				 void *data, u16 len) +{ +	struct mgmt_cp_set_phy_confguration *cp = data; +	struct hci_cp_le_set_default_phy cp_phy; +	struct mgmt_pending_cmd *cmd; +	struct hci_request req; +	u32 selected_phys, configurable_phys, supported_phys, unconfigure_phys; +	u16 pkt_type = (HCI_DH1 | HCI_DM1); +	bool changed = false; +	int err; + +	BT_DBG("sock %p %s", sk, hdev->name); + +	configurable_phys = get_configurable_phys(hdev); +	supported_phys = get_supported_phys(hdev); +	selected_phys = __le32_to_cpu(cp->selected_phys); + +	if (selected_phys & ~supported_phys) +		return mgmt_cmd_status(sk, hdev->id, +				       MGMT_OP_SET_PHY_CONFIGURATION, +				       MGMT_STATUS_INVALID_PARAMS); + +	unconfigure_phys = supported_phys & ~configurable_phys; + +	if ((selected_phys & unconfigure_phys) != unconfigure_phys) +		return mgmt_cmd_status(sk, hdev->id, +				       MGMT_OP_SET_PHY_CONFIGURATION, +				       MGMT_STATUS_INVALID_PARAMS); + +	if (selected_phys == get_selected_phys(hdev)) +		return mgmt_cmd_complete(sk, hdev->id, +					 MGMT_OP_SET_PHY_CONFIGURATION, +					 0, NULL, 0); + +	hci_dev_lock(hdev); + +	if (!hdev_is_powered(hdev)) { +		err = mgmt_cmd_status(sk, hdev->id, +				      MGMT_OP_SET_PHY_CONFIGURATION, +				      MGMT_STATUS_REJECTED); +		goto unlock; +	} + +	if (pending_find(MGMT_OP_SET_PHY_CONFIGURATION, hdev)) { +		err = mgmt_cmd_status(sk, hdev->id, +				      MGMT_OP_SET_PHY_CONFIGURATION, +				      MGMT_STATUS_BUSY); +		goto unlock; +	} + +	if (selected_phys & MGMT_PHY_BR_1M_3SLOT) +		pkt_type |= (HCI_DH3 | HCI_DM3); +	else +		pkt_type &= ~(HCI_DH3 | HCI_DM3); + +	if (selected_phys & MGMT_PHY_BR_1M_5SLOT) +		pkt_type |= (HCI_DH5 | HCI_DM5); +	else +		pkt_type &= ~(HCI_DH5 | HCI_DM5); + +	if (selected_phys & MGMT_PHY_EDR_2M_1SLOT) +		pkt_type &= ~HCI_2DH1; +	else +		pkt_type |= HCI_2DH1; + +	if (selected_phys & MGMT_PHY_EDR_2M_3SLOT) +		pkt_type &= ~HCI_2DH3; +	else +		pkt_type |= HCI_2DH3; + +	if (selected_phys & MGMT_PHY_EDR_2M_5SLOT) +		pkt_type &= ~HCI_2DH5; +	else +		pkt_type |= HCI_2DH5; + +	if (selected_phys & MGMT_PHY_EDR_3M_1SLOT) +		pkt_type &= ~HCI_3DH1; +	else +		pkt_type |= HCI_3DH1; + +	if (selected_phys & MGMT_PHY_EDR_3M_3SLOT) +		pkt_type &= ~HCI_3DH3; +	else +		pkt_type |= HCI_3DH3; + +	if (selected_phys & MGMT_PHY_EDR_3M_5SLOT) +		pkt_type &= ~HCI_3DH5; +	else +		pkt_type |= HCI_3DH5; + +	if (pkt_type != hdev->pkt_type) { +		hdev->pkt_type = pkt_type; +		changed = true; +	} + +	if ((selected_phys & MGMT_PHY_LE_MASK) == +	    (get_selected_phys(hdev) & MGMT_PHY_LE_MASK)) { +		if (changed) +			mgmt_phy_configuration_changed(hdev, sk); + +		err = mgmt_cmd_complete(sk, hdev->id, +					MGMT_OP_SET_PHY_CONFIGURATION, +					0, NULL, 0); + +		goto unlock; +	} + +	cmd = mgmt_pending_add(sk, MGMT_OP_SET_PHY_CONFIGURATION, hdev, data, +			       len); +	if (!cmd) { +		err = -ENOMEM; +		goto unlock; +	} + +	hci_req_init(&req, hdev); + +	memset(&cp_phy, 0, sizeof(cp_phy)); + +	if (!(selected_phys & MGMT_PHY_LE_TX_MASK)) +		cp_phy.all_phys |= 0x01; + +	if (!(selected_phys & MGMT_PHY_LE_RX_MASK)) +		cp_phy.all_phys |= 0x02; + +	if (selected_phys & MGMT_PHY_LE_1M_TX) +		cp_phy.tx_phys |= HCI_LE_SET_PHY_1M; + +	if (selected_phys & MGMT_PHY_LE_2M_TX) +		cp_phy.tx_phys |= HCI_LE_SET_PHY_2M; + +	if (selected_phys & MGMT_PHY_LE_CODED_TX) +		cp_phy.tx_phys |= HCI_LE_SET_PHY_CODED; + +	if (selected_phys & MGMT_PHY_LE_1M_RX) +		cp_phy.rx_phys |= HCI_LE_SET_PHY_1M; + +	if (selected_phys & MGMT_PHY_LE_2M_RX) +		cp_phy.rx_phys |= HCI_LE_SET_PHY_2M; + +	if (selected_phys & MGMT_PHY_LE_CODED_RX) +		cp_phy.rx_phys |= HCI_LE_SET_PHY_CODED; + +	hci_req_add(&req, HCI_OP_LE_SET_DEFAULT_PHY, sizeof(cp_phy), &cp_phy); + +	err = hci_req_run_skb(&req, set_default_phy_complete); +	if (err < 0) +		mgmt_pending_remove(cmd); + +unlock: +	hci_dev_unlock(hdev); + +	return err; +} +  static void read_local_oob_data_complete(struct hci_dev *hdev, u8 status,  				         u16 opcode, struct sk_buff *skb)  { @@ -4037,9 +4392,14 @@ static int set_advertising(struct sock *sk, struct hci_dev *hdev, void *data,  		 * HCI_ADVERTISING flag is not yet set.  		 */  		hdev->cur_adv_instance = 0x00; -		__hci_req_update_adv_data(&req, 0x00); -		__hci_req_update_scan_rsp_data(&req, 0x00); -		__hci_req_enable_advertising(&req); + +		if (ext_adv_capable(hdev)) { +			__hci_req_start_ext_adv(&req, 0x00); +		} else { +			__hci_req_update_adv_data(&req, 0x00); +			__hci_req_update_scan_rsp_data(&req, 0x00); +			__hci_req_enable_advertising(&req); +		}  	} else {  		__hci_req_disable_advertising(&req);  	} @@ -4609,6 +4969,7 @@ static int set_privacy(struct sock *sk, struct hci_dev *hdev, void *cp_data,  		changed = !hci_dev_test_and_set_flag(hdev, HCI_PRIVACY);  		memcpy(hdev->irk, cp->irk, sizeof(hdev->irk));  		hci_dev_set_flag(hdev, HCI_RPA_EXPIRED); +		hci_adv_instances_set_rpa_expired(hdev, true);  		if (cp->privacy == 0x02)  			hci_dev_set_flag(hdev, HCI_LIMITED_PRIVACY);  		else @@ -4617,6 +4978,7 @@ static int set_privacy(struct sock *sk, struct hci_dev *hdev, void *cp_data,  		changed = hci_dev_test_and_clear_flag(hdev, HCI_PRIVACY);  		memset(hdev->irk, 0, sizeof(hdev->irk));  		hci_dev_clear_flag(hdev, HCI_RPA_EXPIRED); +		hci_adv_instances_set_rpa_expired(hdev, false);  		hci_dev_clear_flag(hdev, HCI_LIMITED_PRIVACY);  	} @@ -5967,9 +6329,23 @@ static u32 get_supported_adv_flags(struct hci_dev *hdev)  	flags |= MGMT_ADV_FLAG_APPEARANCE;  	flags |= MGMT_ADV_FLAG_LOCAL_NAME; -	if (hdev->adv_tx_power != HCI_TX_POWER_INVALID) +	/* In extended adv TX_POWER returned from Set Adv Param +	 * will be always valid. +	 */ +	if ((hdev->adv_tx_power != HCI_TX_POWER_INVALID) || +	    ext_adv_capable(hdev))  		flags |= MGMT_ADV_FLAG_TX_POWER; +	if (ext_adv_capable(hdev)) { +		flags |= MGMT_ADV_FLAG_SEC_1M; + +		if (hdev->le_features[1] & HCI_LE_PHY_2M) +			flags |= MGMT_ADV_FLAG_SEC_2M; + +		if (hdev->le_features[1] & HCI_LE_PHY_CODED) +			flags |= MGMT_ADV_FLAG_SEC_CODED; +	} +  	return flags;  } @@ -6175,7 +6551,7 @@ static int add_advertising(struct sock *sk, struct hci_dev *hdev,  	struct mgmt_cp_add_advertising *cp = data;  	struct mgmt_rp_add_advertising rp;  	u32 flags; -	u32 supported_flags; +	u32 supported_flags, phy_flags;  	u8 status;  	u16 timeout, duration;  	unsigned int prev_instance_cnt = hdev->adv_instance_cnt; @@ -6205,10 +6581,12 @@ static int add_advertising(struct sock *sk, struct hci_dev *hdev,  	duration = __le16_to_cpu(cp->duration);  	/* The current implementation only supports a subset of the specified -	 * flags. +	 * flags. Also need to check mutual exclusiveness of sec flags.  	 */  	supported_flags = get_supported_adv_flags(hdev); -	if (flags & ~supported_flags) +	phy_flags = flags & MGMT_ADV_FLAG_SEC_MASK; +	if (flags & ~supported_flags || +	    ((phy_flags && (phy_flags ^ (phy_flags & -phy_flags)))))  		return mgmt_cmd_status(sk, hdev->id, MGMT_OP_ADD_ADVERTISING,  				       MGMT_STATUS_INVALID_PARAMS); @@ -6544,6 +6922,8 @@ static const struct hci_mgmt_handler mgmt_handlers[] = {  	{ read_ext_controller_info,MGMT_READ_EXT_INFO_SIZE,  						HCI_MGMT_UNTRUSTED },  	{ set_appearance,	   MGMT_SET_APPEARANCE_SIZE }, +	{ get_phy_configuration,   MGMT_GET_PHY_CONFIGURATION_SIZE }, +	{ set_phy_configuration,   MGMT_SET_PHY_CONFIGURATION_SIZE },  };  void mgmt_index_added(struct hci_dev *hdev) | 
