|
NAME | DESCRIPTION | NOTES | SEE ALSO | COLOPHON |
|
|
|
io_uring_cancelation(7) Linux Programmer's Manual io_uring_cancelation(7)
io_uring_cancelation - io_uring request cancelation overview
io_uring provides mechanisms to cancel in-flight requests before
they complete naturally. This is useful for implementing timeouts,
handling connection drops, closing connections that go away, or
when a request is no longer needed.
Why cancel requests?
Common scenarios requiring cancelation:
• Timeouts: Cancel a read or accept that has been pending too long
• Connection management: Cancel pending operations when a
connection is closed
• Resource cleanup: Cancel operations on a file descriptor being
closed
• Multishot termination: Stop a multishot operation that is no
longer needed
Basic cancelation
The primary cancelation mechanism is IORING_OP_ASYNC_CANCEL (set
up with io_uring_prep_cancel(3) or related functions). By default,
it cancels a request matching a specific user_data:
/* Submit a read with user_data = 1234 */
sqe = io_uring_get_sqe(ring);
io_uring_prep_read(sqe, fd, buf, len, 0);
io_uring_sqe_set_data64(sqe, 1234);
io_uring_submit(ring);
/* Later, cancel it */
sqe = io_uring_get_sqe(ring);
io_uring_prep_cancel64(sqe, 1234, 0);
io_uring_submit(ring);
Cancelation results
When a cancelation is submitted, two CQEs are generated:
The canceled request's CQE:
• res is set to -ECANCELED (or occasionally -EINTR if the
operation was already in progress)
• The user_data identifies which request was canceled
The cancel request's CQE:
• res is 0 on success (request was found and canceled)
• res is -ENOENT if no matching request was found
• res is -EALREADY if the request was found but already
completing
The order of these CQEs is not guaranteed. The application may
receive the cancel CQE before or after the canceled request's CQE.
Cancelation flags
Various flags modify cancelation behavior:
IORING_ASYNC_CANCEL_ALL
Cancel all matching requests, not just the first one found.
The cancel CQE's res indicates how many requests were
canceled.
IORING_ASYNC_CANCEL_FD
Match requests by file descriptor instead of user_data.
Cancels requests operating on the specified fd:
io_uring_prep_cancel_fd(sqe, fd, IORING_ASYNC_CANCEL_FD);
IORING_ASYNC_CANCEL_ANY
Cancel any single request, ignoring user_data matching. Useful
for draining a ring of all pending requests when combined with
IORING_ASYNC_CANCEL_ALL.
IORING_ASYNC_CANCEL_FD_FIXED
The file descriptor is a fixed file (registered file index)
rather than a regular fd.
Flags can be combined:
/* Cancel all requests on a specific fd */
io_uring_prep_cancel_fd(sqe, fd,
IORING_ASYNC_CANCEL_FD | IORING_ASYNC_CANCEL_ALL);
/* Cancel all pending requests in the ring */
io_uring_prep_cancel(sqe, NULL,
IORING_ASYNC_CANCEL_ANY | IORING_ASYNC_CANCEL_ALL);
Race conditions
Cancelation is inherently racy. Between submitting the cancel
request and the kernel processing it:
• The target request may complete successfully
• The target request may fail for another reason
• The target request may already be in a non-cancelable state
Applications must handle these cases:
io_uring_wait_cqe(ring, &cqe);
if (cqe->user_data == cancel_user_data) {
/* This is the cancel operation's result */
if (cqe->res == -ENOENT) {
/* Request already completed or not found */
} else if (cqe->res == -EALREADY) {
/* Request was found but completing */
} else if (cqe->res >= 0) {
/* Successfully canceled res requests */
}
} else {
/* This is the original request's result */
if (cqe->res == -ECANCELED) {
/* Request was canceled */
} else {
/* Request completed normally (or with error) */
}
}
Link timeouts
For timing out a single operation, link timeouts are often simpler
than explicit cancelation. See io_uring_linked_requests(7) for
details:
sqe = io_uring_get_sqe(ring);
io_uring_prep_read(sqe, fd, buf, len, 0);
sqe->flags |= IOSQE_IO_LINK;
sqe = io_uring_get_sqe(ring);
io_uring_prep_link_timeout(sqe, &timeout, 0);
The kernel handles the cancelation automatically if the timeout
expires.
Canceling multishot requests
Multishot requests (see io_uring_multishot(7)) continue generating
completions until canceled or an error occurs. To stop a
multishot request:
/* Cancel a multishot accept */
io_uring_prep_cancel64(sqe, accept_user_data, 0);
After cancelation:
• The multishot generates a final CQE with -ECANCELED
• The IORING_CQE_F_MORE flag is not set on this final CQE
• The cancel CQE indicates success
Canceling by file descriptor
When a file descriptor is closed (either via close(2) or
IORING_OP_CLOSE), pending requests operating on that fd are not
automatically canceled. This differs from synchronous I/O behavior
and is a common source of confusion.
In synchronous I/O, closing a file descriptor is typically the
last reference to the underlying file, so the close completes any
pending operations. However, io_uring holds its own reference to
the file for each pending request. Closing the application's fd
does not release these references — the pending read, recv, or
other operation continues to hold a reference and will not
automatically complete or fail.
If an application expects a pending read on an fd to post a
completion when the fd is closed, that will not happen. The
request must be explicitly canceled:
/* Cancel all operations on fd before closing */
sqe = io_uring_get_sqe(ring);
io_uring_prep_cancel_fd(sqe, fd,
IORING_ASYNC_CANCEL_FD | IORING_ASYNC_CANCEL_ALL);
io_uring_submit(ring);
/* Wait for cancelations, then close */
Shutdown cancelation
When an io_uring instance is closed (via io_uring_queue_exit(3) or
closing the ring file descriptor), all pending requests are
automatically canceled. Manual cancelation before shutdown is not
required.
However, if the application needs to ensure all requests are
completed before proceeding (e.g., to process their results or
free associated resources), explicit cancelation can be used:
/* Cancel everything */
sqe = io_uring_get_sqe(ring);
io_uring_prep_cancel(sqe, NULL,
IORING_ASYNC_CANCEL_ANY | IORING_ASYNC_CANCEL_ALL);
io_uring_submit(ring);
/* Wait for all CQEs */
while (pending_count > 0) {
io_uring_wait_cqe(ring, &cqe);
pending_count--;
io_uring_cqe_seen(ring, cqe);
}
Synchronous cancelation
For cases where the application needs to cancel requests and wait
for the cancelation to complete in a single blocking call,
io_uring_register_sync_cancel(3) provides a synchronous interface:
struct io_uring_sync_cancel_reg reg = {
.addr = user_data,
.timeout.tv_sec = 5,
};
ret = io_uring_register_sync_cancel(ring, ®);
This blocks until the matching request is canceled or the timeout
expires. It is useful when the application cannot easily integrate
asynchronous cancelation into its event loop.
• Not all operations are cancelable. Operations that have already
been submitted to hardware (e.g., disk I/O in progress)
typically cannot be canceled.
• Cancelation is asynchronous. The cancel request itself may take
time to complete.
• When using IORING_ASYNC_CANCEL_ALL, the cancel CQE's res field
contains the count of canceled requests.
• Fixed files can be canceled using IORING_ASYNC_CANCEL_FD_FIXED
with the file index instead of a regular fd.
• Poll operations and multishot requests are generally good
candidates for cancelation. Completed disk I/O is not.
io_uring(7), io_uring_linked_requests(7), io_uring_multishot(7),
io_uring_prep_cancel(3), io_uring_prep_cancel64(3),
io_uring_prep_cancel_fd(3), io_uring_prep_link_timeout(3),
io_uring_register_sync_cancel(3)
This page is part of the liburing (A library for io_uring)
project. Information about the project can be found at
⟨https://github.com/axboe/liburing⟩. If you have a bug report for
this manual page, send it to io-uring@vger.kernel.org. This page
was obtained from the project's upstream Git repository
⟨https://github.com/axboe/liburing⟩ on 2026-05-24. (At that time,
the date of the most recent commit that was found in the
repository was 2026-05-18.) If you discover any rendering
problems in this HTML version of the page, or you believe there is
a better or more up-to-date source for the page, or you have
corrections or improvements to the information in this COLOPHON
(which is not part of the original manual page), send a mail to
man-pages@man7.org
Linux January 18, 2025 io_uring_cancelation(7)