aboutsummaryrefslogtreecommitdiffstats
path: root/net/atm
diff options
context:
space:
mode:
authorAl Viro <viro@zeniv.linux.org.uk>2014-11-20 07:01:29 -0500
committerAl Viro <viro@zeniv.linux.org.uk>2014-11-24 05:16:42 -0500
commit7424ce65065852bdf7a040bf2490da4a8fc4b464 (patch)
treeb5a44fdeca7a925a2c763bb60a290c0b9bc6a11d /net/atm
parent0f7db23a07af5de22940f08c3f88a0741d4df0bc (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.c17
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
572int vcc_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *m, 572int 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;