diff options
author | Sjur Braendeland <sjur.brandeland@stericsson.com> | 2010-05-20 22:16:12 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2010-05-24 02:57:43 -0400 |
commit | dcda138d2f27e32bd0d6250cc42839b0d70bb4b8 (patch) | |
tree | 679ff3b9ef79f7fbb5821bff2d050fd0638d2366 /net/caif | |
parent | a9a8f1070d8733b37418b3a2d58df4e771b61f88 (diff) |
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>
Diffstat (limited to 'net/caif')
-rw-r--r-- | net/caif/caif_socket.c | 47 |
1 files changed, 18 insertions, 29 deletions
diff --git a/net/caif/caif_socket.c b/net/caif/caif_socket.c index 691a5710974e..3d0e09584fae 100644 --- a/net/caif/caif_socket.c +++ b/net/caif/caif_socket.c | |||
@@ -292,53 +292,42 @@ static void caif_check_flow_release(struct sock *sk) | |||
292 | caif_flow_ctrl(sk, CAIF_MODEMCMD_FLOW_ON_REQ); | 292 | caif_flow_ctrl(sk, CAIF_MODEMCMD_FLOW_ON_REQ); |
293 | } | 293 | } |
294 | } | 294 | } |
295 | |||
295 | /* | 296 | /* |
296 | * Copied from sock.c:sock_queue_rcv_skb(), and added check that user buffer | 297 | * Copied from unix_dgram_recvmsg, but removed credit checks, |
297 | * has sufficient size. | 298 | * changed locking, address handling and added MSG_TRUNC. |
298 | */ | 299 | */ |
299 | |||
300 | static int caif_seqpkt_recvmsg(struct kiocb *iocb, struct socket *sock, | 300 | static int caif_seqpkt_recvmsg(struct kiocb *iocb, struct socket *sock, |
301 | struct msghdr *m, size_t buf_len, int flags) | 301 | struct msghdr *m, size_t len, int flags) |
302 | 302 | ||
303 | { | 303 | { |
304 | struct sock *sk = sock->sk; | 304 | struct sock *sk = sock->sk; |
305 | struct sk_buff *skb; | 305 | struct sk_buff *skb; |
306 | int ret = 0; | 306 | int ret; |
307 | int len; | 307 | int copylen; |
308 | 308 | ||
309 | if (unlikely(!buf_len)) | 309 | ret = -EOPNOTSUPP; |
310 | return -EINVAL; | 310 | if (m->msg_flags&MSG_OOB) |
311 | goto read_error; | ||
311 | 312 | ||
312 | skb = skb_recv_datagram(sk, flags, 0 , &ret); | 313 | skb = skb_recv_datagram(sk, flags, 0 , &ret); |
313 | if (!skb) | 314 | if (!skb) |
314 | goto read_error; | 315 | goto read_error; |
315 | 316 | copylen = skb->len; | |
316 | len = skb->len; | 317 | if (len < copylen) { |
317 | 318 | m->msg_flags |= MSG_TRUNC; | |
318 | if (skb && skb->len > buf_len && !(flags & MSG_PEEK)) { | 319 | copylen = len; |
319 | len = buf_len; | ||
320 | /* | ||
321 | * Push skb back on receive queue if buffer too small. | ||
322 | * This has a built-in race where multi-threaded receive | ||
323 | * may get packet in wrong order, but multiple read does | ||
324 | * not really guarantee ordered delivery anyway. | ||
325 | * Let's optimize for speed without taking locks. | ||
326 | */ | ||
327 | |||
328 | skb_queue_head(&sk->sk_receive_queue, skb); | ||
329 | ret = -EMSGSIZE; | ||
330 | goto read_error; | ||
331 | } | 320 | } |
332 | 321 | ||
333 | ret = skb_copy_datagram_iovec(skb, 0, m->msg_iov, len); | 322 | ret = skb_copy_datagram_iovec(skb, 0, m->msg_iov, copylen); |
334 | if (ret) | 323 | if (ret) |
335 | goto read_error; | 324 | goto out_free; |
336 | 325 | ||
326 | ret = (flags & MSG_TRUNC) ? skb->len : copylen; | ||
327 | out_free: | ||
337 | skb_free_datagram(sk, skb); | 328 | skb_free_datagram(sk, skb); |
338 | |||
339 | caif_check_flow_release(sk); | 329 | caif_check_flow_release(sk); |
340 | 330 | return ret; | |
341 | return len; | ||
342 | 331 | ||
343 | read_error: | 332 | read_error: |
344 | return ret; | 333 | return ret; |