mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	io_uring: move reissue into regular IO path
It's non-obvious how retry is done for block backed files, when it happens off the kiocb done path. It also makes it tricky to deal with the iov_iter handling. Just mark the req as needing a reissue, and handling it from the submission path instead. This makes it directly obvious that we're not re-importing the iovec from userspace past the submit point, and it means that we can just reuse our usual -EAGAIN retry path from the read/write handling. At some point in the future, we'll gain the ability to always reliably return -EAGAIN through the stack. A previous attempt on the block side didn't pan out and got reverted, hence the need to check for this information out-of-band right now. Signed-off-by: Jens Axboe <axboe@kernel.dk>
This commit is contained in:
		
							parent
							
								
									07204f2157
								
							
						
					
					
						commit
						230d50d448
					
				| @ -697,6 +697,7 @@ enum { | |||||||
| 	REQ_F_NO_FILE_TABLE_BIT, | 	REQ_F_NO_FILE_TABLE_BIT, | ||||||
| 	REQ_F_LTIMEOUT_ACTIVE_BIT, | 	REQ_F_LTIMEOUT_ACTIVE_BIT, | ||||||
| 	REQ_F_COMPLETE_INLINE_BIT, | 	REQ_F_COMPLETE_INLINE_BIT, | ||||||
|  | 	REQ_F_REISSUE_BIT, | ||||||
| 
 | 
 | ||||||
| 	/* not a real bit, just to check we're not overflowing the space */ | 	/* not a real bit, just to check we're not overflowing the space */ | ||||||
| 	__REQ_F_LAST_BIT, | 	__REQ_F_LAST_BIT, | ||||||
| @ -740,6 +741,8 @@ enum { | |||||||
| 	REQ_F_LTIMEOUT_ACTIVE	= BIT(REQ_F_LTIMEOUT_ACTIVE_BIT), | 	REQ_F_LTIMEOUT_ACTIVE	= BIT(REQ_F_LTIMEOUT_ACTIVE_BIT), | ||||||
| 	/* completion is deferred through io_comp_state */ | 	/* completion is deferred through io_comp_state */ | ||||||
| 	REQ_F_COMPLETE_INLINE	= BIT(REQ_F_COMPLETE_INLINE_BIT), | 	REQ_F_COMPLETE_INLINE	= BIT(REQ_F_COMPLETE_INLINE_BIT), | ||||||
|  | 	/* caller should reissue async */ | ||||||
|  | 	REQ_F_REISSUE		= BIT(REQ_F_REISSUE_BIT), | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct async_poll { | struct async_poll { | ||||||
| @ -2503,8 +2506,10 @@ static void __io_complete_rw(struct io_kiocb *req, long res, long res2, | |||||||
| 
 | 
 | ||||||
| 	if (req->rw.kiocb.ki_flags & IOCB_WRITE) | 	if (req->rw.kiocb.ki_flags & IOCB_WRITE) | ||||||
| 		kiocb_end_write(req); | 		kiocb_end_write(req); | ||||||
| 	if ((res == -EAGAIN || res == -EOPNOTSUPP) && io_rw_reissue(req)) | 	if ((res == -EAGAIN || res == -EOPNOTSUPP) && io_rw_should_reissue(req)) { | ||||||
|  | 		req->flags |= REQ_F_REISSUE; | ||||||
| 		return; | 		return; | ||||||
|  | 	} | ||||||
| 	if (res != req->result) | 	if (res != req->result) | ||||||
| 		req_set_fail_links(req); | 		req_set_fail_links(req); | ||||||
| 	if (req->flags & REQ_F_BUFFER_SELECTED) | 	if (req->flags & REQ_F_BUFFER_SELECTED) | ||||||
| @ -3283,9 +3288,7 @@ static int io_read(struct io_kiocb *req, unsigned int issue_flags) | |||||||
| 
 | 
 | ||||||
| 	ret = io_iter_do_read(req, iter); | 	ret = io_iter_do_read(req, iter); | ||||||
| 
 | 
 | ||||||
| 	if (ret == -EIOCBQUEUED) { | 	if (ret == -EAGAIN || (req->flags & REQ_F_REISSUE)) { | ||||||
| 		goto out_free; |  | ||||||
| 	} else if (ret == -EAGAIN) { |  | ||||||
| 		/* IOPOLL retry should happen for io-wq threads */ | 		/* IOPOLL retry should happen for io-wq threads */ | ||||||
| 		if (!force_nonblock && !(req->ctx->flags & IORING_SETUP_IOPOLL)) | 		if (!force_nonblock && !(req->ctx->flags & IORING_SETUP_IOPOLL)) | ||||||
| 			goto done; | 			goto done; | ||||||
| @ -3295,6 +3298,8 @@ static int io_read(struct io_kiocb *req, unsigned int issue_flags) | |||||||
| 		/* some cases will consume bytes even on error returns */ | 		/* some cases will consume bytes even on error returns */ | ||||||
| 		iov_iter_revert(iter, io_size - iov_iter_count(iter)); | 		iov_iter_revert(iter, io_size - iov_iter_count(iter)); | ||||||
| 		ret = 0; | 		ret = 0; | ||||||
|  | 	} else if (ret == -EIOCBQUEUED) { | ||||||
|  | 		goto out_free; | ||||||
| 	} else if (ret <= 0 || ret == io_size || !force_nonblock || | 	} else if (ret <= 0 || ret == io_size || !force_nonblock || | ||||||
| 		   (req->flags & REQ_F_NOWAIT) || !(req->flags & REQ_F_ISREG)) { | 		   (req->flags & REQ_F_NOWAIT) || !(req->flags & REQ_F_ISREG)) { | ||||||
| 		/* read all, failed, already did sync or don't want to retry */ | 		/* read all, failed, already did sync or don't want to retry */ | ||||||
| @ -3407,6 +3412,9 @@ static int io_write(struct io_kiocb *req, unsigned int issue_flags) | |||||||
| 	else | 	else | ||||||
| 		ret2 = -EINVAL; | 		ret2 = -EINVAL; | ||||||
| 
 | 
 | ||||||
|  | 	if (req->flags & REQ_F_REISSUE) | ||||||
|  | 		ret2 = -EAGAIN; | ||||||
|  | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Raw bdev writes will return -EOPNOTSUPP for IOCB_NOWAIT. Just | 	 * Raw bdev writes will return -EOPNOTSUPP for IOCB_NOWAIT. Just | ||||||
| 	 * retry them without IOCB_NOWAIT. | 	 * retry them without IOCB_NOWAIT. | ||||||
| @ -6160,6 +6168,7 @@ static void io_wq_submit_work(struct io_wq_work *work) | |||||||
| 		ret = -ECANCELED; | 		ret = -ECANCELED; | ||||||
| 
 | 
 | ||||||
| 	if (!ret) { | 	if (!ret) { | ||||||
|  | 		req->flags &= ~REQ_F_REISSUE; | ||||||
| 		do { | 		do { | ||||||
| 			ret = io_issue_sqe(req, 0); | 			ret = io_issue_sqe(req, 0); | ||||||
| 			/*
 | 			/*
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Jens Axboe
						Jens Axboe