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