18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci   BlueZ - Bluetooth protocol stack for Linux
38c2ecf20Sopenharmony_ci   Copyright (C) 2014 Intel Corporation
48c2ecf20Sopenharmony_ci
58c2ecf20Sopenharmony_ci   This program is free software; you can redistribute it and/or modify
68c2ecf20Sopenharmony_ci   it under the terms of the GNU General Public License version 2 as
78c2ecf20Sopenharmony_ci   published by the Free Software Foundation;
88c2ecf20Sopenharmony_ci
98c2ecf20Sopenharmony_ci   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
108c2ecf20Sopenharmony_ci   OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
118c2ecf20Sopenharmony_ci   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.
128c2ecf20Sopenharmony_ci   IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY
138c2ecf20Sopenharmony_ci   CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES
148c2ecf20Sopenharmony_ci   WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
158c2ecf20Sopenharmony_ci   ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
168c2ecf20Sopenharmony_ci   OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
178c2ecf20Sopenharmony_ci
188c2ecf20Sopenharmony_ci   ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,
198c2ecf20Sopenharmony_ci   COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS
208c2ecf20Sopenharmony_ci   SOFTWARE IS DISCLAIMED.
218c2ecf20Sopenharmony_ci*/
228c2ecf20Sopenharmony_ci
238c2ecf20Sopenharmony_ci#include <asm/unaligned.h>
248c2ecf20Sopenharmony_ci
258c2ecf20Sopenharmony_ci#define hci_req_sync_lock(hdev)   mutex_lock(&hdev->req_lock)
268c2ecf20Sopenharmony_ci#define hci_req_sync_unlock(hdev) mutex_unlock(&hdev->req_lock)
278c2ecf20Sopenharmony_ci
288c2ecf20Sopenharmony_cistruct hci_request {
298c2ecf20Sopenharmony_ci	struct hci_dev		*hdev;
308c2ecf20Sopenharmony_ci	struct sk_buff_head	cmd_q;
318c2ecf20Sopenharmony_ci
328c2ecf20Sopenharmony_ci	/* If something goes wrong when building the HCI request, the error
338c2ecf20Sopenharmony_ci	 * value is stored in this field.
348c2ecf20Sopenharmony_ci	 */
358c2ecf20Sopenharmony_ci	int			err;
368c2ecf20Sopenharmony_ci};
378c2ecf20Sopenharmony_ci
388c2ecf20Sopenharmony_civoid hci_req_init(struct hci_request *req, struct hci_dev *hdev);
398c2ecf20Sopenharmony_civoid hci_req_purge(struct hci_request *req);
408c2ecf20Sopenharmony_cibool hci_req_status_pend(struct hci_dev *hdev);
418c2ecf20Sopenharmony_ciint hci_req_run(struct hci_request *req, hci_req_complete_t complete);
428c2ecf20Sopenharmony_ciint hci_req_run_skb(struct hci_request *req, hci_req_complete_skb_t complete);
438c2ecf20Sopenharmony_civoid hci_req_add(struct hci_request *req, u16 opcode, u32 plen,
448c2ecf20Sopenharmony_ci		 const void *param);
458c2ecf20Sopenharmony_civoid hci_req_add_ev(struct hci_request *req, u16 opcode, u32 plen,
468c2ecf20Sopenharmony_ci		    const void *param, u8 event);
478c2ecf20Sopenharmony_civoid hci_req_cmd_complete(struct hci_dev *hdev, u16 opcode, u8 status,
488c2ecf20Sopenharmony_ci			  hci_req_complete_t *req_complete,
498c2ecf20Sopenharmony_ci			  hci_req_complete_skb_t *req_complete_skb);
508c2ecf20Sopenharmony_ci
518c2ecf20Sopenharmony_ciint hci_req_sync(struct hci_dev *hdev, int (*req)(struct hci_request *req,
528c2ecf20Sopenharmony_ci						  unsigned long opt),
538c2ecf20Sopenharmony_ci		 unsigned long opt, u32 timeout, u8 *hci_status);
548c2ecf20Sopenharmony_ciint __hci_req_sync(struct hci_dev *hdev, int (*func)(struct hci_request *req,
558c2ecf20Sopenharmony_ci						     unsigned long opt),
568c2ecf20Sopenharmony_ci		   unsigned long opt, u32 timeout, u8 *hci_status);
578c2ecf20Sopenharmony_civoid hci_req_sync_cancel(struct hci_dev *hdev, int err);
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistruct sk_buff *hci_prepare_cmd(struct hci_dev *hdev, u16 opcode, u32 plen,
608c2ecf20Sopenharmony_ci				const void *param);
618c2ecf20Sopenharmony_ci
628c2ecf20Sopenharmony_ciint __hci_req_hci_power_on(struct hci_dev *hdev);
638c2ecf20Sopenharmony_ci
648c2ecf20Sopenharmony_civoid __hci_req_write_fast_connectable(struct hci_request *req, bool enable);
658c2ecf20Sopenharmony_civoid __hci_req_update_name(struct hci_request *req);
668c2ecf20Sopenharmony_civoid __hci_req_update_eir(struct hci_request *req);
678c2ecf20Sopenharmony_ci
688c2ecf20Sopenharmony_civoid hci_req_add_le_scan_disable(struct hci_request *req, bool rpa_le_conn);
698c2ecf20Sopenharmony_civoid hci_req_add_le_passive_scan(struct hci_request *req);
708c2ecf20Sopenharmony_ci
718c2ecf20Sopenharmony_civoid hci_req_prepare_suspend(struct hci_dev *hdev, enum suspended_state next);
728c2ecf20Sopenharmony_ci
738c2ecf20Sopenharmony_civoid hci_req_disable_address_resolution(struct hci_dev *hdev);
748c2ecf20Sopenharmony_civoid hci_req_reenable_advertising(struct hci_dev *hdev);
758c2ecf20Sopenharmony_civoid __hci_req_enable_advertising(struct hci_request *req);
768c2ecf20Sopenharmony_civoid __hci_req_disable_advertising(struct hci_request *req);
778c2ecf20Sopenharmony_civoid __hci_req_update_adv_data(struct hci_request *req, u8 instance);
788c2ecf20Sopenharmony_ciint hci_req_update_adv_data(struct hci_dev *hdev, u8 instance);
798c2ecf20Sopenharmony_civoid __hci_req_update_scan_rsp_data(struct hci_request *req, u8 instance);
808c2ecf20Sopenharmony_ci
818c2ecf20Sopenharmony_ciint __hci_req_schedule_adv_instance(struct hci_request *req, u8 instance,
828c2ecf20Sopenharmony_ci				    bool force);
838c2ecf20Sopenharmony_civoid hci_req_clear_adv_instance(struct hci_dev *hdev, struct sock *sk,
848c2ecf20Sopenharmony_ci				struct hci_request *req, u8 instance,
858c2ecf20Sopenharmony_ci				bool force);
868c2ecf20Sopenharmony_ci
878c2ecf20Sopenharmony_ciint __hci_req_setup_ext_adv_instance(struct hci_request *req, u8 instance);
888c2ecf20Sopenharmony_ciint __hci_req_start_ext_adv(struct hci_request *req, u8 instance);
898c2ecf20Sopenharmony_ciint __hci_req_enable_ext_advertising(struct hci_request *req, u8 instance);
908c2ecf20Sopenharmony_ciint __hci_req_disable_ext_adv_instance(struct hci_request *req, u8 instance);
918c2ecf20Sopenharmony_ciint __hci_req_remove_ext_adv_instance(struct hci_request *req, u8 instance);
928c2ecf20Sopenharmony_civoid __hci_req_clear_ext_adv_sets(struct hci_request *req);
938c2ecf20Sopenharmony_ciint hci_get_random_address(struct hci_dev *hdev, bool require_privacy,
948c2ecf20Sopenharmony_ci			   bool use_rpa, struct adv_info *adv_instance,
958c2ecf20Sopenharmony_ci			   u8 *own_addr_type, bdaddr_t *rand_addr);
968c2ecf20Sopenharmony_ci
978c2ecf20Sopenharmony_civoid __hci_req_update_class(struct hci_request *req);
988c2ecf20Sopenharmony_ci
998c2ecf20Sopenharmony_ci/* Returns true if HCI commands were queued */
1008c2ecf20Sopenharmony_cibool hci_req_stop_discovery(struct hci_request *req);
1018c2ecf20Sopenharmony_ci
1028c2ecf20Sopenharmony_cistatic inline void hci_req_update_scan(struct hci_dev *hdev)
1038c2ecf20Sopenharmony_ci{
1048c2ecf20Sopenharmony_ci	queue_work(hdev->req_workqueue, &hdev->scan_update);
1058c2ecf20Sopenharmony_ci}
1068c2ecf20Sopenharmony_ci
1078c2ecf20Sopenharmony_civoid __hci_req_update_scan(struct hci_request *req);
1088c2ecf20Sopenharmony_ci
1098c2ecf20Sopenharmony_ciint hci_update_random_address(struct hci_request *req, bool require_privacy,
1108c2ecf20Sopenharmony_ci			      bool use_rpa, u8 *own_addr_type);
1118c2ecf20Sopenharmony_ci
1128c2ecf20Sopenharmony_ciint hci_abort_conn(struct hci_conn *conn, u8 reason);
1138c2ecf20Sopenharmony_civoid __hci_abort_conn(struct hci_request *req, struct hci_conn *conn,
1148c2ecf20Sopenharmony_ci		      u8 reason);
1158c2ecf20Sopenharmony_ci
1168c2ecf20Sopenharmony_cistatic inline void hci_update_background_scan(struct hci_dev *hdev)
1178c2ecf20Sopenharmony_ci{
1188c2ecf20Sopenharmony_ci	queue_work(hdev->req_workqueue, &hdev->bg_scan_update);
1198c2ecf20Sopenharmony_ci}
1208c2ecf20Sopenharmony_ci
1218c2ecf20Sopenharmony_civoid hci_request_setup(struct hci_dev *hdev);
1228c2ecf20Sopenharmony_civoid hci_request_cancel_all(struct hci_dev *hdev);
1238c2ecf20Sopenharmony_ci
1248c2ecf20Sopenharmony_ciu8 append_local_name(struct hci_dev *hdev, u8 *ptr, u8 ad_len);
1258c2ecf20Sopenharmony_ci
1268c2ecf20Sopenharmony_cistatic inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type,
1278c2ecf20Sopenharmony_ci				  u8 *data, u8 data_len)
1288c2ecf20Sopenharmony_ci{
1298c2ecf20Sopenharmony_ci	eir[eir_len++] = sizeof(type) + data_len;
1308c2ecf20Sopenharmony_ci	eir[eir_len++] = type;
1318c2ecf20Sopenharmony_ci	memcpy(&eir[eir_len], data, data_len);
1328c2ecf20Sopenharmony_ci	eir_len += data_len;
1338c2ecf20Sopenharmony_ci
1348c2ecf20Sopenharmony_ci	return eir_len;
1358c2ecf20Sopenharmony_ci}
1368c2ecf20Sopenharmony_ci
1378c2ecf20Sopenharmony_cistatic inline u16 eir_append_le16(u8 *eir, u16 eir_len, u8 type, u16 data)
1388c2ecf20Sopenharmony_ci{
1398c2ecf20Sopenharmony_ci	eir[eir_len++] = sizeof(type) + sizeof(data);
1408c2ecf20Sopenharmony_ci	eir[eir_len++] = type;
1418c2ecf20Sopenharmony_ci	put_unaligned_le16(data, &eir[eir_len]);
1428c2ecf20Sopenharmony_ci	eir_len += sizeof(data);
1438c2ecf20Sopenharmony_ci
1448c2ecf20Sopenharmony_ci	return eir_len;
1458c2ecf20Sopenharmony_ci}
146