diff options
author | Al Viro <viro@zeniv.linux.org.uk> | 2014-11-20 07:01:29 -0500 |
---|---|---|
committer | Al Viro <viro@zeniv.linux.org.uk> | 2014-11-24 05:16:42 -0500 |
commit | 7424ce65065852bdf7a040bf2490da4a8fc4b464 (patch) | |
tree | b5a44fdeca7a925a2c763bb60a290c0b9bc6a11d /net/atm | |
parent | 0f7db23a07af5de22940f08c3f88a0741d4df0bc (diff) |
[atm] switch vcc_sendmsg() to copy_from_iter()
... and make it handle multi-segment iovecs - deals with that
"fix this later" issue for free. A bit of shame, really - it
had been there since 2.3.15pre3 when the whole thing went into the
tree, practically a historical artefact by now...
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Diffstat (limited to 'net/atm')
-rw-r--r-- | net/atm/common.c | 17 |
1 files changed, 6 insertions, 11 deletions
diff --git a/net/atm/common.c b/net/atm/common.c index 9cd1ccae9a11..f59112944c91 100644 --- a/net/atm/common.c +++ b/net/atm/common.c | |||
@@ -570,15 +570,16 @@ int vcc_recvmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *msg, | |||
570 | } | 570 | } |
571 | 571 | ||
572 | int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m, | 572 | int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m, |
573 | size_t total_len) | 573 | size_t size) |
574 | { | 574 | { |
575 | struct sock *sk = sock->sk; | 575 | struct sock *sk = sock->sk; |
576 | DEFINE_WAIT(wait); | 576 | DEFINE_WAIT(wait); |
577 | struct atm_vcc *vcc; | 577 | struct atm_vcc *vcc; |
578 | struct sk_buff *skb; | 578 | struct sk_buff *skb; |
579 | int eff, error; | 579 | int eff, error; |
580 | const void __user *buff; | 580 | struct iov_iter from; |
581 | int size; | 581 | |
582 | iov_iter_init(&from, WRITE, m->msg_iov, m->msg_iovlen, size); | ||
582 | 583 | ||
583 | lock_sock(sk); | 584 | lock_sock(sk); |
584 | if (sock->state != SS_CONNECTED) { | 585 | if (sock->state != SS_CONNECTED) { |
@@ -589,12 +590,6 @@ int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m, | |||
589 | error = -EISCONN; | 590 | error = -EISCONN; |
590 | goto out; | 591 | goto out; |
591 | } | 592 | } |
592 | if (m->msg_iovlen != 1) { | ||
593 | error = -ENOSYS; /* fix this later @@@ */ | ||
594 | goto out; | ||
595 | } | ||
596 | buff = m->msg_iov->iov_base; | ||
597 | size = m->msg_iov->iov_len; | ||
598 | vcc = ATM_SD(sock); | 593 | vcc = ATM_SD(sock); |
599 | if (test_bit(ATM_VF_RELEASED, &vcc->flags) || | 594 | if (test_bit(ATM_VF_RELEASED, &vcc->flags) || |
600 | test_bit(ATM_VF_CLOSE, &vcc->flags) || | 595 | test_bit(ATM_VF_CLOSE, &vcc->flags) || |
@@ -607,7 +602,7 @@ int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m, | |||
607 | error = 0; | 602 | error = 0; |
608 | goto out; | 603 | goto out; |
609 | } | 604 | } |
610 | if (size < 0 || size > vcc->qos.txtp.max_sdu) { | 605 | if (size > vcc->qos.txtp.max_sdu) { |
611 | error = -EMSGSIZE; | 606 | error = -EMSGSIZE; |
612 | goto out; | 607 | goto out; |
613 | } | 608 | } |
@@ -639,7 +634,7 @@ int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m, | |||
639 | goto out; | 634 | goto out; |
640 | skb->dev = NULL; /* for paths shared with net_device interfaces */ | 635 | skb->dev = NULL; /* for paths shared with net_device interfaces */ |
641 | ATM_SKB(skb)->atm_options = vcc->atm_options; | 636 | ATM_SKB(skb)->atm_options = vcc->atm_options; |
642 | if (copy_from_user(skb_put(skb, size), buff, size)) { | 637 | if (copy_from_iter(skb_put(skb, size), size, &from) != size) { |
643 | kfree_skb(skb); | 638 | kfree_skb(skb); |
644 | error = -EFAULT; | 639 | error = -EFAULT; |
645 | goto out; | 640 | goto out; |