mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
caif: Bugfix - use MSG_TRUNC in receive
Fixed handling when skb don't fit in user buffer, instead of returning -EMSGSIZE, the buffer is truncated (just as unix seqpakcet does). Signed-off-by: Sjur Braendeland <sjur.brandeland@stericsson.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
a9a8f1070d
commit
dcda138d2f
1 changed files with 19 additions and 30 deletions
|
@ -292,53 +292,42 @@ static void caif_check_flow_release(struct sock *sk)
|
||||||
caif_flow_ctrl(sk, CAIF_MODEMCMD_FLOW_ON_REQ);
|
caif_flow_ctrl(sk, CAIF_MODEMCMD_FLOW_ON_REQ);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/*
|
|
||||||
* Copied from sock.c:sock_queue_rcv_skb(), and added check that user buffer
|
|
||||||
* has sufficient size.
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copied from unix_dgram_recvmsg, but removed credit checks,
|
||||||
|
* changed locking, address handling and added MSG_TRUNC.
|
||||||
|
*/
|
||||||
static int caif_seqpkt_recvmsg(struct kiocb *iocb, struct socket *sock,
|
static int caif_seqpkt_recvmsg(struct kiocb *iocb, struct socket *sock,
|
||||||
struct msghdr *m, size_t buf_len, int flags)
|
struct msghdr *m, size_t len, int flags)
|
||||||
|
|
||||||
{
|
{
|
||||||
struct sock *sk = sock->sk;
|
struct sock *sk = sock->sk;
|
||||||
struct sk_buff *skb;
|
struct sk_buff *skb;
|
||||||
int ret = 0;
|
int ret;
|
||||||
int len;
|
int copylen;
|
||||||
|
|
||||||
if (unlikely(!buf_len))
|
ret = -EOPNOTSUPP;
|
||||||
return -EINVAL;
|
if (m->msg_flags&MSG_OOB)
|
||||||
|
goto read_error;
|
||||||
|
|
||||||
skb = skb_recv_datagram(sk, flags, 0 , &ret);
|
skb = skb_recv_datagram(sk, flags, 0 , &ret);
|
||||||
if (!skb)
|
if (!skb)
|
||||||
goto read_error;
|
goto read_error;
|
||||||
|
copylen = skb->len;
|
||||||
len = skb->len;
|
if (len < copylen) {
|
||||||
|
m->msg_flags |= MSG_TRUNC;
|
||||||
if (skb && skb->len > buf_len && !(flags & MSG_PEEK)) {
|
copylen = len;
|
||||||
len = buf_len;
|
|
||||||
/*
|
|
||||||
* Push skb back on receive queue if buffer too small.
|
|
||||||
* This has a built-in race where multi-threaded receive
|
|
||||||
* may get packet in wrong order, but multiple read does
|
|
||||||
* not really guarantee ordered delivery anyway.
|
|
||||||
* Let's optimize for speed without taking locks.
|
|
||||||
*/
|
|
||||||
|
|
||||||
skb_queue_head(&sk->sk_receive_queue, skb);
|
|
||||||
ret = -EMSGSIZE;
|
|
||||||
goto read_error;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = skb_copy_datagram_iovec(skb, 0, m->msg_iov, len);
|
ret = skb_copy_datagram_iovec(skb, 0, m->msg_iov, copylen);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto read_error;
|
goto out_free;
|
||||||
|
|
||||||
|
ret = (flags & MSG_TRUNC) ? skb->len : copylen;
|
||||||
|
out_free:
|
||||||
skb_free_datagram(sk, skb);
|
skb_free_datagram(sk, skb);
|
||||||
|
|
||||||
caif_check_flow_release(sk);
|
caif_check_flow_release(sk);
|
||||||
|
return ret;
|
||||||
return len;
|
|
||||||
|
|
||||||
read_error:
|
read_error:
|
||||||
return ret;
|
return ret;
|
||||||
|
|
Loading…
Reference in a new issue