Commit | Line | Data |
---|---|---|
0857dd3b JH |
1 | /* |
2 | BlueZ - Bluetooth protocol stack for Linux | |
3 | Copyright (C) 2014 Intel Corporation | |
4 | ||
5 | This program is free software; you can redistribute it and/or modify | |
6 | it under the terms of the GNU General Public License version 2 as | |
7 | published by the Free Software Foundation; | |
8 | ||
9 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | |
10 | OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
11 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS. | |
12 | IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY | |
13 | CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES | |
14 | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |
15 | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | |
16 | OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |
17 | ||
18 | ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS, | |
19 | COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS | |
20 | SOFTWARE IS DISCLAIMED. | |
21 | */ | |
22 | ||
1b422066 MN |
23 | #include <asm/unaligned.h> |
24 | ||
b504430c JH |
25 | #define hci_req_sync_lock(hdev) mutex_lock(&hdev->req_lock) |
26 | #define hci_req_sync_unlock(hdev) mutex_unlock(&hdev->req_lock) | |
be91cd05 | 27 | |
0857dd3b JH |
28 | struct hci_request { |
29 | struct hci_dev *hdev; | |
30 | struct sk_buff_head cmd_q; | |
31 | ||
32 | /* If something goes wrong when building the HCI request, the error | |
33 | * value is stored in this field. | |
34 | */ | |
35 | int err; | |
36 | }; | |
37 | ||
38 | void hci_req_init(struct hci_request *req, struct hci_dev *hdev); | |
f17d858e | 39 | void hci_req_purge(struct hci_request *req); |
f80c5dad | 40 | bool hci_req_status_pend(struct hci_dev *hdev); |
0857dd3b | 41 | int hci_req_run(struct hci_request *req, hci_req_complete_t complete); |
e6214487 | 42 | int hci_req_run_skb(struct hci_request *req, hci_req_complete_skb_t complete); |
0857dd3b JH |
43 | void hci_req_add(struct hci_request *req, u16 opcode, u32 plen, |
44 | const void *param); | |
45 | void hci_req_add_ev(struct hci_request *req, u16 opcode, u32 plen, | |
46 | const void *param, u8 event); | |
e6214487 JH |
47 | void hci_req_cmd_complete(struct hci_dev *hdev, u16 opcode, u8 status, |
48 | hci_req_complete_t *req_complete, | |
49 | hci_req_complete_skb_t *req_complete_skb); | |
0857dd3b | 50 | |
a1d01db1 JH |
51 | int hci_req_sync(struct hci_dev *hdev, int (*req)(struct hci_request *req, |
52 | unsigned long opt), | |
4ebeee2d | 53 | unsigned long opt, u32 timeout, u8 *hci_status); |
a1d01db1 JH |
54 | int __hci_req_sync(struct hci_dev *hdev, int (*func)(struct hci_request *req, |
55 | unsigned long opt), | |
4ebeee2d | 56 | unsigned long opt, u32 timeout, u8 *hci_status); |
b504430c | 57 | void hci_req_sync_cancel(struct hci_dev *hdev, int err); |
be91cd05 | 58 | |
0857dd3b JH |
59 | struct sk_buff *hci_prepare_cmd(struct hci_dev *hdev, u16 opcode, u32 plen, |
60 | const void *param); | |
61 | ||
2ff13894 JH |
62 | int __hci_req_hci_power_on(struct hci_dev *hdev); |
63 | ||
bf943cbf | 64 | void __hci_req_write_fast_connectable(struct hci_request *req, bool enable); |
00cf5040 | 65 | void __hci_req_update_name(struct hci_request *req); |
b1a8917c | 66 | void __hci_req_update_eir(struct hci_request *req); |
00cf5040 | 67 | |
5c49bcce | 68 | void hci_req_add_le_scan_disable(struct hci_request *req, bool rpa_le_conn); |
0857dd3b JH |
69 | void hci_req_add_le_passive_scan(struct hci_request *req); |
70 | ||
9952d90e APS |
71 | void hci_req_prepare_suspend(struct hci_dev *hdev, enum suspended_state next); |
72 | ||
5c49bcce | 73 | void hci_req_disable_address_resolution(struct hci_dev *hdev); |
2943d8ed DW |
74 | void __hci_req_pause_adv_instances(struct hci_request *req); |
75 | int hci_req_resume_adv_instances(struct hci_dev *hdev); | |
f2252570 JH |
76 | void hci_req_reenable_advertising(struct hci_dev *hdev); |
77 | void __hci_req_enable_advertising(struct hci_request *req); | |
78 | void __hci_req_disable_advertising(struct hci_request *req); | |
cab054ab JH |
79 | void __hci_req_update_adv_data(struct hci_request *req, u8 instance); |
80 | int hci_req_update_adv_data(struct hci_dev *hdev, u8 instance); | |
81 | void __hci_req_update_scan_rsp_data(struct hci_request *req, u8 instance); | |
f2252570 JH |
82 | |
83 | int __hci_req_schedule_adv_instance(struct hci_request *req, u8 instance, | |
84 | bool force); | |
37d3a1fa JH |
85 | void hci_req_clear_adv_instance(struct hci_dev *hdev, struct sock *sk, |
86 | struct hci_request *req, u8 instance, | |
87 | bool force); | |
f2252570 | 88 | |
a0fb3726 | 89 | int __hci_req_setup_ext_adv_instance(struct hci_request *req, u8 instance); |
de181e88 | 90 | int __hci_req_start_ext_adv(struct hci_request *req, u8 instance); |
1d0fac2c | 91 | int __hci_req_enable_ext_advertising(struct hci_request *req, u8 instance); |
37adf701 DW |
92 | int __hci_req_disable_ext_adv_instance(struct hci_request *req, u8 instance); |
93 | int __hci_req_remove_ext_adv_instance(struct hci_request *req, u8 instance); | |
45b7749f | 94 | void __hci_req_clear_ext_adv_sets(struct hci_request *req); |
a73c046a JK |
95 | int hci_get_random_address(struct hci_dev *hdev, bool require_privacy, |
96 | bool use_rpa, struct adv_info *adv_instance, | |
97 | u8 *own_addr_type, bdaddr_t *rand_addr); | |
de181e88 | 98 | |
14bf5eac JH |
99 | void __hci_req_update_class(struct hci_request *req); |
100 | ||
2154d3f4 JH |
101 | /* Returns true if HCI commands were queued */ |
102 | bool hci_req_stop_discovery(struct hci_request *req); | |
103 | ||
01b1cb87 JH |
104 | static inline void hci_req_update_scan(struct hci_dev *hdev) |
105 | { | |
106 | queue_work(hdev->req_workqueue, &hdev->scan_update); | |
107 | } | |
108 | ||
109 | void __hci_req_update_scan(struct hci_request *req); | |
0857dd3b JH |
110 | |
111 | int hci_update_random_address(struct hci_request *req, bool require_privacy, | |
82a37ade | 112 | bool use_rpa, u8 *own_addr_type); |
2cf22218 | 113 | |
dcc0f0d9 JH |
114 | int hci_abort_conn(struct hci_conn *conn, u8 reason); |
115 | void __hci_abort_conn(struct hci_request *req, struct hci_conn *conn, | |
116 | u8 reason); | |
5fc16cc4 | 117 | |
2e93e53b JH |
118 | static inline void hci_update_background_scan(struct hci_dev *hdev) |
119 | { | |
120 | queue_work(hdev->req_workqueue, &hdev->bg_scan_update); | |
121 | } | |
122 | ||
5fc16cc4 JH |
123 | void hci_request_setup(struct hci_dev *hdev); |
124 | void hci_request_cancel_all(struct hci_dev *hdev); | |
1b422066 | 125 | |
f61851f6 MN |
126 | u8 append_local_name(struct hci_dev *hdev, u8 *ptr, u8 ad_len); |
127 | ||
1b422066 MN |
128 | static inline u16 eir_append_data(u8 *eir, u16 eir_len, u8 type, |
129 | u8 *data, u8 data_len) | |
130 | { | |
131 | eir[eir_len++] = sizeof(type) + data_len; | |
132 | eir[eir_len++] = type; | |
133 | memcpy(&eir[eir_len], data, data_len); | |
134 | eir_len += data_len; | |
135 | ||
136 | return eir_len; | |
137 | } | |
138 | ||
139 | static inline u16 eir_append_le16(u8 *eir, u16 eir_len, u8 type, u16 data) | |
140 | { | |
141 | eir[eir_len++] = sizeof(type) + sizeof(data); | |
142 | eir[eir_len++] = type; | |
143 | put_unaligned_le16(data, &eir[eir_len]); | |
144 | eir_len += sizeof(data); | |
145 | ||
146 | return eir_len; | |
147 | } |