blob: 9c7e4edc650404776d43aa6846c8a2b6d1210b71 [file] [log] [blame]
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +09001/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07002 BlueZ - Bluetooth protocol stack for Linux
3 Copyright (C) 2000-2001 Qualcomm Incorporated
4
5 Written 2000,2001 by Maxim Krasnyansky <maxk@qualcomm.com>
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License version 2 as
9 published by the Free Software Foundation;
10
11 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
12 OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
13 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.
14 IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +090015 CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES
16 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
17 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
Linus Torvalds1da177e2005-04-16 15:20:36 -070018 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
19
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +090020 ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,
21 COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS
Linus Torvalds1da177e2005-04-16 15:20:36 -070022 SOFTWARE IS DISCLAIMED.
23*/
24
25/* Bluetooth address family and sockets. */
26
Linus Torvalds1da177e2005-04-16 15:20:36 -070027#include <linux/module.h>
Marcel Holtmann3241ad82008-07-14 20:13:50 +020028#include <asm/ioctls.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070029
30#include <net/bluetooth/bluetooth.h>
Masatake YAMATO256a06c2012-07-26 01:26:32 +090031#include <linux/proc_fs.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070032
Gustavo F. Padovan64274512011-02-07 20:08:52 -020033#define VERSION "2.16"
Linus Torvalds1da177e2005-04-16 15:20:36 -070034
35/* Bluetooth sockets */
36#define BT_MAX_PROTO 8
Stephen Hemmingerec1b4cf2009-10-05 05:58:39 +000037static const struct net_proto_family *bt_proto[BT_MAX_PROTO];
Marcel Holtmanndb7aa1c2008-11-30 12:17:19 +010038static DEFINE_RWLOCK(bt_proto_lock);
Dave Young68845cb2008-04-01 23:58:35 -070039
Dave Young68845cb2008-04-01 23:58:35 -070040static struct lock_class_key bt_lock_key[BT_MAX_PROTO];
Jan Engelhardt36cbd3d2009-08-05 10:42:58 -070041static const char *const bt_key_strings[BT_MAX_PROTO] = {
Dave Young68845cb2008-04-01 23:58:35 -070042 "sk_lock-AF_BLUETOOTH-BTPROTO_L2CAP",
43 "sk_lock-AF_BLUETOOTH-BTPROTO_HCI",
44 "sk_lock-AF_BLUETOOTH-BTPROTO_SCO",
45 "sk_lock-AF_BLUETOOTH-BTPROTO_RFCOMM",
46 "sk_lock-AF_BLUETOOTH-BTPROTO_BNEP",
47 "sk_lock-AF_BLUETOOTH-BTPROTO_CMTP",
48 "sk_lock-AF_BLUETOOTH-BTPROTO_HIDP",
49 "sk_lock-AF_BLUETOOTH-BTPROTO_AVDTP",
50};
51
Marcel Holtmanndb7aa1c2008-11-30 12:17:19 +010052static struct lock_class_key bt_slock_key[BT_MAX_PROTO];
Jan Engelhardt36cbd3d2009-08-05 10:42:58 -070053static const char *const bt_slock_key_strings[BT_MAX_PROTO] = {
Dave Young68845cb2008-04-01 23:58:35 -070054 "slock-AF_BLUETOOTH-BTPROTO_L2CAP",
55 "slock-AF_BLUETOOTH-BTPROTO_HCI",
56 "slock-AF_BLUETOOTH-BTPROTO_SCO",
57 "slock-AF_BLUETOOTH-BTPROTO_RFCOMM",
58 "slock-AF_BLUETOOTH-BTPROTO_BNEP",
59 "slock-AF_BLUETOOTH-BTPROTO_CMTP",
60 "slock-AF_BLUETOOTH-BTPROTO_HIDP",
61 "slock-AF_BLUETOOTH-BTPROTO_AVDTP",
62};
Marcel Holtmanndb7aa1c2008-11-30 12:17:19 +010063
Octavian Purdilab5a30dd2012-01-22 00:28:34 +020064void bt_sock_reclassify_lock(struct sock *sk, int proto)
Marcel Holtmanndb7aa1c2008-11-30 12:17:19 +010065{
Octavian Purdilab5a30dd2012-01-22 00:28:34 +020066 BUG_ON(!sk);
Marcel Holtmanndb7aa1c2008-11-30 12:17:19 +010067 BUG_ON(sock_owned_by_user(sk));
68
69 sock_lock_init_class_and_name(sk,
70 bt_slock_key_strings[proto], &bt_slock_key[proto],
71 bt_key_strings[proto], &bt_lock_key[proto]);
72}
Octavian Purdilab5a30dd2012-01-22 00:28:34 +020073EXPORT_SYMBOL(bt_sock_reclassify_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070074
Stephen Hemmingerec1b4cf2009-10-05 05:58:39 +000075int bt_sock_register(int proto, const struct net_proto_family *ops)
Linus Torvalds1da177e2005-04-16 15:20:36 -070076{
Marcel Holtmann74da6262006-10-15 17:31:14 +020077 int err = 0;
78
Linus Torvalds1da177e2005-04-16 15:20:36 -070079 if (proto < 0 || proto >= BT_MAX_PROTO)
80 return -EINVAL;
81
Marcel Holtmann74da6262006-10-15 17:31:14 +020082 write_lock(&bt_proto_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070083
Marcel Holtmann74da6262006-10-15 17:31:14 +020084 if (bt_proto[proto])
85 err = -EEXIST;
86 else
87 bt_proto[proto] = ops;
88
89 write_unlock(&bt_proto_lock);
90
91 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -070092}
93EXPORT_SYMBOL(bt_sock_register);
94
David Herrmannbe9f97f2013-02-24 19:36:52 +010095void bt_sock_unregister(int proto)
Linus Torvalds1da177e2005-04-16 15:20:36 -070096{
97 if (proto < 0 || proto >= BT_MAX_PROTO)
David Herrmannbe9f97f2013-02-24 19:36:52 +010098 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -070099
Marcel Holtmann74da6262006-10-15 17:31:14 +0200100 write_lock(&bt_proto_lock);
David Herrmannbe9f97f2013-02-24 19:36:52 +0100101 bt_proto[proto] = NULL;
Marcel Holtmann74da6262006-10-15 17:31:14 +0200102 write_unlock(&bt_proto_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103}
104EXPORT_SYMBOL(bt_sock_unregister);
105
Eric Paris3f378b62009-11-05 22:18:14 -0800106static int bt_sock_create(struct net *net, struct socket *sock, int proto,
107 int kern)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700108{
Marcel Holtmann74da6262006-10-15 17:31:14 +0200109 int err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700110
Eric W. Biederman1b8d7ae2007-10-08 23:24:22 -0700111 if (net != &init_net)
112 return -EAFNOSUPPORT;
113
Linus Torvalds1da177e2005-04-16 15:20:36 -0700114 if (proto < 0 || proto >= BT_MAX_PROTO)
115 return -EINVAL;
116
Johannes Berg95a5afc2008-10-16 15:24:51 -0700117 if (!bt_proto[proto])
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118 request_module("bt-proto-%d", proto);
Marcel Holtmann74da6262006-10-15 17:31:14 +0200119
Linus Torvalds1da177e2005-04-16 15:20:36 -0700120 err = -EPROTONOSUPPORT;
Marcel Holtmann74da6262006-10-15 17:31:14 +0200121
122 read_lock(&bt_proto_lock);
123
Linus Torvalds1da177e2005-04-16 15:20:36 -0700124 if (bt_proto[proto] && try_module_get(bt_proto[proto]->owner)) {
Eric Paris3f378b62009-11-05 22:18:14 -0800125 err = bt_proto[proto]->create(net, sock, proto, kern);
Octavian Purdilab5a30dd2012-01-22 00:28:34 +0200126 if (!err)
127 bt_sock_reclassify_lock(sock->sk, proto);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700128 module_put(bt_proto[proto]->owner);
129 }
Marcel Holtmann74da6262006-10-15 17:31:14 +0200130
131 read_unlock(&bt_proto_lock);
132
YOSHIFUJI Hideaki8e87d142007-02-09 23:24:33 +0900133 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700134}
135
136void bt_sock_link(struct bt_sock_list *l, struct sock *sk)
137{
Gustavo F. Padovan94f5bfb2011-12-27 15:28:48 -0200138 write_lock(&l->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700139 sk_add_node(sk, &l->head);
Gustavo F. Padovan94f5bfb2011-12-27 15:28:48 -0200140 write_unlock(&l->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700141}
142EXPORT_SYMBOL(bt_sock_link);
143
144void bt_sock_unlink(struct bt_sock_list *l, struct sock *sk)
145{
Gustavo F. Padovan94f5bfb2011-12-27 15:28:48 -0200146 write_lock(&l->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700147 sk_del_node_init(sk);
Gustavo F. Padovan94f5bfb2011-12-27 15:28:48 -0200148 write_unlock(&l->lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700149}
150EXPORT_SYMBOL(bt_sock_unlink);
151
152void bt_accept_enqueue(struct sock *parent, struct sock *sk)
153{
154 BT_DBG("parent %p, sk %p", parent, sk);
155
156 sock_hold(sk);
157 list_add_tail(&bt_sk(sk)->accept_q, &bt_sk(parent)->accept_q);
158 bt_sk(sk)->parent = parent;
159 parent->sk_ack_backlog++;
160}
161EXPORT_SYMBOL(bt_accept_enqueue);
162
163void bt_accept_unlink(struct sock *sk)
164{
165 BT_DBG("sk %p state %d", sk, sk->sk_state);
166
167 list_del_init(&bt_sk(sk)->accept_q);
168 bt_sk(sk)->parent->sk_ack_backlog--;
169 bt_sk(sk)->parent = NULL;
170 sock_put(sk);
171}
172EXPORT_SYMBOL(bt_accept_unlink);
173
174struct sock *bt_accept_dequeue(struct sock *parent, struct socket *newsock)
175{
176 struct list_head *p, *n;
177 struct sock *sk;
178
179 BT_DBG("parent %p", parent);
180
181 list_for_each_safe(p, n, &bt_sk(parent)->accept_q) {
182 sk = (struct sock *) list_entry(p, struct bt_sock, accept_q);
183
Gustavo F. Padovan8a154a82011-12-20 17:15:56 -0200184 lock_sock(sk);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700185
186 /* FIXME: Is this check still needed */
187 if (sk->sk_state == BT_CLOSED) {
Gustavo F. Padovan8a154a82011-12-20 17:15:56 -0200188 release_sock(sk);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700189 bt_accept_unlink(sk);
190 continue;
191 }
192
Marcel Holtmannc4f912e2009-01-15 21:52:16 +0100193 if (sk->sk_state == BT_CONNECTED || !newsock ||
Vinicius Costa Gomesd0609912012-05-31 22:53:39 -0300194 test_bit(BT_SK_DEFER_SETUP, &bt_sk(parent)->flags)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700195 bt_accept_unlink(sk);
196 if (newsock)
197 sock_graft(sk, newsock);
Andrei Emeltchenkod37f50e2011-01-24 10:53:24 +0200198
Gustavo F. Padovan8a154a82011-12-20 17:15:56 -0200199 release_sock(sk);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700200 return sk;
201 }
202
Gustavo F. Padovan8a154a82011-12-20 17:15:56 -0200203 release_sock(sk);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700204 }
Andrei Emeltchenkod37f50e2011-01-24 10:53:24 +0200205
Linus Torvalds1da177e2005-04-16 15:20:36 -0700206 return NULL;
207}
208EXPORT_SYMBOL(bt_accept_dequeue);
209
210int bt_sock_recvmsg(struct kiocb *iocb, struct socket *sock,
Marcel Holtmannc4f912e2009-01-15 21:52:16 +0100211 struct msghdr *msg, size_t len, int flags)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700212{
213 int noblock = flags & MSG_DONTWAIT;
214 struct sock *sk = sock->sk;
215 struct sk_buff *skb;
216 size_t copied;
217 int err;
218
Marcel Holtmanna418b892008-11-30 12:17:28 +0100219 BT_DBG("sock %p sk %p len %zu", sock, sk, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700220
221 if (flags & (MSG_OOB))
222 return -EOPNOTSUPP;
223
Mathias Krause4683f422013-04-07 01:51:49 +0000224 msg->msg_namelen = 0;
225
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +0200226 skb = skb_recv_datagram(sk, flags, noblock, &err);
227 if (!skb) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700228 if (sk->sk_shutdown & RCV_SHUTDOWN)
229 return 0;
230 return err;
231 }
232
Linus Torvalds1da177e2005-04-16 15:20:36 -0700233 copied = skb->len;
234 if (len < copied) {
235 msg->msg_flags |= MSG_TRUNC;
236 copied = len;
237 }
238
Arnaldo Carvalho de Melobadff6d2007-03-13 13:06:52 -0300239 skb_reset_transport_header(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700240 err = skb_copy_datagram_iovec(skb, 0, msg->msg_iov, copied);
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200241 if (err == 0)
Neil Horman3b885782009-10-12 13:26:31 -0700242 sock_recv_ts_and_drops(msg, sk, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243
244 skb_free_datagram(sk, skb);
245
246 return err ? : copied;
247}
248EXPORT_SYMBOL(bt_sock_recvmsg);
249
Mat Martineau796c86e2010-09-08 10:05:27 -0700250static long bt_sock_data_wait(struct sock *sk, long timeo)
251{
252 DECLARE_WAITQUEUE(wait, current);
253
254 add_wait_queue(sk_sleep(sk), &wait);
255 for (;;) {
256 set_current_state(TASK_INTERRUPTIBLE);
257
258 if (!skb_queue_empty(&sk->sk_receive_queue))
259 break;
260
261 if (sk->sk_err || (sk->sk_shutdown & RCV_SHUTDOWN))
262 break;
263
264 if (signal_pending(current) || !timeo)
265 break;
266
267 set_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
268 release_sock(sk);
269 timeo = schedule_timeout(timeo);
270 lock_sock(sk);
271 clear_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
272 }
273
274 __set_current_state(TASK_RUNNING);
275 remove_wait_queue(sk_sleep(sk), &wait);
276 return timeo;
277}
278
279int bt_sock_stream_recvmsg(struct kiocb *iocb, struct socket *sock,
280 struct msghdr *msg, size_t size, int flags)
281{
282 struct sock *sk = sock->sk;
283 int err = 0;
284 size_t target, copied = 0;
285 long timeo;
286
287 if (flags & MSG_OOB)
288 return -EOPNOTSUPP;
289
290 msg->msg_namelen = 0;
291
292 BT_DBG("sk %p size %zu", sk, size);
293
294 lock_sock(sk);
295
296 target = sock_rcvlowat(sk, flags & MSG_WAITALL, size);
297 timeo = sock_rcvtimeo(sk, flags & MSG_DONTWAIT);
298
299 do {
300 struct sk_buff *skb;
301 int chunk;
302
303 skb = skb_dequeue(&sk->sk_receive_queue);
304 if (!skb) {
305 if (copied >= target)
306 break;
307
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +0200308 err = sock_error(sk);
309 if (err)
Mat Martineau796c86e2010-09-08 10:05:27 -0700310 break;
311 if (sk->sk_shutdown & RCV_SHUTDOWN)
312 break;
313
314 err = -EAGAIN;
315 if (!timeo)
316 break;
317
318 timeo = bt_sock_data_wait(sk, timeo);
319
320 if (signal_pending(current)) {
321 err = sock_intr_errno(timeo);
322 goto out;
323 }
324 continue;
325 }
326
327 chunk = min_t(unsigned int, skb->len, size);
Mat Martineau5b668eb2011-07-22 14:53:59 -0700328 if (skb_copy_datagram_iovec(skb, 0, msg->msg_iov, chunk)) {
Mat Martineau796c86e2010-09-08 10:05:27 -0700329 skb_queue_head(&sk->sk_receive_queue, skb);
330 if (!copied)
331 copied = -EFAULT;
332 break;
333 }
334 copied += chunk;
335 size -= chunk;
336
337 sock_recv_ts_and_drops(msg, sk, skb);
338
339 if (!(flags & MSG_PEEK)) {
Mat Martineau5b668eb2011-07-22 14:53:59 -0700340 int skb_len = skb_headlen(skb);
341
342 if (chunk <= skb_len) {
343 __skb_pull(skb, chunk);
344 } else {
345 struct sk_buff *frag;
346
347 __skb_pull(skb, skb_len);
348 chunk -= skb_len;
349
350 skb_walk_frags(skb, frag) {
351 if (chunk <= frag->len) {
352 /* Pulling partial data */
353 skb->len -= chunk;
354 skb->data_len -= chunk;
355 __skb_pull(frag, chunk);
356 break;
357 } else if (frag->len) {
358 /* Pulling all frag data */
359 chunk -= frag->len;
360 skb->len -= frag->len;
361 skb->data_len -= frag->len;
362 __skb_pull(frag, frag->len);
363 }
364 }
365 }
366
Mat Martineau796c86e2010-09-08 10:05:27 -0700367 if (skb->len) {
368 skb_queue_head(&sk->sk_receive_queue, skb);
369 break;
370 }
371 kfree_skb(skb);
372
373 } else {
374 /* put message back and return */
375 skb_queue_head(&sk->sk_receive_queue, skb);
376 break;
377 }
378 } while (size);
379
380out:
381 release_sock(sk);
382 return copied ? : err;
383}
384EXPORT_SYMBOL(bt_sock_stream_recvmsg);
385
Linus Torvalds1da177e2005-04-16 15:20:36 -0700386static inline unsigned int bt_accept_poll(struct sock *parent)
387{
388 struct list_head *p, *n;
389 struct sock *sk;
390
391 list_for_each_safe(p, n, &bt_sk(parent)->accept_q) {
392 sk = (struct sock *) list_entry(p, struct bt_sock, accept_q);
Marcel Holtmannd5f2d2b2009-02-16 02:57:30 +0100393 if (sk->sk_state == BT_CONNECTED ||
Gustavo Padovanc5daa682012-05-16 12:17:10 -0300394 (test_bit(BT_SK_DEFER_SETUP, &bt_sk(parent)->flags) &&
395 sk->sk_state == BT_CONNECT2))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700396 return POLLIN | POLLRDNORM;
397 }
398
399 return 0;
400}
401
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -0300402unsigned int bt_sock_poll(struct file *file, struct socket *sock,
403 poll_table *wait)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404{
405 struct sock *sk = sock->sk;
406 unsigned int mask = 0;
407
408 BT_DBG("sock %p, sk %p", sock, sk);
409
Eric Dumazetaa395142010-04-20 13:03:51 +0000410 poll_wait(file, sk_sleep(sk), wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700411
412 if (sk->sk_state == BT_LISTEN)
413 return bt_accept_poll(sk);
414
415 if (sk->sk_err || !skb_queue_empty(&sk->sk_error_queue))
Keller, Jacob E7d4c04f2013-03-28 11:19:25 +0000416 mask |= POLLERR |
Jacob Keller8facd5f2013-04-02 13:55:40 -0700417 (sock_flag(sk, SOCK_SELECT_ERR_QUEUE) ? POLLPRI : 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700418
Davide Libenzif348d702006-03-25 03:07:39 -0800419 if (sk->sk_shutdown & RCV_SHUTDOWN)
Eric Dumazetdb409802010-09-06 11:13:50 +0000420 mask |= POLLRDHUP | POLLIN | POLLRDNORM;
Davide Libenzif348d702006-03-25 03:07:39 -0800421
Linus Torvalds1da177e2005-04-16 15:20:36 -0700422 if (sk->sk_shutdown == SHUTDOWN_MASK)
423 mask |= POLLHUP;
424
Eric Dumazetdb409802010-09-06 11:13:50 +0000425 if (!skb_queue_empty(&sk->sk_receive_queue))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700426 mask |= POLLIN | POLLRDNORM;
427
428 if (sk->sk_state == BT_CLOSED)
429 mask |= POLLHUP;
430
431 if (sk->sk_state == BT_CONNECT ||
432 sk->sk_state == BT_CONNECT2 ||
433 sk->sk_state == BT_CONFIG)
434 return mask;
435
Gustavo Padovanc5daa682012-05-16 12:17:10 -0300436 if (!test_bit(BT_SK_SUSPEND, &bt_sk(sk)->flags) && sock_writeable(sk))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700437 mask |= POLLOUT | POLLWRNORM | POLLWRBAND;
438 else
439 set_bit(SOCK_ASYNC_NOSPACE, &sk->sk_socket->flags);
440
441 return mask;
442}
443EXPORT_SYMBOL(bt_sock_poll);
444
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200445int bt_sock_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)
446{
447 struct sock *sk = sock->sk;
Marcel Holtmann43cbeee2008-07-14 20:13:51 +0200448 struct sk_buff *skb;
449 long amount;
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200450 int err;
451
452 BT_DBG("sk %p cmd %x arg %lx", sk, cmd, arg);
453
454 switch (cmd) {
Marcel Holtmann43cbeee2008-07-14 20:13:51 +0200455 case TIOCOUTQ:
456 if (sk->sk_state == BT_LISTEN)
457 return -EINVAL;
458
Eric Dumazet31e6d362009-06-17 19:05:41 -0700459 amount = sk->sk_sndbuf - sk_wmem_alloc_get(sk);
Marcel Holtmann43cbeee2008-07-14 20:13:51 +0200460 if (amount < 0)
461 amount = 0;
462 err = put_user(amount, (int __user *) arg);
463 break;
464
465 case TIOCINQ:
466 if (sk->sk_state == BT_LISTEN)
467 return -EINVAL;
468
469 lock_sock(sk);
470 skb = skb_peek(&sk->sk_receive_queue);
471 amount = skb ? skb->len : 0;
472 release_sock(sk);
473 err = put_user(amount, (int __user *) arg);
474 break;
475
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200476 case SIOCGSTAMP:
477 err = sock_get_timestamp(sk, (struct timeval __user *) arg);
478 break;
479
480 case SIOCGSTAMPNS:
481 err = sock_get_timestampns(sk, (struct timespec __user *) arg);
482 break;
483
484 default:
485 err = -ENOIOCTLCMD;
486 break;
487 }
488
489 return err;
490}
491EXPORT_SYMBOL(bt_sock_ioctl);
492
Johan Hedberg0fba96f2013-09-25 13:26:04 +0300493/* This function expects the sk lock to be held when called */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700494int bt_sock_wait_state(struct sock *sk, int state, unsigned long timeo)
495{
496 DECLARE_WAITQUEUE(wait, current);
497 int err = 0;
498
499 BT_DBG("sk %p", sk);
500
Eric Dumazetaa395142010-04-20 13:03:51 +0000501 add_wait_queue(sk_sleep(sk), &wait);
Peter Hurley9be4e3f2011-07-24 00:10:46 -0400502 set_current_state(TASK_INTERRUPTIBLE);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700503 while (sk->sk_state != state) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700504 if (!timeo) {
Marcel Holtmannb4c612a2006-09-23 09:54:38 +0200505 err = -EINPROGRESS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700506 break;
507 }
508
509 if (signal_pending(current)) {
510 err = sock_intr_errno(timeo);
511 break;
512 }
513
514 release_sock(sk);
515 timeo = schedule_timeout(timeo);
516 lock_sock(sk);
Peter Hurley9be4e3f2011-07-24 00:10:46 -0400517 set_current_state(TASK_INTERRUPTIBLE);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700518
Benjamin LaHaisec1cbe4b2005-12-13 23:22:19 -0800519 err = sock_error(sk);
520 if (err)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700521 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700522 }
Peter Hurley9be4e3f2011-07-24 00:10:46 -0400523 __set_current_state(TASK_RUNNING);
Eric Dumazetaa395142010-04-20 13:03:51 +0000524 remove_wait_queue(sk_sleep(sk), &wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700525 return err;
526}
527EXPORT_SYMBOL(bt_sock_wait_state);
528
Johan Hedberge793dcf2013-09-16 13:05:19 +0300529/* This function expects the sk lock to be held when called */
530int bt_sock_wait_ready(struct sock *sk, unsigned long flags)
531{
532 DECLARE_WAITQUEUE(wait, current);
533 unsigned long timeo;
534 int err = 0;
535
536 BT_DBG("sk %p", sk);
537
538 timeo = sock_sndtimeo(sk, flags & O_NONBLOCK);
539
540 add_wait_queue(sk_sleep(sk), &wait);
541 set_current_state(TASK_INTERRUPTIBLE);
542 while (test_bit(BT_SK_SUSPEND, &bt_sk(sk)->flags)) {
543 if (!timeo) {
544 err = -EAGAIN;
545 break;
546 }
547
548 if (signal_pending(current)) {
549 err = sock_intr_errno(timeo);
550 break;
551 }
552
553 release_sock(sk);
554 timeo = schedule_timeout(timeo);
555 lock_sock(sk);
556 set_current_state(TASK_INTERRUPTIBLE);
557
558 err = sock_error(sk);
559 if (err)
560 break;
561 }
562 __set_current_state(TASK_RUNNING);
563 remove_wait_queue(sk_sleep(sk), &wait);
564
565 return err;
566}
567EXPORT_SYMBOL(bt_sock_wait_ready);
568
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900569#ifdef CONFIG_PROC_FS
570struct bt_seq_state {
571 struct bt_sock_list *l;
572};
573
574static void *bt_seq_start(struct seq_file *seq, loff_t *pos)
575 __acquires(seq->private->l->lock)
576{
577 struct bt_seq_state *s = seq->private;
578 struct bt_sock_list *l = s->l;
579
580 read_lock(&l->lock);
581 return seq_hlist_start_head(&l->head, *pos);
582}
583
584static void *bt_seq_next(struct seq_file *seq, void *v, loff_t *pos)
585{
586 struct bt_seq_state *s = seq->private;
587 struct bt_sock_list *l = s->l;
588
589 return seq_hlist_next(v, &l->head, pos);
590}
591
592static void bt_seq_stop(struct seq_file *seq, void *v)
593 __releases(seq->private->l->lock)
594{
595 struct bt_seq_state *s = seq->private;
596 struct bt_sock_list *l = s->l;
597
598 read_unlock(&l->lock);
599}
600
601static int bt_seq_show(struct seq_file *seq, void *v)
602{
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900603 struct bt_seq_state *s = seq->private;
604 struct bt_sock_list *l = s->l;
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900605
606 if (v == SEQ_START_TOKEN) {
607 seq_puts(seq ,"sk RefCnt Rmem Wmem User Inode Src Dst Parent");
608
609 if (l->custom_seq_show) {
610 seq_putc(seq, ' ');
611 l->custom_seq_show(seq, v);
612 }
613
614 seq_putc(seq, '\n');
615 } else {
Andrei Emeltchenko09d5d4a2012-08-07 18:05:04 +0300616 struct sock *sk = sk_entry(v);
617 struct bt_sock *bt = bt_sk(sk);
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900618
Andrei Emeltchenko7028a882012-09-25 12:49:45 +0300619 seq_printf(seq,
Marcel Holtmann5f6cd792013-10-13 10:34:03 -0700620 "%pK %-6d %-6u %-6u %-6u %-6lu %-6lu",
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900621 sk,
622 atomic_read(&sk->sk_refcnt),
623 sk_rmem_alloc_get(sk),
624 sk_wmem_alloc_get(sk),
Eric W. Biederman1bbb3092012-10-03 20:32:17 -0700625 from_kuid(seq_user_ns(seq), sock_i_uid(sk)),
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900626 sock_i_ino(sk),
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900627 bt->parent? sock_i_ino(bt->parent): 0LU);
628
629 if (l->custom_seq_show) {
630 seq_putc(seq, ' ');
631 l->custom_seq_show(seq, v);
632 }
633
634 seq_putc(seq, '\n');
635 }
636 return 0;
637}
638
639static struct seq_operations bt_seq_ops = {
640 .start = bt_seq_start,
641 .next = bt_seq_next,
642 .stop = bt_seq_stop,
643 .show = bt_seq_show,
644};
645
646static int bt_seq_open(struct inode *inode, struct file *file)
647{
648 struct bt_sock_list *sk_list;
649 struct bt_seq_state *s;
650
Al Virod9dda782013-03-31 18:16:14 -0400651 sk_list = PDE_DATA(inode);
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900652 s = __seq_open_private(file, &bt_seq_ops,
653 sizeof(struct bt_seq_state));
Andrei Emeltchenko31f47072012-08-07 18:05:06 +0300654 if (!s)
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900655 return -ENOMEM;
656
657 s->l = sk_list;
658 return 0;
659}
660
Al Viro14805352013-04-04 19:12:06 -0400661static const struct file_operations bt_fops = {
662 .open = bt_seq_open,
663 .read = seq_read,
664 .llseek = seq_lseek,
665 .release = seq_release_private
666};
667
Al Virob0316612013-04-04 19:14:33 -0400668int bt_procfs_init(struct net *net, const char *name,
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900669 struct bt_sock_list* sk_list,
670 int (* seq_show)(struct seq_file *, void *))
671{
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900672 sk_list->custom_seq_show = seq_show;
673
Al Viro4d006262013-04-04 19:16:06 -0400674 if (!proc_create_data(name, 0, net->proc_net, &bt_fops, sk_list))
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900675 return -ENOMEM;
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900676 return 0;
677}
678
679void bt_procfs_cleanup(struct net *net, const char *name)
680{
Gao fengece31ff2013-02-18 01:34:56 +0000681 remove_proc_entry(name, net->proc_net);
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900682}
683#else
Al Virob0316612013-04-04 19:14:33 -0400684int bt_procfs_init(struct net *net, const char *name,
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900685 struct bt_sock_list* sk_list,
686 int (* seq_show)(struct seq_file *, void *))
687{
688 return 0;
689}
690
691void bt_procfs_cleanup(struct net *net, const char *name)
692{
693}
694#endif
695EXPORT_SYMBOL(bt_procfs_init);
696EXPORT_SYMBOL(bt_procfs_cleanup);
697
Linus Torvalds1da177e2005-04-16 15:20:36 -0700698static struct net_proto_family bt_sock_family_ops = {
699 .owner = THIS_MODULE,
700 .family = PF_BLUETOOTH,
701 .create = bt_sock_create,
702};
703
Linus Torvalds1da177e2005-04-16 15:20:36 -0700704static int __init bt_init(void)
705{
Marcel Holtmann27d35282006-07-03 10:02:37 +0200706 int err;
707
Linus Torvalds1da177e2005-04-16 15:20:36 -0700708 BT_INFO("Core ver %s", VERSION);
709
Marcel Holtmann27d35282006-07-03 10:02:37 +0200710 err = bt_sysfs_init();
711 if (err < 0)
712 return err;
713
714 err = sock_register(&bt_sock_family_ops);
715 if (err < 0) {
716 bt_sysfs_cleanup();
717 return err;
718 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700719
720 BT_INFO("HCI device and connection manager initialized");
721
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200722 err = hci_sock_init();
723 if (err < 0)
724 goto error;
725
726 err = l2cap_init();
Anand Gadiyar0ed54da2011-02-22 12:43:26 +0530727 if (err < 0)
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200728 goto sock_err;
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200729
730 err = sco_init();
731 if (err < 0) {
732 l2cap_exit();
733 goto sock_err;
734 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700735
736 return 0;
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200737
738sock_err:
739 hci_sock_cleanup();
740
741error:
742 sock_unregister(PF_BLUETOOTH);
743 bt_sysfs_cleanup();
744
745 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700746}
747
748static void __exit bt_exit(void)
749{
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200750
751 sco_exit();
752
753 l2cap_exit();
754
Linus Torvalds1da177e2005-04-16 15:20:36 -0700755 hci_sock_cleanup();
756
Linus Torvalds1da177e2005-04-16 15:20:36 -0700757 sock_unregister(PF_BLUETOOTH);
Marcel Holtmann27d35282006-07-03 10:02:37 +0200758
759 bt_sysfs_cleanup();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700760}
761
762subsys_initcall(bt_init);
763module_exit(bt_exit);
764
Marcel Holtmann63fbd242008-08-18 13:23:53 +0200765MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700766MODULE_DESCRIPTION("Bluetooth Core ver " VERSION);
767MODULE_VERSION(VERSION);
768MODULE_LICENSE("GPL");
769MODULE_ALIAS_NETPROTO(PF_BLUETOOTH);