| // SPDX-License-Identifier: GPL-2.0-only | 
 | /* | 
 |  * linux/net/sunrpc/socklib.c | 
 |  * | 
 |  * Common socket helper routines for RPC client and server | 
 |  * | 
 |  * Copyright (C) 1995, 1996 Olaf Kirch <okir@monad.swb.de> | 
 |  */ | 
 |  | 
 | #include <linux/compiler.h> | 
 | #include <linux/netdevice.h> | 
 | #include <linux/gfp.h> | 
 | #include <linux/skbuff.h> | 
 | #include <linux/types.h> | 
 | #include <linux/pagemap.h> | 
 | #include <linux/udp.h> | 
 | #include <linux/sunrpc/msg_prot.h> | 
 | #include <linux/sunrpc/sched.h> | 
 | #include <linux/sunrpc/xdr.h> | 
 | #include <linux/export.h> | 
 |  | 
 | #include "socklib.h" | 
 |  | 
 | /* | 
 |  * Helper structure for copying from an sk_buff. | 
 |  */ | 
 | struct xdr_skb_reader { | 
 | 	struct sk_buff	*skb; | 
 | 	unsigned int	offset; | 
 | 	bool		need_checksum; | 
 | 	size_t		count; | 
 | 	__wsum		csum; | 
 | }; | 
 |  | 
 | /** | 
 |  * xdr_skb_read_bits - copy some data bits from skb to internal buffer | 
 |  * @desc: sk_buff copy helper | 
 |  * @to: copy destination | 
 |  * @len: number of bytes to copy | 
 |  * | 
 |  * Possibly called several times to iterate over an sk_buff and copy data out of | 
 |  * it. | 
 |  */ | 
 | static size_t | 
 | xdr_skb_read_bits(struct xdr_skb_reader *desc, void *to, size_t len) | 
 | { | 
 | 	len = min(len, desc->count); | 
 |  | 
 | 	if (desc->need_checksum) { | 
 | 		__wsum csum; | 
 |  | 
 | 		csum = skb_copy_and_csum_bits(desc->skb, desc->offset, to, len); | 
 | 		desc->csum = csum_block_add(desc->csum, csum, desc->offset); | 
 | 	} else { | 
 | 		if (unlikely(skb_copy_bits(desc->skb, desc->offset, to, len))) | 
 | 			return 0; | 
 | 	} | 
 |  | 
 | 	desc->count -= len; | 
 | 	desc->offset += len; | 
 | 	return len; | 
 | } | 
 |  | 
 | static ssize_t | 
 | xdr_partial_copy_from_skb(struct xdr_buf *xdr, struct xdr_skb_reader *desc) | 
 | { | 
 | 	struct page **ppage = xdr->pages + (xdr->page_base >> PAGE_SHIFT); | 
 | 	unsigned int poff = xdr->page_base & ~PAGE_MASK; | 
 | 	unsigned int pglen = xdr->page_len; | 
 | 	ssize_t copied = 0; | 
 | 	size_t ret; | 
 |  | 
 | 	if (xdr->head[0].iov_len == 0) | 
 | 		return 0; | 
 |  | 
 | 	ret = xdr_skb_read_bits(desc, xdr->head[0].iov_base, | 
 | 				xdr->head[0].iov_len); | 
 | 	if (ret != xdr->head[0].iov_len || !desc->count) | 
 | 		return ret; | 
 | 	copied += ret; | 
 |  | 
 | 	while (pglen) { | 
 | 		unsigned int len = min(PAGE_SIZE - poff, pglen); | 
 | 		char *kaddr; | 
 |  | 
 | 		/* ACL likes to be lazy in allocating pages - ACLs | 
 | 		 * are small by default but can get huge. */ | 
 | 		if ((xdr->flags & XDRBUF_SPARSE_PAGES) && *ppage == NULL) { | 
 | 			*ppage = alloc_page(GFP_NOWAIT); | 
 | 			if (unlikely(*ppage == NULL)) { | 
 | 				if (copied == 0) | 
 | 					return -ENOMEM; | 
 | 				return copied; | 
 | 			} | 
 | 		} | 
 |  | 
 | 		kaddr = kmap_atomic(*ppage); | 
 | 		ret = xdr_skb_read_bits(desc, kaddr + poff, len); | 
 | 		flush_dcache_page(*ppage); | 
 | 		kunmap_atomic(kaddr); | 
 |  | 
 | 		copied += ret; | 
 | 		if (ret != len || !desc->count) | 
 | 			return copied; | 
 | 		ppage++; | 
 | 		pglen -= len; | 
 | 		poff = 0; | 
 | 	} | 
 |  | 
 | 	if (xdr->tail[0].iov_len) { | 
 | 		copied += xdr_skb_read_bits(desc, xdr->tail[0].iov_base, | 
 | 					xdr->tail[0].iov_len); | 
 | 	} | 
 |  | 
 | 	return copied; | 
 | } | 
 |  | 
 | /** | 
 |  * csum_partial_copy_to_xdr - checksum and copy data | 
 |  * @xdr: target XDR buffer | 
 |  * @skb: source skb | 
 |  * | 
 |  * We have set things up such that we perform the checksum of the UDP | 
 |  * packet in parallel with the copies into the RPC client iovec.  -DaveM | 
 |  */ | 
 | int csum_partial_copy_to_xdr(struct xdr_buf *xdr, struct sk_buff *skb) | 
 | { | 
 | 	struct xdr_skb_reader desc = { | 
 | 		.skb		= skb, | 
 | 		.count		= skb->len - desc.offset, | 
 | 	}; | 
 |  | 
 | 	if (skb_csum_unnecessary(skb)) { | 
 | 		if (xdr_partial_copy_from_skb(xdr, &desc) < 0) | 
 | 			return -1; | 
 | 		if (desc.count) | 
 | 			return -1; | 
 | 		return 0; | 
 | 	} | 
 |  | 
 | 	desc.need_checksum = true; | 
 | 	desc.csum = csum_partial(skb->data, desc.offset, skb->csum); | 
 | 	if (xdr_partial_copy_from_skb(xdr, &desc) < 0) | 
 | 		return -1; | 
 | 	if (desc.offset != skb->len) { | 
 | 		__wsum csum2; | 
 | 		csum2 = skb_checksum(skb, desc.offset, skb->len - desc.offset, 0); | 
 | 		desc.csum = csum_block_add(desc.csum, csum2, desc.offset); | 
 | 	} | 
 | 	if (desc.count) | 
 | 		return -1; | 
 | 	if (csum_fold(desc.csum)) | 
 | 		return -1; | 
 | 	if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE) && | 
 | 	    !skb->csum_complete_sw) | 
 | 		netdev_rx_csum_fault(skb->dev, skb); | 
 | 	return 0; | 
 | } | 
 |  | 
 | static inline int xprt_sendmsg(struct socket *sock, struct msghdr *msg, | 
 | 			       size_t seek) | 
 | { | 
 | 	if (seek) | 
 | 		iov_iter_advance(&msg->msg_iter, seek); | 
 | 	return sock_sendmsg(sock, msg); | 
 | } | 
 |  | 
 | static int xprt_send_kvec(struct socket *sock, struct msghdr *msg, | 
 | 			  struct kvec *vec, size_t seek) | 
 | { | 
 | 	iov_iter_kvec(&msg->msg_iter, ITER_SOURCE, vec, 1, vec->iov_len); | 
 | 	return xprt_sendmsg(sock, msg, seek); | 
 | } | 
 |  | 
 | static int xprt_send_pagedata(struct socket *sock, struct msghdr *msg, | 
 | 			      struct xdr_buf *xdr, size_t base) | 
 | { | 
 | 	iov_iter_bvec(&msg->msg_iter, ITER_SOURCE, xdr->bvec, xdr_buf_pagecount(xdr), | 
 | 		      xdr->page_len + xdr->page_base); | 
 | 	return xprt_sendmsg(sock, msg, base + xdr->page_base); | 
 | } | 
 |  | 
 | /* Common case: | 
 |  *  - stream transport | 
 |  *  - sending from byte 0 of the message | 
 |  *  - the message is wholly contained in @xdr's head iovec | 
 |  */ | 
 | static int xprt_send_rm_and_kvec(struct socket *sock, struct msghdr *msg, | 
 | 				 rpc_fraghdr marker, struct kvec *vec, | 
 | 				 size_t base) | 
 | { | 
 | 	struct kvec iov[2] = { | 
 | 		[0] = { | 
 | 			.iov_base	= &marker, | 
 | 			.iov_len	= sizeof(marker) | 
 | 		}, | 
 | 		[1] = *vec, | 
 | 	}; | 
 | 	size_t len = iov[0].iov_len + iov[1].iov_len; | 
 |  | 
 | 	iov_iter_kvec(&msg->msg_iter, ITER_SOURCE, iov, 2, len); | 
 | 	return xprt_sendmsg(sock, msg, base); | 
 | } | 
 |  | 
 | /** | 
 |  * xprt_sock_sendmsg - write an xdr_buf directly to a socket | 
 |  * @sock: open socket to send on | 
 |  * @msg: socket message metadata | 
 |  * @xdr: xdr_buf containing this request | 
 |  * @base: starting position in the buffer | 
 |  * @marker: stream record marker field | 
 |  * @sent_p: return the total number of bytes successfully queued for sending | 
 |  * | 
 |  * Return values: | 
 |  *   On success, returns zero and fills in @sent_p. | 
 |  *   %-ENOTSOCK if  @sock is not a struct socket. | 
 |  */ | 
 | int xprt_sock_sendmsg(struct socket *sock, struct msghdr *msg, | 
 | 		      struct xdr_buf *xdr, unsigned int base, | 
 | 		      rpc_fraghdr marker, unsigned int *sent_p) | 
 | { | 
 | 	unsigned int rmsize = marker ? sizeof(marker) : 0; | 
 | 	unsigned int remainder = rmsize + xdr->len - base; | 
 | 	unsigned int want; | 
 | 	int err = 0; | 
 |  | 
 | 	*sent_p = 0; | 
 |  | 
 | 	if (unlikely(!sock)) | 
 | 		return -ENOTSOCK; | 
 |  | 
 | 	msg->msg_flags |= MSG_MORE; | 
 | 	want = xdr->head[0].iov_len + rmsize; | 
 | 	if (base < want) { | 
 | 		unsigned int len = want - base; | 
 |  | 
 | 		remainder -= len; | 
 | 		if (remainder == 0) | 
 | 			msg->msg_flags &= ~MSG_MORE; | 
 | 		if (rmsize) | 
 | 			err = xprt_send_rm_and_kvec(sock, msg, marker, | 
 | 						    &xdr->head[0], base); | 
 | 		else | 
 | 			err = xprt_send_kvec(sock, msg, &xdr->head[0], base); | 
 | 		if (remainder == 0 || err != len) | 
 | 			goto out; | 
 | 		*sent_p += err; | 
 | 		base = 0; | 
 | 	} else { | 
 | 		base -= want; | 
 | 	} | 
 |  | 
 | 	if (base < xdr->page_len) { | 
 | 		unsigned int len = xdr->page_len - base; | 
 |  | 
 | 		remainder -= len; | 
 | 		if (remainder == 0) | 
 | 			msg->msg_flags &= ~MSG_MORE; | 
 | 		err = xprt_send_pagedata(sock, msg, xdr, base); | 
 | 		if (remainder == 0 || err != len) | 
 | 			goto out; | 
 | 		*sent_p += err; | 
 | 		base = 0; | 
 | 	} else { | 
 | 		base -= xdr->page_len; | 
 | 	} | 
 |  | 
 | 	if (base >= xdr->tail[0].iov_len) | 
 | 		return 0; | 
 | 	msg->msg_flags &= ~MSG_MORE; | 
 | 	err = xprt_send_kvec(sock, msg, &xdr->tail[0], base); | 
 | out: | 
 | 	if (err > 0) { | 
 | 		*sent_p += err; | 
 | 		err = 0; | 
 | 	} | 
 | 	return err; | 
 | } |