| /* |
| BlueZ - Bluetooth protocol stack for Linux |
| Copyright (C) 2014 Intel Corporation |
| |
| 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 |
| 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 |
| OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
| |
| ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS, |
| COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS |
| SOFTWARE IS DISCLAIMED. |
| */ |
| |
| #include <asm/unaligned.h> |
| |
| #define hci_req_sync_lock(hdev) mutex_lock(&hdev->req_lock) |
| #define hci_req_sync_unlock(hdev) mutex_unlock(&hdev->req_lock) |
| |
| struct hci_request { |
| struct hci_dev *hdev; |
| struct sk_buff_head cmd_q; |
| |
| /* If something goes wrong when building the HCI request, the error |
| * value is stored in this field. |
| */ |
| int err; |
| }; |
| |
| void hci_req_init(struct hci_request *req, struct hci_dev *hdev); |
| void hci_req_purge(struct hci_request *req); |
| bool hci_req_status_pend(struct hci_dev *hdev); |
| int hci_req_run(struct hci_request *req, hci_req_complete_t complete); |
| int hci_req_run_skb(struct hci_request *req, hci_req_complete_skb_t complete); |
| void hci_req_add(struct hci_request *req, u16 opcode, u32 plen, |
| const void *param); |
| void hci_req_add_ev(struct hci_request *req, u16 opcode, u32 plen, |
| const void *param, u8 event); |
| void hci_req_cmd_complete(struct hci_dev *hdev, u16 opcode, u8 status, |
| hci_req_complete_t *req_complete, |
| hci_req_complete_skb_t *req_complete_skb); |
| |
| int hci_req_sync(struct hci_dev *hdev, int (*req)(struct hci_request *req, |
| unsigned long opt), |
| unsigned long opt, u32 timeout, u8 *hci_status); |
| int __hci_req_sync(struct hci_dev *hdev, int (*func)(struct hci_request *req, |
| unsigned long opt), |
| unsigned long opt, u32 timeout, u8 *hci_status); |
| void hci_req_sync_cancel(struct hci_dev *hdev, int err); |
| |
| struct sk_buff *hci_prepare_cmd(struct hci_dev *hdev, u16 opcode, u32 plen, |
| const void *param); |
| |
| int __hci_req_hci_power_on(struct hci_dev *hdev); |
| |
| void __hci_req_write_fast_connectable(struct hci_request *req, bool enable); |
| void __hci_req_update_name(struct hci_request *req); |
| void __hci_req_update_eir(struct hci_request *req); |
| |
| void hci_req_add_le_scan_disable(struct hci_request *req); |
| void hci_req_add_le_passive_scan(struct hci_request *req); |
| |
| void hci_req_prepare_suspend(struct hci_dev *hdev, enum suspended_state next); |
| |
| void hci_req_reenable_advertising(struct hci_dev *hdev); |
| void __hci_req_enable_advertising(struct hci_request *req); |
| void __hci_req_disable_advertising(struct hci_request *req); |
| void __hci_req_update_adv_data(struct hci_request *req, u8 instance); |
| int hci_req_update_adv_data(struct hci_dev *hdev, u8 instance); |
| void __hci_req_update_scan_rsp_data(struct hci_request *req, u8 instance); |
| |
| int __hci_req_schedule_adv_instance(struct hci_request *req, u8 instance, |
| bool force); |
| void hci_req_clear_adv_instance(struct hci_dev *hdev, struct sock *sk, |
| struct hci_request *req, u8 instance, |
| bool force); |
| |
| int __hci_req_setup_ext_adv_instance(struct hci_request *req, u8 instance); |
| int __hci_req_start_ext_adv(struct hci_request *req, u8 instance); |
| int __hci_req_enable_ext_advertising(struct hci_request *req, u8 instance); |
| void __hci_req_clear_ext_adv_sets(struct hci_request *req); |
| int hci_get_random_address(struct hci_dev *hdev, bool require_privacy, |
| bool use_rpa, struct adv_info *adv_instance, |
| u8 *own_addr_type, bdaddr_t *rand_addr); |
| |
| void __hci_req_update_class(struct hci_request *req); |
| |
| /* Returns true if HCI commands were queued */ |
| bool hci_req_stop_discovery(struct hci_request *req); |
| |
| static inline void hci_req_update_scan(struct hci_dev *hdev) |
| { |
| queue_work(hdev->req_workqueue, &hdev->scan_update); |
| } |
| |
| void __hci_req_update_scan(struct hci_request *req); |
| |
| int hci_update_random_address(struct hci_request *req, bool require_privacy, |
| bool use_rpa, u8 *own_addr_type); |
| |
| int hci_abort_conn(struct hci_conn *conn, u8 reason); |
| void __hci_abort_conn(struct hci_request *req, struct hci_conn *conn, |
| u8 reason); |
| |
| static inline void hci_update_background_scan(struct hci_dev *hdev) |
| { |
| queue_work(hdev->req_workqueue, &hdev->bg_scan_update); |
| } |
| |
| void hci_request_setup(struct hci_dev *hdev); |
| void hci_request_cancel_all(struct hci_dev *hdev); |
| |
| u8 append_local_name(struct hci_dev *hdev, u8 *ptr, u8 ad_len); |
| |
| static inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type, |
| u8 *data, u8 data_len) |
| { |
| eir[eir_len++] = sizeof(type) + data_len; |
| eir[eir_len++] = type; |
| memcpy(&eir[eir_len], data, data_len); |
| eir_len += data_len; |
| |
| return eir_len; |
| } |
| |
| static inline u16 eir_append_le16(u8 *eir, u16 eir_len, u8 type, u16 data) |
| { |
| eir[eir_len++] = sizeof(type) + sizeof(data); |
| eir[eir_len++] = type; |
| put_unaligned_le16(data, &eir[eir_len]); |
| eir_len += sizeof(data); |
| |
| return eir_len; |
| } |