⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 l2cap.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 4 页
字号:
/*   BlueZ - Bluetooth protocol stack for Linux   Copyright (C) 2000-2001 Qualcomm Incorporated   Written 2000,2001 by Maxim Krasnyansky <maxk@qualcomm.com>   This program is free software; you can redistribute it and/or modify   it under the terms of the GNU General Public License version 2 as   published by the Free Software Foundation;   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS   OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT OF THIRD PARTY RIGHTS.   IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) AND AUTHOR(S) BE LIABLE FOR ANY   CLAIM, OR ANY SPECIAL INDIRECT OR CONSEQUENTIAL DAMAGES, OR ANY DAMAGES   WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN   ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF   OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.   ALL LIABILITY, INCLUDING LIABILITY FOR INFRINGEMENT OF ANY PATENTS,   COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS, RELATING TO USE OF THIS   SOFTWARE IS DISCLAIMED.*//* Bluetooth L2CAP core and sockets. */#include <linux/module.h>#include <linux/types.h>#include <linux/capability.h>#include <linux/errno.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/slab.h>#include <linux/poll.h>#include <linux/fcntl.h>#include <linux/init.h>#include <linux/interrupt.h>#include <linux/socket.h>#include <linux/skbuff.h>#include <linux/list.h>#include <linux/device.h>#include <net/sock.h>#include <asm/system.h>#include <asm/uaccess.h>#include <asm/unaligned.h>#include <net/bluetooth/bluetooth.h>#include <net/bluetooth/hci_core.h>#include <net/bluetooth/l2cap.h>#ifndef CONFIG_BT_L2CAP_DEBUG#undef  BT_DBG#define BT_DBG(D...)#endif#define VERSION "2.9"static u32 l2cap_feat_mask = 0x0000;static const struct proto_ops l2cap_sock_ops;static struct bt_sock_list l2cap_sk_list = {	.lock = RW_LOCK_UNLOCKED};static void __l2cap_sock_close(struct sock *sk, int reason);static void l2cap_sock_close(struct sock *sk);static void l2cap_sock_kill(struct sock *sk);static struct sk_buff *l2cap_build_cmd(struct l2cap_conn *conn,				u8 code, u8 ident, u16 dlen, void *data);/* ---- L2CAP timers ---- */static void l2cap_sock_timeout(unsigned long arg){	struct sock *sk = (struct sock *) arg;	BT_DBG("sock %p state %d", sk, sk->sk_state);	bh_lock_sock(sk);	__l2cap_sock_close(sk, ETIMEDOUT);	bh_unlock_sock(sk);	l2cap_sock_kill(sk);	sock_put(sk);}static void l2cap_sock_set_timer(struct sock *sk, long timeout){	BT_DBG("sk %p state %d timeout %ld", sk, sk->sk_state, timeout);	sk_reset_timer(sk, &sk->sk_timer, jiffies + timeout);}static void l2cap_sock_clear_timer(struct sock *sk){	BT_DBG("sock %p state %d", sk, sk->sk_state);	sk_stop_timer(sk, &sk->sk_timer);}static void l2cap_sock_init_timer(struct sock *sk){	init_timer(&sk->sk_timer);	sk->sk_timer.function = l2cap_sock_timeout;	sk->sk_timer.data = (unsigned long)sk;}/* ---- L2CAP channels ---- */static struct sock *__l2cap_get_chan_by_dcid(struct l2cap_chan_list *l, u16 cid){	struct sock *s;	for (s = l->head; s; s = l2cap_pi(s)->next_c) {		if (l2cap_pi(s)->dcid == cid)			break;	}	return s;}static struct sock *__l2cap_get_chan_by_scid(struct l2cap_chan_list *l, u16 cid){	struct sock *s;	for (s = l->head; s; s = l2cap_pi(s)->next_c) {		if (l2cap_pi(s)->scid == cid)			break;	}	return s;}/* Find channel with given SCID. * Returns locked socket */static inline struct sock *l2cap_get_chan_by_scid(struct l2cap_chan_list *l, u16 cid){	struct sock *s;	read_lock(&l->lock);	s = __l2cap_get_chan_by_scid(l, cid);	if (s) bh_lock_sock(s);	read_unlock(&l->lock);	return s;}static struct sock *__l2cap_get_chan_by_ident(struct l2cap_chan_list *l, u8 ident){	struct sock *s;	for (s = l->head; s; s = l2cap_pi(s)->next_c) {		if (l2cap_pi(s)->ident == ident)			break;	}	return s;}static inline struct sock *l2cap_get_chan_by_ident(struct l2cap_chan_list *l, u8 ident){	struct sock *s;	read_lock(&l->lock);	s = __l2cap_get_chan_by_ident(l, ident);	if (s) bh_lock_sock(s);	read_unlock(&l->lock);	return s;}static u16 l2cap_alloc_cid(struct l2cap_chan_list *l){	u16 cid = 0x0040;	for (; cid < 0xffff; cid++) {		if(!__l2cap_get_chan_by_scid(l, cid))			return cid;	}	return 0;}static inline void __l2cap_chan_link(struct l2cap_chan_list *l, struct sock *sk){	sock_hold(sk);	if (l->head)		l2cap_pi(l->head)->prev_c = sk;	l2cap_pi(sk)->next_c = l->head;	l2cap_pi(sk)->prev_c = NULL;	l->head = sk;}static inline void l2cap_chan_unlink(struct l2cap_chan_list *l, struct sock *sk){	struct sock *next = l2cap_pi(sk)->next_c, *prev = l2cap_pi(sk)->prev_c;	write_lock_bh(&l->lock);	if (sk == l->head)		l->head = next;	if (next)		l2cap_pi(next)->prev_c = prev;	if (prev)		l2cap_pi(prev)->next_c = next;	write_unlock_bh(&l->lock);	__sock_put(sk);}static void __l2cap_chan_add(struct l2cap_conn *conn, struct sock *sk, struct sock *parent){	struct l2cap_chan_list *l = &conn->chan_list;	BT_DBG("conn %p, psm 0x%2.2x, dcid 0x%4.4x", conn, l2cap_pi(sk)->psm, l2cap_pi(sk)->dcid);	l2cap_pi(sk)->conn = conn;	if (sk->sk_type == SOCK_SEQPACKET) {		/* Alloc CID for connection-oriented socket */		l2cap_pi(sk)->scid = l2cap_alloc_cid(l);	} else if (sk->sk_type == SOCK_DGRAM) {		/* Connectionless socket */		l2cap_pi(sk)->scid = 0x0002;		l2cap_pi(sk)->dcid = 0x0002;		l2cap_pi(sk)->omtu = L2CAP_DEFAULT_MTU;	} else {		/* Raw socket can send/recv signalling messages only */		l2cap_pi(sk)->scid = 0x0001;		l2cap_pi(sk)->dcid = 0x0001;		l2cap_pi(sk)->omtu = L2CAP_DEFAULT_MTU;	}	__l2cap_chan_link(l, sk);	if (parent)		bt_accept_enqueue(parent, sk);}/* Delete channel. * Must be called on the locked socket. */static void l2cap_chan_del(struct sock *sk, int err){	struct l2cap_conn *conn = l2cap_pi(sk)->conn;	struct sock *parent = bt_sk(sk)->parent;	l2cap_sock_clear_timer(sk);	BT_DBG("sk %p, conn %p, err %d", sk, conn, err);	if (conn) {		/* Unlink from channel list */		l2cap_chan_unlink(&conn->chan_list, sk);		l2cap_pi(sk)->conn = NULL;		hci_conn_put(conn->hcon);	}	sk->sk_state  = BT_CLOSED;	sock_set_flag(sk, SOCK_ZAPPED);	if (err)		sk->sk_err = err;	if (parent) {		bt_accept_unlink(sk);		parent->sk_data_ready(parent, 0);	} else		sk->sk_state_change(sk);}static inline u8 l2cap_get_ident(struct l2cap_conn *conn){	u8 id;	/* Get next available identificator.	 *    1 - 128 are used by kernel.	 *  129 - 199 are reserved.	 *  200 - 254 are used by utilities like l2ping, etc.	 */	spin_lock_bh(&conn->lock);	if (++conn->tx_ident > 128)		conn->tx_ident = 1;	id = conn->tx_ident;	spin_unlock_bh(&conn->lock);	return id;}static inline int l2cap_send_cmd(struct l2cap_conn *conn, u8 ident, u8 code, u16 len, void *data){	struct sk_buff *skb = l2cap_build_cmd(conn, code, ident, len, data);	BT_DBG("code 0x%2.2x", code);	if (!skb)		return -ENOMEM;	return hci_send_acl(conn->hcon, skb, 0);}/* ---- L2CAP connections ---- */static void l2cap_conn_start(struct l2cap_conn *conn){	struct l2cap_chan_list *l = &conn->chan_list;	struct sock *sk;	BT_DBG("conn %p", conn);	read_lock(&l->lock);	for (sk = l->head; sk; sk = l2cap_pi(sk)->next_c) {		bh_lock_sock(sk);		if (sk->sk_type != SOCK_SEQPACKET) {			l2cap_sock_clear_timer(sk);			sk->sk_state = BT_CONNECTED;			sk->sk_state_change(sk);		} else if (sk->sk_state == BT_CONNECT) {			struct l2cap_conn_req req;			l2cap_pi(sk)->ident = l2cap_get_ident(conn);			req.scid = cpu_to_le16(l2cap_pi(sk)->scid);			req.psm  = l2cap_pi(sk)->psm;			l2cap_send_cmd(conn, l2cap_pi(sk)->ident,					L2CAP_CONN_REQ, sizeof(req), &req);		}		bh_unlock_sock(sk);	}	read_unlock(&l->lock);}static void l2cap_conn_ready(struct l2cap_conn *conn){	BT_DBG("conn %p", conn);	if (conn->chan_list.head || !hlist_empty(&l2cap_sk_list.head)) {		struct l2cap_info_req req;		req.type = cpu_to_le16(L2CAP_IT_FEAT_MASK);		conn->info_state |= L2CAP_INFO_FEAT_MASK_REQ_SENT;		conn->info_ident = l2cap_get_ident(conn);		mod_timer(&conn->info_timer,			jiffies + msecs_to_jiffies(L2CAP_INFO_TIMEOUT));		l2cap_send_cmd(conn, conn->info_ident,					L2CAP_INFO_REQ, sizeof(req), &req);	}}/* Notify sockets that we cannot guaranty reliability anymore */static void l2cap_conn_unreliable(struct l2cap_conn *conn, int err){	struct l2cap_chan_list *l = &conn->chan_list;	struct sock *sk;	BT_DBG("conn %p", conn);	read_lock(&l->lock);	for (sk = l->head; sk; sk = l2cap_pi(sk)->next_c) {		if (l2cap_pi(sk)->link_mode & L2CAP_LM_RELIABLE)			sk->sk_err = err;	}	read_unlock(&l->lock);}static void l2cap_info_timeout(unsigned long arg){	struct l2cap_conn *conn = (void *) arg;	conn->info_ident = 0;	l2cap_conn_start(conn);}static struct l2cap_conn *l2cap_conn_add(struct hci_conn *hcon, u8 status){	struct l2cap_conn *conn = hcon->l2cap_data;	if (conn || status)		return conn;	conn = kzalloc(sizeof(struct l2cap_conn), GFP_ATOMIC);	if (!conn)		return NULL;	hcon->l2cap_data = conn;	conn->hcon = hcon;	BT_DBG("hcon %p conn %p", hcon, conn);	conn->mtu = hcon->hdev->acl_mtu;	conn->src = &hcon->hdev->bdaddr;	conn->dst = &hcon->dst;	conn->feat_mask = 0;	init_timer(&conn->info_timer);	conn->info_timer.function = l2cap_info_timeout;	conn->info_timer.data = (unsigned long) conn;	spin_lock_init(&conn->lock);	rwlock_init(&conn->chan_list.lock);	return conn;}static void l2cap_conn_del(struct hci_conn *hcon, int err){	struct l2cap_conn *conn = hcon->l2cap_data;	struct sock *sk;	if (!conn)		return;	BT_DBG("hcon %p conn %p, err %d", hcon, conn, err);	if (conn->rx_skb)		kfree_skb(conn->rx_skb);	/* Kill channels */	while ((sk = conn->chan_list.head)) {		bh_lock_sock(sk);		l2cap_chan_del(sk, err);		bh_unlock_sock(sk);		l2cap_sock_kill(sk);	}	hcon->l2cap_data = NULL;	kfree(conn);}static inline void l2cap_chan_add(struct l2cap_conn *conn, struct sock *sk, struct sock *parent){	struct l2cap_chan_list *l = &conn->chan_list;	write_lock_bh(&l->lock);	__l2cap_chan_add(conn, sk, parent);	write_unlock_bh(&l->lock);}/* ---- Socket interface ---- */static struct sock *__l2cap_get_sock_by_addr(__le16 psm, bdaddr_t *src){	struct sock *sk;	struct hlist_node *node;	sk_for_each(sk, node, &l2cap_sk_list.head)		if (l2cap_pi(sk)->sport == psm && !bacmp(&bt_sk(sk)->src, src))			goto found;	sk = NULL;found:	return sk;}/* Find socket with psm and source bdaddr. * Returns closest match. */static struct sock *__l2cap_get_sock_by_psm(int state, __le16 psm, bdaddr_t *src){	struct sock *sk = NULL, *sk1 = NULL;	struct hlist_node *node;	sk_for_each(sk, node, &l2cap_sk_list.head) {		if (state && sk->sk_state != state)			continue;		if (l2cap_pi(sk)->psm == psm) {			/* Exact match. */			if (!bacmp(&bt_sk(sk)->src, src))				break;			/* Closest match */			if (!bacmp(&bt_sk(sk)->src, BDADDR_ANY))				sk1 = sk;		}	}	return node ? sk : sk1;}/* Find socket with given address (psm, src). * Returns locked socket */static inline struct sock *l2cap_get_sock_by_psm(int state, __le16 psm, bdaddr_t *src){	struct sock *s;	read_lock(&l2cap_sk_list.lock);	s = __l2cap_get_sock_by_psm(state, psm, src);	if (s) bh_lock_sock(s);	read_unlock(&l2cap_sk_list.lock);	return s;}static void l2cap_sock_destruct(struct sock *sk){	BT_DBG("sk %p", sk);	skb_queue_purge(&sk->sk_receive_queue);	skb_queue_purge(&sk->sk_write_queue);}static void l2cap_sock_cleanup_listen(struct sock *parent){	struct sock *sk;	BT_DBG("parent %p", parent);	/* Close not yet accepted channels */	while ((sk = bt_accept_dequeue(parent, NULL)))		l2cap_sock_close(sk);	parent->sk_state  = BT_CLOSED;	sock_set_flag(parent, SOCK_ZAPPED);}/* Kill socket (only if zapped and orphan) * Must be called on unlocked socket. */static void l2cap_sock_kill(struct sock *sk){	if (!sock_flag(sk, SOCK_ZAPPED) || sk->sk_socket)		return;	BT_DBG("sk %p state %d", sk, sk->sk_state);	/* Kill poor orphan */	bt_sock_unlink(&l2cap_sk_list, sk);	sock_set_flag(sk, SOCK_DEAD);	sock_put(sk);}static void __l2cap_sock_close(struct sock *sk, int reason){	BT_DBG("sk %p state %d socket %p", sk, sk->sk_state, sk->sk_socket);	switch (sk->sk_state) {	case BT_LISTEN:		l2cap_sock_cleanup_listen(sk);		break;	case BT_CONNECTED:	case BT_CONFIG:	case BT_CONNECT2:		if (sk->sk_type == SOCK_SEQPACKET) {			struct l2cap_conn *conn = l2cap_pi(sk)->conn;			struct l2cap_disconn_req req;			sk->sk_state = BT_DISCONN;			l2cap_sock_set_timer(sk, sk->sk_sndtimeo);			req.dcid = cpu_to_le16(l2cap_pi(sk)->dcid);			req.scid = cpu_to_le16(l2cap_pi(sk)->scid);			l2cap_send_cmd(conn, l2cap_get_ident(conn),					L2CAP_DISCONN_REQ, sizeof(req), &req);		} else {			l2cap_chan_del(sk, reason);		}		break;	case BT_CONNECT:	case BT_DISCONN:		l2cap_chan_del(sk, reason);		break;	default:		sock_set_flag(sk, SOCK_ZAPPED);		break;	}}/* Must be called on unlocked socket. */static void l2cap_sock_close(struct sock *sk){	l2cap_sock_clear_timer(sk);	lock_sock(sk);	__l2cap_sock_close(sk, ECONNRESET);	release_sock(sk);	l2cap_sock_kill(sk);}static void l2cap_sock_init(struct sock *sk, struct sock *parent){	struct l2cap_pinfo *pi = l2cap_pi(sk);	BT_DBG("sk %p", sk);	if (parent) {		sk->sk_type = parent->sk_type;		pi->imtu = l2cap_pi(parent)->imtu;		pi->omtu = l2cap_pi(parent)->omtu;		pi->link_mode = l2cap_pi(parent)->link_mode;	} else {		pi->imtu = L2CAP_DEFAULT_MTU;		pi->omtu = 0;		pi->link_mode = 0;	}	/* Default config options */	pi->conf_len = 0;	pi->flush_to = L2CAP_DEFAULT_FLUSH_TO;}static struct proto l2cap_proto = {

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -