blob: 6629cdc134dc1a439a044027d6826bd6dd4b56f3 [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
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +0200224 skb = skb_recv_datagram(sk, flags, noblock, &err);
225 if (!skb) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700226 if (sk->sk_shutdown & RCV_SHUTDOWN)
227 return 0;
228 return err;
229 }
230
Linus Torvalds1da177e2005-04-16 15:20:36 -0700231 copied = skb->len;
232 if (len < copied) {
233 msg->msg_flags |= MSG_TRUNC;
234 copied = len;
235 }
236
Arnaldo Carvalho de Melobadff6d2007-03-13 13:06:52 -0300237 skb_reset_transport_header(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700238 err = skb_copy_datagram_iovec(skb, 0, msg->msg_iov, copied);
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200239 if (err == 0)
Neil Horman3b885782009-10-12 13:26:31 -0700240 sock_recv_ts_and_drops(msg, sk, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700241
242 skb_free_datagram(sk, skb);
243
244 return err ? : copied;
245}
246EXPORT_SYMBOL(bt_sock_recvmsg);
247
Mat Martineau796c86e2010-09-08 10:05:27 -0700248static long bt_sock_data_wait(struct sock *sk, long timeo)
249{
250 DECLARE_WAITQUEUE(wait, current);
251
252 add_wait_queue(sk_sleep(sk), &wait);
253 for (;;) {
254 set_current_state(TASK_INTERRUPTIBLE);
255
256 if (!skb_queue_empty(&sk->sk_receive_queue))
257 break;
258
259 if (sk->sk_err || (sk->sk_shutdown & RCV_SHUTDOWN))
260 break;
261
262 if (signal_pending(current) || !timeo)
263 break;
264
265 set_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
266 release_sock(sk);
267 timeo = schedule_timeout(timeo);
268 lock_sock(sk);
269 clear_bit(SOCK_ASYNC_WAITDATA, &sk->sk_socket->flags);
270 }
271
272 __set_current_state(TASK_RUNNING);
273 remove_wait_queue(sk_sleep(sk), &wait);
274 return timeo;
275}
276
277int bt_sock_stream_recvmsg(struct kiocb *iocb, struct socket *sock,
278 struct msghdr *msg, size_t size, int flags)
279{
280 struct sock *sk = sock->sk;
281 int err = 0;
282 size_t target, copied = 0;
283 long timeo;
284
285 if (flags & MSG_OOB)
286 return -EOPNOTSUPP;
287
Mat Martineau796c86e2010-09-08 10:05:27 -0700288 BT_DBG("sk %p size %zu", sk, size);
289
290 lock_sock(sk);
291
292 target = sock_rcvlowat(sk, flags & MSG_WAITALL, size);
293 timeo = sock_rcvtimeo(sk, flags & MSG_DONTWAIT);
294
295 do {
296 struct sk_buff *skb;
297 int chunk;
298
299 skb = skb_dequeue(&sk->sk_receive_queue);
300 if (!skb) {
301 if (copied >= target)
302 break;
303
Andrei Emeltchenko5a08ecc2011-01-11 17:20:20 +0200304 err = sock_error(sk);
305 if (err)
Mat Martineau796c86e2010-09-08 10:05:27 -0700306 break;
307 if (sk->sk_shutdown & RCV_SHUTDOWN)
308 break;
309
310 err = -EAGAIN;
311 if (!timeo)
312 break;
313
314 timeo = bt_sock_data_wait(sk, timeo);
315
316 if (signal_pending(current)) {
317 err = sock_intr_errno(timeo);
318 goto out;
319 }
320 continue;
321 }
322
323 chunk = min_t(unsigned int, skb->len, size);
Mat Martineau5b668eb2011-07-22 14:53:59 -0700324 if (skb_copy_datagram_iovec(skb, 0, msg->msg_iov, chunk)) {
Mat Martineau796c86e2010-09-08 10:05:27 -0700325 skb_queue_head(&sk->sk_receive_queue, skb);
326 if (!copied)
327 copied = -EFAULT;
328 break;
329 }
330 copied += chunk;
331 size -= chunk;
332
333 sock_recv_ts_and_drops(msg, sk, skb);
334
335 if (!(flags & MSG_PEEK)) {
Mat Martineau5b668eb2011-07-22 14:53:59 -0700336 int skb_len = skb_headlen(skb);
337
338 if (chunk <= skb_len) {
339 __skb_pull(skb, chunk);
340 } else {
341 struct sk_buff *frag;
342
343 __skb_pull(skb, skb_len);
344 chunk -= skb_len;
345
346 skb_walk_frags(skb, frag) {
347 if (chunk <= frag->len) {
348 /* Pulling partial data */
349 skb->len -= chunk;
350 skb->data_len -= chunk;
351 __skb_pull(frag, chunk);
352 break;
353 } else if (frag->len) {
354 /* Pulling all frag data */
355 chunk -= frag->len;
356 skb->len -= frag->len;
357 skb->data_len -= frag->len;
358 __skb_pull(frag, frag->len);
359 }
360 }
361 }
362
Mat Martineau796c86e2010-09-08 10:05:27 -0700363 if (skb->len) {
364 skb_queue_head(&sk->sk_receive_queue, skb);
365 break;
366 }
367 kfree_skb(skb);
368
369 } else {
370 /* put message back and return */
371 skb_queue_head(&sk->sk_receive_queue, skb);
372 break;
373 }
374 } while (size);
375
376out:
377 release_sock(sk);
378 return copied ? : err;
379}
380EXPORT_SYMBOL(bt_sock_stream_recvmsg);
381
Linus Torvalds1da177e2005-04-16 15:20:36 -0700382static inline unsigned int bt_accept_poll(struct sock *parent)
383{
384 struct list_head *p, *n;
385 struct sock *sk;
386
387 list_for_each_safe(p, n, &bt_sk(parent)->accept_q) {
388 sk = (struct sock *) list_entry(p, struct bt_sock, accept_q);
Marcel Holtmannd5f2d2b2009-02-16 02:57:30 +0100389 if (sk->sk_state == BT_CONNECTED ||
Gustavo Padovanc5daa682012-05-16 12:17:10 -0300390 (test_bit(BT_SK_DEFER_SETUP, &bt_sk(parent)->flags) &&
391 sk->sk_state == BT_CONNECT2))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392 return POLLIN | POLLRDNORM;
393 }
394
395 return 0;
396}
397
Gustavo Padovan8fc9ced2012-05-23 04:04:21 -0300398unsigned int bt_sock_poll(struct file *file, struct socket *sock,
399 poll_table *wait)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700400{
401 struct sock *sk = sock->sk;
402 unsigned int mask = 0;
403
404 BT_DBG("sock %p, sk %p", sock, sk);
405
Eric Dumazetaa395142010-04-20 13:03:51 +0000406 poll_wait(file, sk_sleep(sk), wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700407
408 if (sk->sk_state == BT_LISTEN)
409 return bt_accept_poll(sk);
410
411 if (sk->sk_err || !skb_queue_empty(&sk->sk_error_queue))
Keller, Jacob E7d4c04f2013-03-28 11:19:25 +0000412 mask |= POLLERR |
Jacob Keller8facd5f2013-04-02 13:55:40 -0700413 (sock_flag(sk, SOCK_SELECT_ERR_QUEUE) ? POLLPRI : 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700414
Davide Libenzif348d702006-03-25 03:07:39 -0800415 if (sk->sk_shutdown & RCV_SHUTDOWN)
Eric Dumazetdb409802010-09-06 11:13:50 +0000416 mask |= POLLRDHUP | POLLIN | POLLRDNORM;
Davide Libenzif348d702006-03-25 03:07:39 -0800417
Linus Torvalds1da177e2005-04-16 15:20:36 -0700418 if (sk->sk_shutdown == SHUTDOWN_MASK)
419 mask |= POLLHUP;
420
Eric Dumazetdb409802010-09-06 11:13:50 +0000421 if (!skb_queue_empty(&sk->sk_receive_queue))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700422 mask |= POLLIN | POLLRDNORM;
423
424 if (sk->sk_state == BT_CLOSED)
425 mask |= POLLHUP;
426
427 if (sk->sk_state == BT_CONNECT ||
428 sk->sk_state == BT_CONNECT2 ||
429 sk->sk_state == BT_CONFIG)
430 return mask;
431
Gustavo Padovanc5daa682012-05-16 12:17:10 -0300432 if (!test_bit(BT_SK_SUSPEND, &bt_sk(sk)->flags) && sock_writeable(sk))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700433 mask |= POLLOUT | POLLWRNORM | POLLWRBAND;
434 else
435 set_bit(SOCK_ASYNC_NOSPACE, &sk->sk_socket->flags);
436
437 return mask;
438}
439EXPORT_SYMBOL(bt_sock_poll);
440
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200441int bt_sock_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)
442{
443 struct sock *sk = sock->sk;
Marcel Holtmann43cbeee2008-07-14 20:13:51 +0200444 struct sk_buff *skb;
445 long amount;
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200446 int err;
447
448 BT_DBG("sk %p cmd %x arg %lx", sk, cmd, arg);
449
450 switch (cmd) {
Marcel Holtmann43cbeee2008-07-14 20:13:51 +0200451 case TIOCOUTQ:
452 if (sk->sk_state == BT_LISTEN)
453 return -EINVAL;
454
Eric Dumazet31e6d362009-06-17 19:05:41 -0700455 amount = sk->sk_sndbuf - sk_wmem_alloc_get(sk);
Marcel Holtmann43cbeee2008-07-14 20:13:51 +0200456 if (amount < 0)
457 amount = 0;
458 err = put_user(amount, (int __user *) arg);
459 break;
460
461 case TIOCINQ:
462 if (sk->sk_state == BT_LISTEN)
463 return -EINVAL;
464
465 lock_sock(sk);
466 skb = skb_peek(&sk->sk_receive_queue);
467 amount = skb ? skb->len : 0;
468 release_sock(sk);
469 err = put_user(amount, (int __user *) arg);
470 break;
471
Marcel Holtmann3241ad82008-07-14 20:13:50 +0200472 case SIOCGSTAMP:
473 err = sock_get_timestamp(sk, (struct timeval __user *) arg);
474 break;
475
476 case SIOCGSTAMPNS:
477 err = sock_get_timestampns(sk, (struct timespec __user *) arg);
478 break;
479
480 default:
481 err = -ENOIOCTLCMD;
482 break;
483 }
484
485 return err;
486}
487EXPORT_SYMBOL(bt_sock_ioctl);
488
Linus Torvalds1da177e2005-04-16 15:20:36 -0700489int bt_sock_wait_state(struct sock *sk, int state, unsigned long timeo)
490{
491 DECLARE_WAITQUEUE(wait, current);
492 int err = 0;
493
494 BT_DBG("sk %p", sk);
495
Eric Dumazetaa395142010-04-20 13:03:51 +0000496 add_wait_queue(sk_sleep(sk), &wait);
Peter Hurley9be4e3f2011-07-24 00:10:46 -0400497 set_current_state(TASK_INTERRUPTIBLE);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700498 while (sk->sk_state != state) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700499 if (!timeo) {
Marcel Holtmannb4c612a2006-09-23 09:54:38 +0200500 err = -EINPROGRESS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700501 break;
502 }
503
504 if (signal_pending(current)) {
505 err = sock_intr_errno(timeo);
506 break;
507 }
508
509 release_sock(sk);
510 timeo = schedule_timeout(timeo);
511 lock_sock(sk);
Peter Hurley9be4e3f2011-07-24 00:10:46 -0400512 set_current_state(TASK_INTERRUPTIBLE);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700513
Benjamin LaHaisec1cbe4b2005-12-13 23:22:19 -0800514 err = sock_error(sk);
515 if (err)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700516 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700517 }
Peter Hurley9be4e3f2011-07-24 00:10:46 -0400518 __set_current_state(TASK_RUNNING);
Eric Dumazetaa395142010-04-20 13:03:51 +0000519 remove_wait_queue(sk_sleep(sk), &wait);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700520 return err;
521}
522EXPORT_SYMBOL(bt_sock_wait_state);
523
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900524#ifdef CONFIG_PROC_FS
525struct bt_seq_state {
526 struct bt_sock_list *l;
527};
528
529static void *bt_seq_start(struct seq_file *seq, loff_t *pos)
530 __acquires(seq->private->l->lock)
531{
532 struct bt_seq_state *s = seq->private;
533 struct bt_sock_list *l = s->l;
534
535 read_lock(&l->lock);
536 return seq_hlist_start_head(&l->head, *pos);
537}
538
539static void *bt_seq_next(struct seq_file *seq, void *v, loff_t *pos)
540{
541 struct bt_seq_state *s = seq->private;
542 struct bt_sock_list *l = s->l;
543
544 return seq_hlist_next(v, &l->head, pos);
545}
546
547static void bt_seq_stop(struct seq_file *seq, void *v)
548 __releases(seq->private->l->lock)
549{
550 struct bt_seq_state *s = seq->private;
551 struct bt_sock_list *l = s->l;
552
553 read_unlock(&l->lock);
554}
555
556static int bt_seq_show(struct seq_file *seq, void *v)
557{
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900558 struct bt_seq_state *s = seq->private;
559 struct bt_sock_list *l = s->l;
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900560
561 if (v == SEQ_START_TOKEN) {
562 seq_puts(seq ,"sk RefCnt Rmem Wmem User Inode Src Dst Parent");
563
564 if (l->custom_seq_show) {
565 seq_putc(seq, ' ');
566 l->custom_seq_show(seq, v);
567 }
568
569 seq_putc(seq, '\n');
570 } else {
Andrei Emeltchenko09d5d4a2012-08-07 18:05:04 +0300571 struct sock *sk = sk_entry(v);
572 struct bt_sock *bt = bt_sk(sk);
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900573
Andrei Emeltchenko7028a882012-09-25 12:49:45 +0300574 seq_printf(seq,
575 "%pK %-6d %-6u %-6u %-6u %-6lu %pMR %pMR %-6lu",
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900576 sk,
577 atomic_read(&sk->sk_refcnt),
578 sk_rmem_alloc_get(sk),
579 sk_wmem_alloc_get(sk),
Eric W. Biederman1bbb3092012-10-03 20:32:17 -0700580 from_kuid(seq_user_ns(seq), sock_i_uid(sk)),
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900581 sock_i_ino(sk),
Andrei Emeltchenko7028a882012-09-25 12:49:45 +0300582 &bt->src,
583 &bt->dst,
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900584 bt->parent? sock_i_ino(bt->parent): 0LU);
585
586 if (l->custom_seq_show) {
587 seq_putc(seq, ' ');
588 l->custom_seq_show(seq, v);
589 }
590
591 seq_putc(seq, '\n');
592 }
593 return 0;
594}
595
596static struct seq_operations bt_seq_ops = {
597 .start = bt_seq_start,
598 .next = bt_seq_next,
599 .stop = bt_seq_stop,
600 .show = bt_seq_show,
601};
602
603static int bt_seq_open(struct inode *inode, struct file *file)
604{
605 struct bt_sock_list *sk_list;
606 struct bt_seq_state *s;
607
Al Virod9dda782013-03-31 18:16:14 -0400608 sk_list = PDE_DATA(inode);
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900609 s = __seq_open_private(file, &bt_seq_ops,
610 sizeof(struct bt_seq_state));
Andrei Emeltchenko31f47072012-08-07 18:05:06 +0300611 if (!s)
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900612 return -ENOMEM;
613
614 s->l = sk_list;
615 return 0;
616}
617
Al Viro14805352013-04-04 19:12:06 -0400618static const struct file_operations bt_fops = {
619 .open = bt_seq_open,
620 .read = seq_read,
621 .llseek = seq_lseek,
622 .release = seq_release_private
623};
624
Al Virob0316612013-04-04 19:14:33 -0400625int bt_procfs_init(struct net *net, const char *name,
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900626 struct bt_sock_list* sk_list,
627 int (* seq_show)(struct seq_file *, void *))
628{
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900629 sk_list->custom_seq_show = seq_show;
630
Al Viro4d0062632013-04-04 19:16:06 -0400631 if (!proc_create_data(name, 0, net->proc_net, &bt_fops, sk_list))
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900632 return -ENOMEM;
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900633 return 0;
634}
635
636void bt_procfs_cleanup(struct net *net, const char *name)
637{
Gao fengece31ff2013-02-18 01:34:56 +0000638 remove_proc_entry(name, net->proc_net);
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900639}
640#else
Al Virob0316612013-04-04 19:14:33 -0400641int bt_procfs_init(struct net *net, const char *name,
Masatake YAMATO256a06c2012-07-26 01:26:32 +0900642 struct bt_sock_list* sk_list,
643 int (* seq_show)(struct seq_file *, void *))
644{
645 return 0;
646}
647
648void bt_procfs_cleanup(struct net *net, const char *name)
649{
650}
651#endif
652EXPORT_SYMBOL(bt_procfs_init);
653EXPORT_SYMBOL(bt_procfs_cleanup);
654
Linus Torvalds1da177e2005-04-16 15:20:36 -0700655static struct net_proto_family bt_sock_family_ops = {
656 .owner = THIS_MODULE,
657 .family = PF_BLUETOOTH,
658 .create = bt_sock_create,
659};
660
Linus Torvalds1da177e2005-04-16 15:20:36 -0700661static int __init bt_init(void)
662{
Marcel Holtmann27d35282006-07-03 10:02:37 +0200663 int err;
664
Linus Torvalds1da177e2005-04-16 15:20:36 -0700665 BT_INFO("Core ver %s", VERSION);
666
Marcel Holtmann27d35282006-07-03 10:02:37 +0200667 err = bt_sysfs_init();
668 if (err < 0)
669 return err;
670
671 err = sock_register(&bt_sock_family_ops);
672 if (err < 0) {
673 bt_sysfs_cleanup();
674 return err;
675 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700676
677 BT_INFO("HCI device and connection manager initialized");
678
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200679 err = hci_sock_init();
680 if (err < 0)
681 goto error;
682
683 err = l2cap_init();
Anand Gadiyar0ed54da2011-02-22 12:43:26 +0530684 if (err < 0)
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200685 goto sock_err;
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200686
687 err = sco_init();
688 if (err < 0) {
689 l2cap_exit();
690 goto sock_err;
691 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700692
693 return 0;
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200694
695sock_err:
696 hci_sock_cleanup();
697
698error:
699 sock_unregister(PF_BLUETOOTH);
700 bt_sysfs_cleanup();
701
702 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700703}
704
705static void __exit bt_exit(void)
706{
Gustavo F. Padovan64274512011-02-07 20:08:52 -0200707
708 sco_exit();
709
710 l2cap_exit();
711
Linus Torvalds1da177e2005-04-16 15:20:36 -0700712 hci_sock_cleanup();
713
Linus Torvalds1da177e2005-04-16 15:20:36 -0700714 sock_unregister(PF_BLUETOOTH);
Marcel Holtmann27d35282006-07-03 10:02:37 +0200715
716 bt_sysfs_cleanup();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700717}
718
719subsys_initcall(bt_init);
720module_exit(bt_exit);
721
Marcel Holtmann63fbd242008-08-18 13:23:53 +0200722MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700723MODULE_DESCRIPTION("Bluetooth Core ver " VERSION);
724MODULE_VERSION(VERSION);
725MODULE_LICENSE("GPL");
726MODULE_ALIAS_NETPROTO(PF_BLUETOOTH);