Commit af2b040e authored by Al Viro's avatar Al Viro

rxrpc: switch rxrpc_send_data() to iov_iter primitives

Convert skb_add_data() to iov_iter; allows to get rid of the explicit
messing with iovec in its only caller - skb_add_data() will keep advancing
->msg_iter for us, so there's no need to similate that manually.
Signed-off-by: default avatarAl Viro <viro@zeniv.linux.org.uk>
parent 4c946d9c
...@@ -2484,19 +2484,18 @@ static inline int skb_put_padto(struct sk_buff *skb, unsigned int len) ...@@ -2484,19 +2484,18 @@ static inline int skb_put_padto(struct sk_buff *skb, unsigned int len)
} }
static inline int skb_add_data(struct sk_buff *skb, static inline int skb_add_data(struct sk_buff *skb,
char __user *from, int copy) struct iov_iter *from, int copy)
{ {
const int off = skb->len; const int off = skb->len;
if (skb->ip_summed == CHECKSUM_NONE) { if (skb->ip_summed == CHECKSUM_NONE) {
int err = 0; __wsum csum = 0;
__wsum csum = csum_and_copy_from_user(from, skb_put(skb, copy), if (csum_and_copy_from_iter(skb_put(skb, copy), copy,
copy, 0, &err); &csum, from) == copy) {
if (!err) {
skb->csum = csum_block_add(skb->csum, csum, off); skb->csum = csum_block_add(skb->csum, csum, off);
return 0; return 0;
} }
} else if (!copy_from_user(skb_put(skb, copy), from, copy)) } else if (copy_from_iter(skb_put(skb, copy), copy, from) == copy)
return 0; return 0;
__skb_trim(skb, off); __skb_trim(skb, off);
......
...@@ -529,13 +529,11 @@ static int rxrpc_send_data(struct kiocb *iocb, ...@@ -529,13 +529,11 @@ static int rxrpc_send_data(struct kiocb *iocb,
struct msghdr *msg, size_t len) struct msghdr *msg, size_t len)
{ {
struct rxrpc_skb_priv *sp; struct rxrpc_skb_priv *sp;
unsigned char __user *from;
struct sk_buff *skb; struct sk_buff *skb;
const struct iovec *iov;
struct sock *sk = &rx->sk; struct sock *sk = &rx->sk;
long timeo; long timeo;
bool more; bool more;
int ret, ioc, segment, copied; int ret, copied;
timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT); timeo = sock_sndtimeo(sk, msg->msg_flags & MSG_DONTWAIT);
...@@ -545,25 +543,17 @@ static int rxrpc_send_data(struct kiocb *iocb, ...@@ -545,25 +543,17 @@ static int rxrpc_send_data(struct kiocb *iocb,
if (sk->sk_err || (sk->sk_shutdown & SEND_SHUTDOWN)) if (sk->sk_err || (sk->sk_shutdown & SEND_SHUTDOWN))
return -EPIPE; return -EPIPE;
iov = msg->msg_iter.iov;
ioc = msg->msg_iter.nr_segs - 1;
from = iov->iov_base;
segment = iov->iov_len;
iov++;
more = msg->msg_flags & MSG_MORE; more = msg->msg_flags & MSG_MORE;
skb = call->tx_pending; skb = call->tx_pending;
call->tx_pending = NULL; call->tx_pending = NULL;
copied = 0; copied = 0;
do { if (len > iov_iter_count(&msg->msg_iter))
len = iov_iter_count(&msg->msg_iter);
while (len) {
int copy; int copy;
if (segment > len)
segment = len;
_debug("SEGMENT %d @%p", segment, from);
if (!skb) { if (!skb) {
size_t size, chunk, max, space; size_t size, chunk, max, space;
...@@ -631,13 +621,13 @@ static int rxrpc_send_data(struct kiocb *iocb, ...@@ -631,13 +621,13 @@ static int rxrpc_send_data(struct kiocb *iocb,
/* append next segment of data to the current buffer */ /* append next segment of data to the current buffer */
copy = skb_tailroom(skb); copy = skb_tailroom(skb);
ASSERTCMP(copy, >, 0); ASSERTCMP(copy, >, 0);
if (copy > segment) if (copy > len)
copy = segment; copy = len;
if (copy > sp->remain) if (copy > sp->remain)
copy = sp->remain; copy = sp->remain;
_debug("add"); _debug("add");
ret = skb_add_data(skb, from, copy); ret = skb_add_data(skb, &msg->msg_iter, copy);
_debug("added"); _debug("added");
if (ret < 0) if (ret < 0)
goto efault; goto efault;
...@@ -646,18 +636,6 @@ static int rxrpc_send_data(struct kiocb *iocb, ...@@ -646,18 +636,6 @@ static int rxrpc_send_data(struct kiocb *iocb,
copied += copy; copied += copy;
len -= copy; len -= copy;
segment -= copy;
from += copy;
while (segment == 0 && ioc > 0) {
from = iov->iov_base;
segment = iov->iov_len;
iov++;
ioc--;
}
if (len == 0) {
segment = 0;
ioc = 0;
}
/* check for the far side aborting the call or a network error /* check for the far side aborting the call or a network error
* occurring */ * occurring */
...@@ -665,7 +643,7 @@ static int rxrpc_send_data(struct kiocb *iocb, ...@@ -665,7 +643,7 @@ static int rxrpc_send_data(struct kiocb *iocb,
goto call_aborted; goto call_aborted;
/* add the packet to the send queue if it's now full */ /* add the packet to the send queue if it's now full */
if (sp->remain <= 0 || (segment == 0 && !more)) { if (sp->remain <= 0 || (!len && !more)) {
struct rxrpc_connection *conn = call->conn; struct rxrpc_connection *conn = call->conn;
uint32_t seq; uint32_t seq;
size_t pad; size_t pad;
...@@ -711,11 +689,10 @@ static int rxrpc_send_data(struct kiocb *iocb, ...@@ -711,11 +689,10 @@ static int rxrpc_send_data(struct kiocb *iocb,
memcpy(skb->head, &sp->hdr, memcpy(skb->head, &sp->hdr,
sizeof(struct rxrpc_header)); sizeof(struct rxrpc_header));
rxrpc_queue_packet(call, skb, segment == 0 && !more); rxrpc_queue_packet(call, skb, !iov_iter_count(&msg->msg_iter) && !more);
skb = NULL; skb = NULL;
} }
}
} while (segment > 0);
success: success:
ret = copied; ret = copied;
......
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