Commit 9b108fc0 authored by Gustavo F. Padovan's avatar Gustavo F. Padovan Committed by Marcel Holtmann

Bluetooth: Fix ERTM error reporting to the userspace

If any error occurs during transfers we have to tell userspace that
something wrong happened.
Signed-off-by: default avatarGustavo F. Padovan <padovan@profusion.mobi>
Reviewed-by: default avatarJoão Paulo Rechi Vita <jprvita@profusion.mobi>
Signed-off-by: default avatarMarcel Holtmann <marcel@holtmann.org>
parent 4ea727ef
...@@ -456,7 +456,7 @@ static void l2cap_do_start(struct sock *sk) ...@@ -456,7 +456,7 @@ static void l2cap_do_start(struct sock *sk)
} }
} }
static void l2cap_send_disconn_req(struct l2cap_conn *conn, struct sock *sk) static void l2cap_send_disconn_req(struct l2cap_conn *conn, struct sock *sk, int err)
{ {
struct l2cap_disconn_req req; struct l2cap_disconn_req req;
...@@ -477,6 +477,7 @@ static void l2cap_send_disconn_req(struct l2cap_conn *conn, struct sock *sk) ...@@ -477,6 +477,7 @@ static void l2cap_send_disconn_req(struct l2cap_conn *conn, struct sock *sk)
L2CAP_DISCONN_REQ, sizeof(req), &req); L2CAP_DISCONN_REQ, sizeof(req), &req);
sk->sk_state = BT_DISCONN; sk->sk_state = BT_DISCONN;
sk->sk_err = err;
} }
/* ---- L2CAP connections ---- */ /* ---- L2CAP connections ---- */
...@@ -770,7 +771,7 @@ static void __l2cap_sock_close(struct sock *sk, int reason) ...@@ -770,7 +771,7 @@ static void __l2cap_sock_close(struct sock *sk, int reason)
struct l2cap_conn *conn = l2cap_pi(sk)->conn; struct l2cap_conn *conn = l2cap_pi(sk)->conn;
l2cap_sock_set_timer(sk, sk->sk_sndtimeo); l2cap_sock_set_timer(sk, sk->sk_sndtimeo);
l2cap_send_disconn_req(conn, sk); l2cap_send_disconn_req(conn, sk, reason);
} else } else
l2cap_chan_del(sk, reason); l2cap_chan_del(sk, reason);
break; break;
...@@ -1315,7 +1316,7 @@ static void l2cap_monitor_timeout(unsigned long arg) ...@@ -1315,7 +1316,7 @@ static void l2cap_monitor_timeout(unsigned long arg)
bh_lock_sock(sk); bh_lock_sock(sk);
if (l2cap_pi(sk)->retry_count >= l2cap_pi(sk)->remote_max_tx) { if (l2cap_pi(sk)->retry_count >= l2cap_pi(sk)->remote_max_tx) {
l2cap_send_disconn_req(l2cap_pi(sk)->conn, sk); l2cap_send_disconn_req(l2cap_pi(sk)->conn, sk, ECONNABORTED);
bh_unlock_sock(sk); bh_unlock_sock(sk);
return; return;
} }
...@@ -1423,7 +1424,7 @@ static void l2cap_retransmit_one_frame(struct sock *sk, u8 tx_seq) ...@@ -1423,7 +1424,7 @@ static void l2cap_retransmit_one_frame(struct sock *sk, u8 tx_seq)
if (pi->remote_max_tx && if (pi->remote_max_tx &&
bt_cb(skb)->retries == pi->remote_max_tx) { bt_cb(skb)->retries == pi->remote_max_tx) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNABORTED);
return; return;
} }
...@@ -1463,7 +1464,7 @@ static int l2cap_ertm_send(struct sock *sk) ...@@ -1463,7 +1464,7 @@ static int l2cap_ertm_send(struct sock *sk)
if (pi->remote_max_tx && if (pi->remote_max_tx &&
bt_cb(skb)->retries == pi->remote_max_tx) { bt_cb(skb)->retries == pi->remote_max_tx) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNABORTED);
break; break;
} }
...@@ -2191,6 +2192,10 @@ static int l2cap_sock_shutdown(struct socket *sock, int how) ...@@ -2191,6 +2192,10 @@ static int l2cap_sock_shutdown(struct socket *sock, int how)
err = bt_sock_wait_state(sk, BT_CLOSED, err = bt_sock_wait_state(sk, BT_CLOSED,
sk->sk_lingertime); sk->sk_lingertime);
} }
if (!err && sk->sk_err)
err = -sk->sk_err;
release_sock(sk); release_sock(sk);
return err; return err;
} }
...@@ -2462,7 +2467,7 @@ static int l2cap_build_conf_req(struct sock *sk, void *data) ...@@ -2462,7 +2467,7 @@ static int l2cap_build_conf_req(struct sock *sk, void *data)
case L2CAP_MODE_ERTM: case L2CAP_MODE_ERTM:
pi->conf_state |= L2CAP_CONF_STATE2_DEVICE; pi->conf_state |= L2CAP_CONF_STATE2_DEVICE;
if (!l2cap_mode_supported(pi->mode, pi->conn->feat_mask)) if (!l2cap_mode_supported(pi->mode, pi->conn->feat_mask))
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
break; break;
default: default:
pi->mode = l2cap_select_mode(rfc.mode, pi->conn->feat_mask); pi->mode = l2cap_select_mode(rfc.mode, pi->conn->feat_mask);
...@@ -3030,7 +3035,7 @@ static inline int l2cap_config_req(struct l2cap_conn *conn, struct l2cap_cmd_hdr ...@@ -3030,7 +3035,7 @@ static inline int l2cap_config_req(struct l2cap_conn *conn, struct l2cap_cmd_hdr
/* Complete config. */ /* Complete config. */
len = l2cap_parse_conf_req(sk, rsp); len = l2cap_parse_conf_req(sk, rsp);
if (len < 0) { if (len < 0) {
l2cap_send_disconn_req(conn, sk); l2cap_send_disconn_req(conn, sk, ECONNRESET);
goto unlock; goto unlock;
} }
...@@ -3100,7 +3105,7 @@ static inline int l2cap_config_rsp(struct l2cap_conn *conn, struct l2cap_cmd_hdr ...@@ -3100,7 +3105,7 @@ static inline int l2cap_config_rsp(struct l2cap_conn *conn, struct l2cap_cmd_hdr
char req[64]; char req[64];
if (len > sizeof(req) - sizeof(struct l2cap_conf_req)) { if (len > sizeof(req) - sizeof(struct l2cap_conf_req)) {
l2cap_send_disconn_req(conn, sk); l2cap_send_disconn_req(conn, sk, ECONNRESET);
goto done; goto done;
} }
...@@ -3109,7 +3114,7 @@ static inline int l2cap_config_rsp(struct l2cap_conn *conn, struct l2cap_cmd_hdr ...@@ -3109,7 +3114,7 @@ static inline int l2cap_config_rsp(struct l2cap_conn *conn, struct l2cap_cmd_hdr
len = l2cap_parse_conf_rsp(sk, rsp->data, len = l2cap_parse_conf_rsp(sk, rsp->data,
len, req, &result); len, req, &result);
if (len < 0) { if (len < 0) {
l2cap_send_disconn_req(conn, sk); l2cap_send_disconn_req(conn, sk, ECONNRESET);
goto done; goto done;
} }
...@@ -3124,7 +3129,7 @@ static inline int l2cap_config_rsp(struct l2cap_conn *conn, struct l2cap_cmd_hdr ...@@ -3124,7 +3129,7 @@ static inline int l2cap_config_rsp(struct l2cap_conn *conn, struct l2cap_cmd_hdr
default: default:
sk->sk_err = ECONNRESET; sk->sk_err = ECONNRESET;
l2cap_sock_set_timer(sk, HZ * 5); l2cap_sock_set_timer(sk, HZ * 5);
l2cap_send_disconn_req(conn, sk); l2cap_send_disconn_req(conn, sk, ECONNRESET);
goto done; goto done;
} }
...@@ -3565,7 +3570,7 @@ static int l2cap_ertm_reassembly_sdu(struct sock *sk, struct sk_buff *skb, u16 c ...@@ -3565,7 +3570,7 @@ static int l2cap_ertm_reassembly_sdu(struct sock *sk, struct sk_buff *skb, u16 c
pi->sdu = NULL; pi->sdu = NULL;
disconnect: disconnect:
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
kfree_skb(skb); kfree_skb(skb);
return 0; return 0;
} }
...@@ -3588,7 +3593,7 @@ static void l2cap_busy_work(struct work_struct *work) ...@@ -3588,7 +3593,7 @@ static void l2cap_busy_work(struct work_struct *work)
if (n_tries++ > L2CAP_LOCAL_BUSY_TRIES) { if (n_tries++ > L2CAP_LOCAL_BUSY_TRIES) {
err = -EBUSY; err = -EBUSY;
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, EBUSY);
goto done; goto done;
} }
...@@ -3864,7 +3869,7 @@ static inline int l2cap_data_channel_iframe(struct sock *sk, u16 rx_control, str ...@@ -3864,7 +3869,7 @@ static inline int l2cap_data_channel_iframe(struct sock *sk, u16 rx_control, str
/* invalid tx_seq */ /* invalid tx_seq */
if (tx_seq_offset >= pi->tx_win) { if (tx_seq_offset >= pi->tx_win) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
goto drop; goto drop;
} }
...@@ -4181,7 +4186,7 @@ static inline int l2cap_data_channel(struct l2cap_conn *conn, u16 cid, struct sk ...@@ -4181,7 +4186,7 @@ static inline int l2cap_data_channel(struct l2cap_conn *conn, u16 cid, struct sk
len -= 2; len -= 2;
if (len > pi->mps) { if (len > pi->mps) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
goto drop; goto drop;
} }
...@@ -4197,20 +4202,20 @@ static inline int l2cap_data_channel(struct l2cap_conn *conn, u16 cid, struct sk ...@@ -4197,20 +4202,20 @@ static inline int l2cap_data_channel(struct l2cap_conn *conn, u16 cid, struct sk
/* check for invalid req-seq */ /* check for invalid req-seq */
if (req_seq_offset > next_tx_seq_offset) { if (req_seq_offset > next_tx_seq_offset) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
goto drop; goto drop;
} }
if (__is_iframe(control)) { if (__is_iframe(control)) {
if (len < 0) { if (len < 0) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
goto drop; goto drop;
} }
l2cap_data_channel_iframe(sk, control, skb); l2cap_data_channel_iframe(sk, control, skb);
} else { } else {
if (len != 0) { if (len != 0) {
l2cap_send_disconn_req(pi->conn, sk); l2cap_send_disconn_req(pi->conn, sk, ECONNRESET);
goto drop; goto drop;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment