forked from mirrors/linux
		
	rds: switch rds_message_copy_from_user() to iov_iter
Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
		
							parent
							
								
									c310e72c89
								
							
						
					
					
						commit
						083735f4b0
					
				
					 3 changed files with 16 additions and 33 deletions
				
			
		|  | @ -264,64 +264,46 @@ struct rds_message *rds_message_map_pages(unsigned long *page_addrs, unsigned in | |||
| 	return rm; | ||||
| } | ||||
| 
 | ||||
| int rds_message_copy_from_user(struct rds_message *rm, struct iovec *first_iov, | ||||
| 					       size_t total_len) | ||||
| int rds_message_copy_from_user(struct rds_message *rm, struct iov_iter *from) | ||||
| { | ||||
| 	unsigned long to_copy; | ||||
| 	unsigned long iov_off; | ||||
| 	unsigned long sg_off; | ||||
| 	struct iovec *iov; | ||||
| 	struct scatterlist *sg; | ||||
| 	int ret = 0; | ||||
| 
 | ||||
| 	rm->m_inc.i_hdr.h_len = cpu_to_be32(total_len); | ||||
| 	rm->m_inc.i_hdr.h_len = cpu_to_be32(iov_iter_count(from)); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * now allocate and copy in the data payload. | ||||
| 	 */ | ||||
| 	sg = rm->data.op_sg; | ||||
| 	iov = first_iov; | ||||
| 	iov_off = 0; | ||||
| 	sg_off = 0; /* Dear gcc, sg->page will be null from kzalloc. */ | ||||
| 
 | ||||
| 	while (total_len) { | ||||
| 	while (iov_iter_count(from)) { | ||||
| 		if (!sg_page(sg)) { | ||||
| 			ret = rds_page_remainder_alloc(sg, total_len, | ||||
| 			ret = rds_page_remainder_alloc(sg, iov_iter_count(from), | ||||
| 						       GFP_HIGHUSER); | ||||
| 			if (ret) | ||||
| 				goto out; | ||||
| 				return ret; | ||||
| 			rm->data.op_nents++; | ||||
| 			sg_off = 0; | ||||
| 		} | ||||
| 
 | ||||
| 		while (iov_off == iov->iov_len) { | ||||
| 			iov_off = 0; | ||||
| 			iov++; | ||||
| 		} | ||||
| 		to_copy = min_t(unsigned long, iov_iter_count(from), | ||||
| 				sg->length - sg_off); | ||||
| 
 | ||||
| 		to_copy = min(iov->iov_len - iov_off, sg->length - sg_off); | ||||
| 		to_copy = min_t(size_t, to_copy, total_len); | ||||
| 		rds_stats_add(s_copy_from_user, to_copy); | ||||
| 		ret = copy_page_from_iter(sg_page(sg), sg->offset + sg_off, | ||||
| 					  to_copy, from); | ||||
| 		if (ret != to_copy) | ||||
| 			return -EFAULT; | ||||
| 
 | ||||
| 		rdsdebug("copying %lu bytes from user iov [%p, %zu] + %lu to " | ||||
| 			 "sg [%p, %u, %u] + %lu\n", | ||||
| 			 to_copy, iov->iov_base, iov->iov_len, iov_off, | ||||
| 			 (void *)sg_page(sg), sg->offset, sg->length, sg_off); | ||||
| 
 | ||||
| 		ret = rds_page_copy_from_user(sg_page(sg), sg->offset + sg_off, | ||||
| 					      iov->iov_base + iov_off, | ||||
| 					      to_copy); | ||||
| 		if (ret) | ||||
| 			goto out; | ||||
| 
 | ||||
| 		iov_off += to_copy; | ||||
| 		total_len -= to_copy; | ||||
| 		sg_off += to_copy; | ||||
| 
 | ||||
| 		if (sg_off == sg->length) | ||||
| 			sg++; | ||||
| 	} | ||||
| 
 | ||||
| out: | ||||
| 	return ret; | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -656,8 +656,7 @@ rds_conn_connecting(struct rds_connection *conn) | |||
| /* message.c */ | ||||
| struct rds_message *rds_message_alloc(unsigned int nents, gfp_t gfp); | ||||
| struct scatterlist *rds_message_alloc_sgs(struct rds_message *rm, int nents); | ||||
| int rds_message_copy_from_user(struct rds_message *rm, struct iovec *first_iov, | ||||
| 					       size_t total_len); | ||||
| int rds_message_copy_from_user(struct rds_message *rm, struct iov_iter *from); | ||||
| struct rds_message *rds_message_map_pages(unsigned long *page_addrs, unsigned int total_len); | ||||
| void rds_message_populate_header(struct rds_header *hdr, __be16 sport, | ||||
| 				 __be16 dport, u64 seq); | ||||
|  |  | |||
|  | @ -934,7 +934,9 @@ int rds_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *msg, | |||
| 	int queued = 0, allocated_mr = 0; | ||||
| 	int nonblock = msg->msg_flags & MSG_DONTWAIT; | ||||
| 	long timeo = sock_sndtimeo(sk, nonblock); | ||||
| 	struct iov_iter from; | ||||
| 
 | ||||
| 	iov_iter_init(&from, WRITE, msg->msg_iov, msg->msg_iovlen, payload_len); | ||||
| 	/* Mirror Linux UDP mirror of BSD error message compatibility */ | ||||
| 	/* XXX: Perhaps MSG_MORE someday */ | ||||
| 	if (msg->msg_flags & ~(MSG_DONTWAIT | MSG_CMSG_COMPAT)) { | ||||
|  | @ -982,7 +984,7 @@ int rds_sendmsg(struct kiocb *iocb, struct socket *sock, struct msghdr *msg, | |||
| 			ret = -ENOMEM; | ||||
| 			goto out; | ||||
| 		} | ||||
| 		ret = rds_message_copy_from_user(rm, msg->msg_iov, payload_len); | ||||
| 		ret = rds_message_copy_from_user(rm, &from); | ||||
| 		if (ret) | ||||
| 			goto out; | ||||
| 	} | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue
	
	 Al Viro
						Al Viro