core.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:15k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /* 
  2.    BNEP implementation for Linux Bluetooth stack (BlueZ).
  3.    Copyright (C) 2001-2002 Inventel Systemes
  4.    Written 2001-2002 by
  5. Cl閙ent Moreau <clement.moreau@inventel.fr>
  6. David Libault  <david.libault@inventel.fr>
  7.    Copyright (C) 2002 Maxim Krasnyanskiy <maxk@qualcomm.com>
  8.    This program is free software; you can redistribute it and/or modify
  9.    it under the terms of the GNU General Public License version 2 as
  10.    published by the Free Software Foundation;
  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
  15.    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 
  18.    OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  19.    ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS, 
  20.    COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS 
  21.    SOFTWARE IS DISCLAIMED.
  22. */
  23. /*
  24.  * $Id: core.c,v 1.18 2002/07/14 07:09:19 maxk Exp $
  25.  */ 
  26. #define __KERNEL_SYSCALLS__
  27. #include <linux/config.h>
  28. #include <linux/module.h>
  29. #include <linux/kernel.h>
  30. #include <linux/sched.h>
  31. #include <linux/signal.h>
  32. #include <linux/init.h>
  33. #include <linux/wait.h>
  34. #include <linux/errno.h>
  35. #include <linux/smp_lock.h>
  36. #include <linux/net.h>
  37. #include <net/sock.h>
  38. #include <linux/socket.h>
  39. #include <linux/file.h>
  40. #include <linux/netdevice.h>
  41. #include <linux/etherdevice.h>
  42. #include <linux/skbuff.h>
  43. #include <asm/unaligned.h>
  44. #include <net/bluetooth/bluetooth.h>
  45. #include <net/bluetooth/l2cap.h>
  46. #include "bnep.h"
  47. #ifndef CONFIG_BNEP_DEBUG
  48. #undef  BT_DBG
  49. #define BT_DBG(D...)
  50. #endif
  51. #define VERSION "1.0"
  52. static LIST_HEAD(bnep_session_list);
  53. static DECLARE_RWSEM(bnep_session_sem);
  54. static struct bnep_session *__bnep_get_session(__u8 *dst)
  55. {
  56. struct bnep_session *s;
  57. struct list_head *p;
  58. BT_DBG("");
  59. list_for_each(p, &bnep_session_list) {
  60. s = list_entry(p, struct bnep_session, list);
  61. if (!memcmp(dst, s->eh.h_source, ETH_ALEN))
  62. return s;
  63. }
  64. return NULL;
  65. }
  66. static void __bnep_link_session(struct bnep_session *s)
  67. {
  68. MOD_INC_USE_COUNT;
  69. list_add(&s->list, &bnep_session_list);
  70. }
  71. static void __bnep_unlink_session(struct bnep_session *s)
  72. {
  73. list_del(&s->list);
  74. MOD_DEC_USE_COUNT;
  75. }
  76. static int bnep_send(struct bnep_session *s, void *data, size_t len)
  77. {
  78. struct socket *sock = s->sock;
  79. struct iovec iv = { data, len };
  80. s->msg.msg_iov    = &iv;
  81. s->msg.msg_iovlen = 1;
  82. return sock->ops->sendmsg(sock, &s->msg, len, NULL);
  83. }
  84. static int bnep_send_rsp(struct bnep_session *s, __u8 ctrl, __u16 resp)
  85. {
  86. struct bnep_control_rsp rsp;
  87. rsp.type = BNEP_CONTROL;
  88. rsp.ctrl = ctrl;
  89. rsp.resp = htons(resp);
  90. return bnep_send(s, &rsp, sizeof(rsp));
  91. }
  92. static int bnep_ctrl_set_netfilter(struct bnep_session *s, struct sk_buff *skb)
  93. {
  94. __u16 *data;
  95. int n;
  96. data = (void *) skb->data;
  97. if (!skb_pull(skb, 2))
  98. return -EILSEQ;
  99. n = ntohs(get_unaligned(data));
  100. data = (void *) skb->data;
  101. if (!skb_pull(skb, n))
  102. return -EILSEQ;
  103. BT_DBG("filter len %d", n);
  104. #ifdef CONFIG_BNEP_PROTO_FILTER
  105. n /= 4;
  106. if (n <= BNEP_MAX_PROTO_FILTERS) {
  107. struct bnep_proto_filter *f = s->proto_filter;
  108. int i;
  109. for (i = 0; i < n; i++) {
  110. f[i].start = get_unaligned(data++);
  111. f[i].end   = get_unaligned(data++);
  112. BT_DBG("proto filter start %d end %d",
  113. f[i].start, f[i].end);
  114. }
  115. if (i < BNEP_MAX_PROTO_FILTERS)
  116. memset(f + i, 0, sizeof(*f));
  117. bnep_send_rsp(s, BNEP_FILTER_NET_TYPE_RSP, BNEP_SUCCESS);
  118. } else {
  119. bnep_send_rsp(s, BNEP_FILTER_NET_TYPE_RSP, BNEP_FILTER_LIMIT_REACHED);
  120. }
  121. #else
  122. bnep_send_rsp(s, BNEP_FILTER_NET_TYPE_RSP, BNEP_FILTER_UNSUPPORTED_REQ);
  123. #endif
  124. return 0;
  125. }
  126. static int bnep_ctrl_set_mcfilter(struct bnep_session *s, struct sk_buff *skb)
  127. {
  128. u8 *data;
  129. int n;
  130. data = (void *) skb->data;
  131. if (!skb_pull(skb, 2))
  132. return -EILSEQ;
  133. n = ntohs(get_unaligned((u16 *) data));
  134. data = (void *) skb->data;
  135. if (!skb_pull(skb, n))
  136. return -EILSEQ;
  137. BT_DBG("filter len %d", n);
  138. #ifdef CONFIG_BNEP_MC_FILTER
  139. n /= (ETH_ALEN * 2);
  140. if (n > 0) {
  141. s->mc_filter = 0;
  142. /* Always send broadcast */
  143. set_bit(bnep_mc_hash(s->dev.broadcast), &s->mc_filter);
  144. /* Add address ranges to the multicast hash */
  145. for (; n > 0; n--) {
  146. u8 a1[6], *a2;
  147. memcpy(a1, data, ETH_ALEN); data += ETH_ALEN;
  148. a2 = data; data += ETH_ALEN;
  149. BT_DBG("mc filter %s -> %s",
  150. batostr((void *) a1), batostr((void *) a2));
  151. #define INCA(a) { int i = 5; while (i >=0 && ++a[i--] == 0); }
  152. /* Iterate from a1 to a2 */
  153. set_bit(bnep_mc_hash(a1), &s->mc_filter);
  154. while (memcmp(a1, a2, 6) < 0 && s->mc_filter != ~0LL) {
  155. INCA(a1);
  156. set_bit(bnep_mc_hash(a1), &s->mc_filter);
  157. }
  158. }
  159. }
  160. BT_DBG("mc filter hash 0x%llx", s->mc_filter);
  161. bnep_send_rsp(s, BNEP_FILTER_MULTI_ADDR_RSP, BNEP_SUCCESS);
  162. #else
  163. bnep_send_rsp(s, BNEP_FILTER_MULTI_ADDR_RSP, BNEP_FILTER_UNSUPPORTED_REQ);
  164. #endif
  165. return 0;
  166. }
  167. static int bnep_rx_control(struct bnep_session *s, struct sk_buff *skb)
  168. {
  169. int err = 0;
  170. __u8 cmd = *(__u8 *) skb->data;
  171. skb_pull(skb, 1);
  172. switch (cmd) {
  173. case BNEP_CMD_NOT_UNDERSTOOD:
  174. case BNEP_SETUP_CONN_REQ:
  175. case BNEP_SETUP_CONN_RSP:
  176. case BNEP_FILTER_NET_TYPE_RSP:
  177. case BNEP_FILTER_MULTI_ADDR_RSP:
  178. /* Ignore these for now */
  179. break;
  180. case BNEP_FILTER_NET_TYPE_SET:
  181. err = bnep_ctrl_set_netfilter(s, skb);
  182. break;
  183. case BNEP_FILTER_MULTI_ADDR_SET:
  184. err = bnep_ctrl_set_mcfilter(s, skb);
  185. break;
  186. default: {
  187. __u8 pkt[3];
  188. pkt[0] = BNEP_CONTROL;
  189. pkt[1] = BNEP_CMD_NOT_UNDERSTOOD;
  190. pkt[2] = cmd;
  191. bnep_send(s, pkt, sizeof(pkt));
  192. }
  193. break;
  194. }
  195. return err;
  196. }
  197. static int bnep_rx_extension(struct bnep_session *s, struct sk_buff *skb)
  198. {
  199. struct bnep_ext_hdr *h;
  200. int err = 0;
  201. do {
  202. h = (void *) skb->data;
  203. if (!skb_pull(skb, sizeof(*h))) {
  204. err = -EILSEQ;
  205. break;
  206. }
  207. BT_DBG("type 0x%x len %d", h->type, h->len);
  208. switch (h->type & BNEP_TYPE_MASK) {
  209. case BNEP_EXT_CONTROL:
  210. err = bnep_rx_control(s, skb);
  211. break;
  212. default:
  213. /* Unknown extension */
  214. if (!skb_pull(skb, h->len))
  215. err = -EILSEQ;
  216. break;
  217. }
  218. } while (!err && (h->type & BNEP_EXT_HEADER));
  219. return err;
  220. }
  221. static __u8 __bnep_rx_hlen[] = {
  222. ETH_HLEN,     /* BNEP_GENERAL */
  223. 0,            /* BNEP_CONTROL */
  224. 2,            /* BNEP_COMPRESSED */
  225. ETH_ALEN + 2, /* BNEP_COMPRESSED_SRC_ONLY */
  226. ETH_ALEN + 2  /* BNEP_COMPRESSED_DST_ONLY */
  227. };
  228. #define BNEP_RX_TYPES (sizeof(__bnep_rx_hlen) - 1)
  229. static inline int bnep_rx_frame(struct bnep_session *s, struct sk_buff *skb)
  230. {
  231. struct net_device *dev = &s->dev;
  232. struct sk_buff *nskb;
  233. __u8 type;
  234. dev->last_rx = jiffies;
  235. s->stats.rx_bytes += skb->len;
  236. type = *(__u8 *) skb->data; skb_pull(skb, 1);
  237. if ((type & BNEP_TYPE_MASK) > BNEP_RX_TYPES)
  238. goto badframe;
  239. if ((type & BNEP_TYPE_MASK) == BNEP_CONTROL) {
  240. bnep_rx_control(s, skb);
  241. kfree_skb(skb);
  242. return 0;
  243. }
  244. skb->mac.raw = skb->data;
  245. /* Verify and pull out header */
  246. if (!skb_pull(skb, __bnep_rx_hlen[type & BNEP_TYPE_MASK]))
  247. goto badframe;
  248. s->eh.h_proto = get_unaligned((__u16 *) (skb->data - 2));
  249. if (type & BNEP_EXT_HEADER) {
  250. if (bnep_rx_extension(s, skb) < 0)
  251. goto badframe;
  252. }
  253. /* Strip 802.1p header */
  254. if (ntohs(s->eh.h_proto) == 0x8100) {
  255. if (!skb_pull(skb, 4))
  256. goto badframe;
  257. s->eh.h_proto = get_unaligned((__u16 *) (skb->data - 2));
  258. }
  259. /* We have to alloc new skb and copy data here :(. Because original skb
  260.  * may not be modified and because of the alignment requirements. */
  261. nskb = alloc_skb(2 + ETH_HLEN + skb->len, GFP_KERNEL);
  262. if (!nskb) {
  263. s->stats.rx_dropped++;
  264. kfree_skb(skb);
  265. return -ENOMEM;
  266. }
  267. skb_reserve(nskb, 2);
  268. /* Decompress header and construct ether frame */
  269. switch (type & BNEP_TYPE_MASK) {
  270. case BNEP_COMPRESSED:
  271. memcpy(__skb_put(nskb, ETH_HLEN), &s->eh, ETH_HLEN);
  272. break;
  273. case BNEP_COMPRESSED_SRC_ONLY:
  274. memcpy(__skb_put(nskb, ETH_ALEN), s->eh.h_dest, ETH_ALEN);
  275. memcpy(__skb_put(nskb, ETH_ALEN), skb->mac.raw, ETH_ALEN);
  276. put_unaligned(s->eh.h_proto, (__u16 *) __skb_put(nskb, 2));
  277. break;
  278. case BNEP_COMPRESSED_DST_ONLY:
  279. memcpy(__skb_put(nskb, ETH_ALEN), skb->mac.raw, ETH_ALEN);
  280. memcpy(__skb_put(nskb, ETH_ALEN + 2), s->eh.h_source, ETH_ALEN + 2);
  281. break;
  282. case BNEP_GENERAL:
  283. memcpy(__skb_put(nskb, ETH_ALEN * 2), skb->mac.raw, ETH_ALEN * 2);
  284. put_unaligned(s->eh.h_proto, (__u16 *) __skb_put(nskb, 2));
  285. break;
  286. }
  287. memcpy(__skb_put(nskb, skb->len), skb->data, skb->len);
  288. kfree_skb(skb);
  289. s->stats.rx_packets++;
  290. nskb->dev       = dev;
  291. nskb->ip_summed = CHECKSUM_UNNECESSARY;
  292. nskb->protocol  = eth_type_trans(nskb, dev);
  293. netif_rx_ni(nskb);
  294. return 0;
  295. badframe:
  296. s->stats.rx_errors++;
  297. kfree_skb(skb);
  298. return 0;
  299. }
  300. static __u8 __bnep_tx_types[] = {
  301. BNEP_GENERAL,
  302. BNEP_COMPRESSED_SRC_ONLY,
  303. BNEP_COMPRESSED_DST_ONLY,
  304. BNEP_COMPRESSED
  305. };
  306. static inline int bnep_tx_frame(struct bnep_session *s, struct sk_buff *skb)
  307. {
  308. struct ethhdr *eh = (void *) skb->data;
  309. struct socket *sock = s->sock;
  310. struct iovec iv[3];
  311. int len = 0, il = 0;
  312. __u8 type = 0;
  313. BT_DBG("skb %p dev %p type %d", skb, skb->dev, skb->pkt_type);
  314. if (!skb->dev) {
  315. /* Control frame sent by us */
  316. goto send;
  317. }
  318. iv[il++] = (struct iovec) { &type, 1 };
  319. len++;
  320. if (!memcmp(eh->h_dest, s->eh.h_source, ETH_ALEN))
  321. type |= 0x01;
  322. if (!memcmp(eh->h_source, s->eh.h_dest, ETH_ALEN))
  323. type |= 0x02;
  324. if (type)
  325. skb_pull(skb, ETH_ALEN * 2);
  326. type = __bnep_tx_types[type];
  327. switch (type) {
  328. case BNEP_COMPRESSED_SRC_ONLY:
  329. iv[il++] = (struct iovec) { eh->h_source, ETH_ALEN };
  330. len += ETH_ALEN;
  331. break;
  332. case BNEP_COMPRESSED_DST_ONLY:
  333. iv[il++] = (struct iovec) { eh->h_dest, ETH_ALEN };
  334. len += ETH_ALEN;
  335. break;
  336. }
  337. send:
  338. iv[il++] = (struct iovec) { skb->data, skb->len };
  339. len += skb->len;
  340. /* FIXME: linearize skb */
  341. s->msg.msg_iov    = iv;
  342. s->msg.msg_iovlen = il;
  343. len = sock->ops->sendmsg(sock, &s->msg, len, NULL);
  344. kfree_skb(skb);
  345. if (len > 0) {
  346. s->stats.tx_bytes += len;
  347. s->stats.tx_packets++;
  348. return 0;
  349. }
  350. return len;
  351. }
  352. static int bnep_session(void *arg)
  353. {
  354. struct bnep_session *s = arg;
  355. struct net_device *dev = &s->dev;
  356. struct sock *sk = s->sock->sk;
  357. struct sk_buff *skb;
  358. wait_queue_t wait;
  359. BT_DBG("");
  360.         daemonize(); reparent_to_init();
  361.         sprintf(current->comm, "kbnepd %s", dev->name);
  362.         sigfillset(&current->blocked);
  363. flush_signals(current);
  364. current->nice = -15;
  365.         set_fs(KERNEL_DS);
  366. init_waitqueue_entry(&wait, current);
  367. add_wait_queue(sk->sleep, &wait);
  368. while (!atomic_read(&s->killed)) {
  369. set_current_state(TASK_INTERRUPTIBLE);
  370. // RX
  371. while ((skb = skb_dequeue(&sk->receive_queue))) {
  372. skb_orphan(skb);
  373. bnep_rx_frame(s, skb);
  374. }
  375. if (sk->state != BT_CONNECTED)
  376. break;
  377. // TX
  378. while ((skb = skb_dequeue(&sk->write_queue)))
  379. if (bnep_tx_frame(s, skb))
  380. break;
  381. netif_wake_queue(dev);
  382. schedule();
  383. }
  384. set_current_state(TASK_RUNNING);
  385. remove_wait_queue(sk->sleep, &wait);
  386. /* Cleanup session */
  387. down_write(&bnep_session_sem);
  388. /* Delete network device */
  389. unregister_netdev(dev);
  390. /* Release the socket */
  391. fput(s->sock->file);
  392. __bnep_unlink_session(s);
  393. up_write(&bnep_session_sem);
  394. kfree(s);
  395. return 0;
  396. }
  397. int bnep_add_connection(struct bnep_conadd_req *req, struct socket *sock)
  398. {
  399. struct net_device *dev;
  400. struct bnep_session *s, *ss;
  401. __u8 dst[ETH_ALEN], src[ETH_ALEN];
  402. int err;
  403. BT_DBG("");
  404. baswap((void *) dst, &bluez_pi(sock->sk)->dst);
  405. baswap((void *) src, &bluez_pi(sock->sk)->src);
  406. s = kmalloc(sizeof(struct bnep_session), GFP_KERNEL);
  407. if (!s) 
  408. return -ENOMEM;
  409. memset(s, 0, sizeof(struct bnep_session));
  410. down_write(&bnep_session_sem);
  411. ss = __bnep_get_session(dst);
  412. if (ss && ss->state == BT_CONNECTED) {
  413. err = -EEXIST;
  414. goto failed;
  415. }
  416. dev = &s->dev;
  417. if (*req->device)
  418. strcpy(dev->name, req->device);
  419. else
  420. strcpy(dev->name, "bnep%d");
  421. /* This is rx header therefor addresses are swaped.
  422.  * ie eh.h_dest is our local address. */
  423. memcpy(s->eh.h_dest,   &src, ETH_ALEN);
  424. memcpy(s->eh.h_source, &dst, ETH_ALEN);
  425. s->sock  = sock;
  426. s->role  = req->role;
  427. s->state = BT_CONNECTED;
  428. s->msg.msg_flags = MSG_NOSIGNAL;
  429. #ifdef CONFIG_BNEP_MC_FILTER
  430. /* Set default mc filter */
  431. set_bit(bnep_mc_hash(dev->broadcast), &s->mc_filter);
  432. #endif
  433. #ifdef CONFIG_BNEP_PROTO_FILTER
  434. /* Set default protocol filter */
  435. /* (IPv4, ARP)  */
  436. s->proto_filter[0].start = htons(0x0800);
  437. s->proto_filter[0].end   = htons(0x0806);
  438. /* (RARP, AppleTalk) */
  439. s->proto_filter[1].start = htons(0x8035);
  440. s->proto_filter[1].end   = htons(0x80F3);
  441. /* (IPX, IPv6) */
  442. s->proto_filter[2].start = htons(0x8137);
  443. s->proto_filter[2].end   = htons(0x86DD);
  444. #endif
  445. dev->init = bnep_net_init;
  446. dev->priv = s;
  447. err = register_netdev(dev);
  448. if (err) {
  449. goto failed;
  450. }
  451. __bnep_link_session(s);
  452. err = kernel_thread(bnep_session, s, CLONE_FS | CLONE_FILES | CLONE_SIGHAND);
  453. if (err < 0) {
  454. /* Session thread start failed, gotta cleanup. */
  455. unregister_netdev(dev);
  456. __bnep_unlink_session(s);
  457. goto failed;
  458. }
  459. up_write(&bnep_session_sem);
  460. strcpy(req->device, dev->name);
  461. return 0;
  462. failed:
  463. up_write(&bnep_session_sem);
  464. kfree(s);
  465. return err;
  466. }
  467. int bnep_del_connection(struct bnep_condel_req *req)
  468. {
  469. struct bnep_session *s;
  470. int  err = 0;
  471. BT_DBG("");
  472. down_read(&bnep_session_sem);
  473. s = __bnep_get_session(req->dst);
  474. if (s) {
  475. /* Wakeup user-space which is polling for socket errors.
  476.  * This is temporary hack untill we have shutdown in L2CAP */
  477. s->sock->sk->err = EUNATCH;
  478. /* Kill session thread */
  479. atomic_inc(&s->killed);
  480. wake_up_interruptible(s->sock->sk->sleep);
  481. } else
  482. err = -ENOENT;
  483. up_read(&bnep_session_sem);
  484. return err;
  485. }
  486. static void __bnep_copy_ci(struct bnep_coninfo *ci, struct bnep_session *s)
  487. {
  488. memcpy(ci->dst, s->eh.h_source, ETH_ALEN);
  489. strcpy(ci->device, s->dev.name);
  490. ci->flags = s->flags;
  491. ci->state = s->state;
  492. ci->role  = s->role;
  493. }
  494. int bnep_get_conlist(struct bnep_conlist_req *req)
  495. {
  496. struct list_head *p;
  497. int err = 0, n = 0;
  498. down_read(&bnep_session_sem);
  499. list_for_each(p, &bnep_session_list) {
  500. struct bnep_session *s;
  501. struct bnep_coninfo ci;
  502. s = list_entry(p, struct bnep_session, list);
  503. __bnep_copy_ci(&ci, s);
  504. if (copy_to_user(req->ci, &ci, sizeof(ci))) {
  505. err = -EFAULT;
  506. break;
  507. }
  508. if (++n >= req->cnum)
  509. break;
  510. req->ci++;
  511. }
  512. req->cnum = n;
  513. up_read(&bnep_session_sem);
  514. return err;
  515. }
  516. int bnep_get_coninfo(struct bnep_coninfo *ci)
  517. {
  518. struct bnep_session *s;
  519. int err = 0;
  520. down_read(&bnep_session_sem);
  521. s = __bnep_get_session(ci->dst);
  522. if (s)
  523. __bnep_copy_ci(ci, s);
  524. else
  525. err = -ENOENT;
  526. up_read(&bnep_session_sem);
  527. return err;
  528. }
  529. static int __init bnep_init_module(void)
  530. {
  531. BT_INFO("BNEP: BNEP2 ver %sn"
  532. "BNEP: Copyright (C) 2002 Inventeln"
  533. "BNEP: Written 2001,2002 byn"
  534. "BNEP: tClement Moreau <clement.moreau@inventel.fr> "
  535. "David Libault <david.libault@inventel.fr>n"
  536. "BNEP: Copyright (C) 2002 Maxim Krasnyanskiy <maxk@qualcomm.com>",
  537. VERSION);
  538. bnep_sock_init();
  539. bnep_crc32_init();
  540. return 0;
  541. }
  542. static void __exit bnep_cleanup_module(void)
  543. {
  544. bnep_sock_cleanup();
  545. bnep_crc32_cleanup();
  546. }
  547. module_init(bnep_init_module);
  548. module_exit(bnep_cleanup_module);
  549. MODULE_DESCRIPTION("BNEP2 ver " VERSION);
  550. MODULE_AUTHOR("David Libault <david.libault@inventel.fr> Maxim Krasnyanskiy <maxk@qualcomm.com>");
  551. MODULE_LICENSE("GPL");