diff options
Diffstat (limited to 'net/bluetooth/rfcomm')
-rw-r--r-- | net/bluetooth/rfcomm/core.c | 54 | ||||
-rw-r--r-- | net/bluetooth/rfcomm/sock.c | 46 |
2 files changed, 75 insertions, 25 deletions
diff --git a/net/bluetooth/rfcomm/core.c b/net/bluetooth/rfcomm/core.c index 25692bc0a342..7dca91bb8c57 100644 --- a/net/bluetooth/rfcomm/core.c +++ b/net/bluetooth/rfcomm/core.c | |||
@@ -33,9 +33,12 @@ | |||
33 | #include <linux/init.h> | 33 | #include <linux/init.h> |
34 | #include <linux/wait.h> | 34 | #include <linux/wait.h> |
35 | #include <linux/device.h> | 35 | #include <linux/device.h> |
36 | #include <linux/debugfs.h> | ||
37 | #include <linux/seq_file.h> | ||
36 | #include <linux/net.h> | 38 | #include <linux/net.h> |
37 | #include <linux/mutex.h> | 39 | #include <linux/mutex.h> |
38 | #include <linux/kthread.h> | 40 | #include <linux/kthread.h> |
41 | #include <linux/slab.h> | ||
39 | 42 | ||
40 | #include <net/sock.h> | 43 | #include <net/sock.h> |
41 | #include <asm/uaccess.h> | 44 | #include <asm/uaccess.h> |
@@ -51,6 +54,7 @@ | |||
51 | static int disable_cfc = 0; | 54 | static int disable_cfc = 0; |
52 | static int channel_mtu = -1; | 55 | static int channel_mtu = -1; |
53 | static unsigned int l2cap_mtu = RFCOMM_MAX_L2CAP_MTU; | 56 | static unsigned int l2cap_mtu = RFCOMM_MAX_L2CAP_MTU; |
57 | static int l2cap_ertm = 0; | ||
54 | 58 | ||
55 | static struct task_struct *rfcomm_thread; | 59 | static struct task_struct *rfcomm_thread; |
56 | 60 | ||
@@ -251,7 +255,6 @@ static void rfcomm_session_timeout(unsigned long arg) | |||
251 | BT_DBG("session %p state %ld", s, s->state); | 255 | BT_DBG("session %p state %ld", s, s->state); |
252 | 256 | ||
253 | set_bit(RFCOMM_TIMED_OUT, &s->flags); | 257 | set_bit(RFCOMM_TIMED_OUT, &s->flags); |
254 | rfcomm_session_put(s); | ||
255 | rfcomm_schedule(RFCOMM_SCHED_TIMEO); | 258 | rfcomm_schedule(RFCOMM_SCHED_TIMEO); |
256 | } | 259 | } |
257 | 260 | ||
@@ -702,6 +705,8 @@ static struct rfcomm_session *rfcomm_session_create(bdaddr_t *src, bdaddr_t *dst | |||
702 | sk = sock->sk; | 705 | sk = sock->sk; |
703 | lock_sock(sk); | 706 | lock_sock(sk); |
704 | l2cap_pi(sk)->imtu = l2cap_mtu; | 707 | l2cap_pi(sk)->imtu = l2cap_mtu; |
708 | if (l2cap_ertm) | ||
709 | l2cap_pi(sk)->mode = L2CAP_MODE_ERTM; | ||
705 | release_sock(sk); | 710 | release_sock(sk); |
706 | 711 | ||
707 | s = rfcomm_session_add(sock, BT_BOUND); | 712 | s = rfcomm_session_add(sock, BT_BOUND); |
@@ -1148,7 +1153,11 @@ static int rfcomm_recv_ua(struct rfcomm_session *s, u8 dlci) | |||
1148 | break; | 1153 | break; |
1149 | 1154 | ||
1150 | case BT_DISCONN: | 1155 | case BT_DISCONN: |
1151 | rfcomm_session_put(s); | 1156 | /* When socket is closed and we are not RFCOMM |
1157 | * initiator rfcomm_process_rx already calls | ||
1158 | * rfcomm_session_put() */ | ||
1159 | if (s->sock->sk->sk_state != BT_CLOSED) | ||
1160 | rfcomm_session_put(s); | ||
1152 | break; | 1161 | break; |
1153 | } | 1162 | } |
1154 | } | 1163 | } |
@@ -1917,6 +1926,7 @@ static inline void rfcomm_process_sessions(void) | |||
1917 | if (test_and_clear_bit(RFCOMM_TIMED_OUT, &s->flags)) { | 1926 | if (test_and_clear_bit(RFCOMM_TIMED_OUT, &s->flags)) { |
1918 | s->state = BT_DISCONN; | 1927 | s->state = BT_DISCONN; |
1919 | rfcomm_send_disc(s, 0); | 1928 | rfcomm_send_disc(s, 0); |
1929 | rfcomm_session_put(s); | ||
1920 | continue; | 1930 | continue; |
1921 | } | 1931 | } |
1922 | 1932 | ||
@@ -2091,11 +2101,10 @@ static struct hci_cb rfcomm_cb = { | |||
2091 | .security_cfm = rfcomm_security_cfm | 2101 | .security_cfm = rfcomm_security_cfm |
2092 | }; | 2102 | }; |
2093 | 2103 | ||
2094 | static ssize_t rfcomm_dlc_sysfs_show(struct class *dev, char *buf) | 2104 | static int rfcomm_dlc_debugfs_show(struct seq_file *f, void *x) |
2095 | { | 2105 | { |
2096 | struct rfcomm_session *s; | 2106 | struct rfcomm_session *s; |
2097 | struct list_head *pp, *p; | 2107 | struct list_head *pp, *p; |
2098 | char *str = buf; | ||
2099 | 2108 | ||
2100 | rfcomm_lock(); | 2109 | rfcomm_lock(); |
2101 | 2110 | ||
@@ -2105,18 +2114,32 @@ static ssize_t rfcomm_dlc_sysfs_show(struct class *dev, char *buf) | |||
2105 | struct sock *sk = s->sock->sk; | 2114 | struct sock *sk = s->sock->sk; |
2106 | struct rfcomm_dlc *d = list_entry(pp, struct rfcomm_dlc, list); | 2115 | struct rfcomm_dlc *d = list_entry(pp, struct rfcomm_dlc, list); |
2107 | 2116 | ||
2108 | str += sprintf(str, "%s %s %ld %d %d %d %d\n", | 2117 | seq_printf(f, "%s %s %ld %d %d %d %d\n", |
2109 | batostr(&bt_sk(sk)->src), batostr(&bt_sk(sk)->dst), | 2118 | batostr(&bt_sk(sk)->src), |
2110 | d->state, d->dlci, d->mtu, d->rx_credits, d->tx_credits); | 2119 | batostr(&bt_sk(sk)->dst), |
2120 | d->state, d->dlci, d->mtu, | ||
2121 | d->rx_credits, d->tx_credits); | ||
2111 | } | 2122 | } |
2112 | } | 2123 | } |
2113 | 2124 | ||
2114 | rfcomm_unlock(); | 2125 | rfcomm_unlock(); |
2115 | 2126 | ||
2116 | return (str - buf); | 2127 | return 0; |
2117 | } | 2128 | } |
2118 | 2129 | ||
2119 | static CLASS_ATTR(rfcomm_dlc, S_IRUGO, rfcomm_dlc_sysfs_show, NULL); | 2130 | static int rfcomm_dlc_debugfs_open(struct inode *inode, struct file *file) |
2131 | { | ||
2132 | return single_open(file, rfcomm_dlc_debugfs_show, inode->i_private); | ||
2133 | } | ||
2134 | |||
2135 | static const struct file_operations rfcomm_dlc_debugfs_fops = { | ||
2136 | .open = rfcomm_dlc_debugfs_open, | ||
2137 | .read = seq_read, | ||
2138 | .llseek = seq_lseek, | ||
2139 | .release = single_release, | ||
2140 | }; | ||
2141 | |||
2142 | static struct dentry *rfcomm_dlc_debugfs; | ||
2120 | 2143 | ||
2121 | /* ---- Initialization ---- */ | 2144 | /* ---- Initialization ---- */ |
2122 | static int __init rfcomm_init(void) | 2145 | static int __init rfcomm_init(void) |
@@ -2133,8 +2156,12 @@ static int __init rfcomm_init(void) | |||
2133 | goto unregister; | 2156 | goto unregister; |
2134 | } | 2157 | } |
2135 | 2158 | ||
2136 | if (class_create_file(bt_class, &class_attr_rfcomm_dlc) < 0) | 2159 | if (bt_debugfs) { |
2137 | BT_ERR("Failed to create RFCOMM info file"); | 2160 | rfcomm_dlc_debugfs = debugfs_create_file("rfcomm_dlc", 0444, |
2161 | bt_debugfs, NULL, &rfcomm_dlc_debugfs_fops); | ||
2162 | if (!rfcomm_dlc_debugfs) | ||
2163 | BT_ERR("Failed to create RFCOMM debug file"); | ||
2164 | } | ||
2138 | 2165 | ||
2139 | err = rfcomm_init_ttys(); | 2166 | err = rfcomm_init_ttys(); |
2140 | if (err < 0) | 2167 | if (err < 0) |
@@ -2162,7 +2189,7 @@ unregister: | |||
2162 | 2189 | ||
2163 | static void __exit rfcomm_exit(void) | 2190 | static void __exit rfcomm_exit(void) |
2164 | { | 2191 | { |
2165 | class_remove_file(bt_class, &class_attr_rfcomm_dlc); | 2192 | debugfs_remove(rfcomm_dlc_debugfs); |
2166 | 2193 | ||
2167 | hci_unregister_cb(&rfcomm_cb); | 2194 | hci_unregister_cb(&rfcomm_cb); |
2168 | 2195 | ||
@@ -2185,6 +2212,9 @@ MODULE_PARM_DESC(channel_mtu, "Default MTU for the RFCOMM channel"); | |||
2185 | module_param(l2cap_mtu, uint, 0644); | 2212 | module_param(l2cap_mtu, uint, 0644); |
2186 | MODULE_PARM_DESC(l2cap_mtu, "Default MTU for the L2CAP connection"); | 2213 | MODULE_PARM_DESC(l2cap_mtu, "Default MTU for the L2CAP connection"); |
2187 | 2214 | ||
2215 | module_param(l2cap_ertm, bool, 0644); | ||
2216 | MODULE_PARM_DESC(l2cap_ertm, "Use L2CAP ERTM mode for connection"); | ||
2217 | |||
2188 | MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>"); | 2218 | MODULE_AUTHOR("Marcel Holtmann <marcel@holtmann.org>"); |
2189 | MODULE_DESCRIPTION("Bluetooth RFCOMM ver " VERSION); | 2219 | MODULE_DESCRIPTION("Bluetooth RFCOMM ver " VERSION); |
2190 | MODULE_VERSION(VERSION); | 2220 | MODULE_VERSION(VERSION); |
diff --git a/net/bluetooth/rfcomm/sock.c b/net/bluetooth/rfcomm/sock.c index 8a20aaf1f231..8ed3c37684fa 100644 --- a/net/bluetooth/rfcomm/sock.c +++ b/net/bluetooth/rfcomm/sock.c | |||
@@ -40,6 +40,8 @@ | |||
40 | #include <linux/skbuff.h> | 40 | #include <linux/skbuff.h> |
41 | #include <linux/list.h> | 41 | #include <linux/list.h> |
42 | #include <linux/device.h> | 42 | #include <linux/device.h> |
43 | #include <linux/debugfs.h> | ||
44 | #include <linux/seq_file.h> | ||
43 | #include <net/sock.h> | 45 | #include <net/sock.h> |
44 | 46 | ||
45 | #include <asm/system.h> | 47 | #include <asm/system.h> |
@@ -323,7 +325,8 @@ static struct sock *rfcomm_sock_alloc(struct net *net, struct socket *sock, int | |||
323 | return sk; | 325 | return sk; |
324 | } | 326 | } |
325 | 327 | ||
326 | static int rfcomm_sock_create(struct net *net, struct socket *sock, int protocol) | 328 | static int rfcomm_sock_create(struct net *net, struct socket *sock, |
329 | int protocol, int kern) | ||
327 | { | 330 | { |
328 | struct sock *sk; | 331 | struct sock *sk; |
329 | 332 | ||
@@ -394,7 +397,8 @@ static int rfcomm_sock_connect(struct socket *sock, struct sockaddr *addr, int a | |||
394 | 397 | ||
395 | BT_DBG("sk %p", sk); | 398 | BT_DBG("sk %p", sk); |
396 | 399 | ||
397 | if (addr->sa_family != AF_BLUETOOTH || alen < sizeof(struct sockaddr_rc)) | 400 | if (alen < sizeof(struct sockaddr_rc) || |
401 | addr->sa_family != AF_BLUETOOTH) | ||
398 | return -EINVAL; | 402 | return -EINVAL; |
399 | 403 | ||
400 | lock_sock(sk); | 404 | lock_sock(sk); |
@@ -703,7 +707,7 @@ static int rfcomm_sock_recvmsg(struct kiocb *iocb, struct socket *sock, | |||
703 | copied += chunk; | 707 | copied += chunk; |
704 | size -= chunk; | 708 | size -= chunk; |
705 | 709 | ||
706 | sock_recv_timestamp(msg, sk, skb); | 710 | sock_recv_ts_and_drops(msg, sk, skb); |
707 | 711 | ||
708 | if (!(flags & MSG_PEEK)) { | 712 | if (!(flags & MSG_PEEK)) { |
709 | atomic_sub(chunk, &sk->sk_rmem_alloc); | 713 | atomic_sub(chunk, &sk->sk_rmem_alloc); |
@@ -1060,26 +1064,38 @@ done: | |||
1060 | return result; | 1064 | return result; |
1061 | } | 1065 | } |
1062 | 1066 | ||
1063 | static ssize_t rfcomm_sock_sysfs_show(struct class *dev, char *buf) | 1067 | static int rfcomm_sock_debugfs_show(struct seq_file *f, void *p) |
1064 | { | 1068 | { |
1065 | struct sock *sk; | 1069 | struct sock *sk; |
1066 | struct hlist_node *node; | 1070 | struct hlist_node *node; |
1067 | char *str = buf; | ||
1068 | 1071 | ||
1069 | read_lock_bh(&rfcomm_sk_list.lock); | 1072 | read_lock_bh(&rfcomm_sk_list.lock); |
1070 | 1073 | ||
1071 | sk_for_each(sk, node, &rfcomm_sk_list.head) { | 1074 | sk_for_each(sk, node, &rfcomm_sk_list.head) { |
1072 | str += sprintf(str, "%s %s %d %d\n", | 1075 | seq_printf(f, "%s %s %d %d\n", |
1073 | batostr(&bt_sk(sk)->src), batostr(&bt_sk(sk)->dst), | 1076 | batostr(&bt_sk(sk)->src), |
1077 | batostr(&bt_sk(sk)->dst), | ||
1074 | sk->sk_state, rfcomm_pi(sk)->channel); | 1078 | sk->sk_state, rfcomm_pi(sk)->channel); |
1075 | } | 1079 | } |
1076 | 1080 | ||
1077 | read_unlock_bh(&rfcomm_sk_list.lock); | 1081 | read_unlock_bh(&rfcomm_sk_list.lock); |
1078 | 1082 | ||
1079 | return (str - buf); | 1083 | return 0; |
1080 | } | 1084 | } |
1081 | 1085 | ||
1082 | static CLASS_ATTR(rfcomm, S_IRUGO, rfcomm_sock_sysfs_show, NULL); | 1086 | static int rfcomm_sock_debugfs_open(struct inode *inode, struct file *file) |
1087 | { | ||
1088 | return single_open(file, rfcomm_sock_debugfs_show, inode->i_private); | ||
1089 | } | ||
1090 | |||
1091 | static const struct file_operations rfcomm_sock_debugfs_fops = { | ||
1092 | .open = rfcomm_sock_debugfs_open, | ||
1093 | .read = seq_read, | ||
1094 | .llseek = seq_lseek, | ||
1095 | .release = single_release, | ||
1096 | }; | ||
1097 | |||
1098 | static struct dentry *rfcomm_sock_debugfs; | ||
1083 | 1099 | ||
1084 | static const struct proto_ops rfcomm_sock_ops = { | 1100 | static const struct proto_ops rfcomm_sock_ops = { |
1085 | .family = PF_BLUETOOTH, | 1101 | .family = PF_BLUETOOTH, |
@@ -1101,7 +1117,7 @@ static const struct proto_ops rfcomm_sock_ops = { | |||
1101 | .mmap = sock_no_mmap | 1117 | .mmap = sock_no_mmap |
1102 | }; | 1118 | }; |
1103 | 1119 | ||
1104 | static struct net_proto_family rfcomm_sock_family_ops = { | 1120 | static const struct net_proto_family rfcomm_sock_family_ops = { |
1105 | .family = PF_BLUETOOTH, | 1121 | .family = PF_BLUETOOTH, |
1106 | .owner = THIS_MODULE, | 1122 | .owner = THIS_MODULE, |
1107 | .create = rfcomm_sock_create | 1123 | .create = rfcomm_sock_create |
@@ -1119,8 +1135,12 @@ int __init rfcomm_init_sockets(void) | |||
1119 | if (err < 0) | 1135 | if (err < 0) |
1120 | goto error; | 1136 | goto error; |
1121 | 1137 | ||
1122 | if (class_create_file(bt_class, &class_attr_rfcomm) < 0) | 1138 | if (bt_debugfs) { |
1123 | BT_ERR("Failed to create RFCOMM info file"); | 1139 | rfcomm_sock_debugfs = debugfs_create_file("rfcomm", 0444, |
1140 | bt_debugfs, NULL, &rfcomm_sock_debugfs_fops); | ||
1141 | if (!rfcomm_sock_debugfs) | ||
1142 | BT_ERR("Failed to create RFCOMM debug file"); | ||
1143 | } | ||
1124 | 1144 | ||
1125 | BT_INFO("RFCOMM socket layer initialized"); | 1145 | BT_INFO("RFCOMM socket layer initialized"); |
1126 | 1146 | ||
@@ -1134,7 +1154,7 @@ error: | |||
1134 | 1154 | ||
1135 | void rfcomm_cleanup_sockets(void) | 1155 | void rfcomm_cleanup_sockets(void) |
1136 | { | 1156 | { |
1137 | class_remove_file(bt_class, &class_attr_rfcomm); | 1157 | debugfs_remove(rfcomm_sock_debugfs); |
1138 | 1158 | ||
1139 | if (bt_sock_unregister(BTPROTO_RFCOMM) < 0) | 1159 | if (bt_sock_unregister(BTPROTO_RFCOMM) < 0) |
1140 | BT_ERR("RFCOMM socket layer unregistration failed"); | 1160 | BT_ERR("RFCOMM socket layer unregistration failed"); |