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