Skip to content

Commit

Permalink
io_uring: Fix XArray usage in io_uring_add_task_file
Browse files Browse the repository at this point in the history
commit 236434c upstream.

The xas_store() wasn't paired with an xas_nomem() loop, so if it couldn't
allocate memory using GFP_NOWAIT, it would leak the reference to the file
descriptor.  Also the node pointed to by the xas could be freed between
the call to xas_load() under the rcu_read_lock() and the acquisition of
the xa_lock.

It's easier to just use the normal xa_load/xa_store interface here.

Signed-off-by: Matthew Wilcox (Oracle) <willy@infradead.org>
[axboe: fix missing assign after alloc, cur_uring -> tctx rename]
Signed-off-by: Jens Axboe <axboe@kernel.dk>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
  • Loading branch information
Matthew Wilcox (Oracle) authored and gregkh committed Nov 1, 2020
1 parent 4f0cc46 commit a05b5f7
Showing 1 changed file with 9 additions and 12 deletions.
21 changes: 9 additions & 12 deletions fs/io_uring.c
Expand Up @@ -8336,27 +8336,24 @@ static void io_uring_cancel_task_requests(struct io_ring_ctx *ctx,
*/
static int io_uring_add_task_file(struct file *file)
{
if (unlikely(!current->io_uring)) {
struct io_uring_task *tctx = current->io_uring;

if (unlikely(!tctx)) {
int ret;

ret = io_uring_alloc_task_context(current);
if (unlikely(ret))
return ret;
tctx = current->io_uring;
}
if (current->io_uring->last != file) {
XA_STATE(xas, &current->io_uring->xa, (unsigned long) file);
void *old;
if (tctx->last != file) {
void *old = xa_load(&tctx->xa, (unsigned long)file);

rcu_read_lock();
old = xas_load(&xas);
if (old != file) {
if (!old) {
get_file(file);
xas_lock(&xas);
xas_store(&xas, file);
xas_unlock(&xas);
xa_store(&tctx->xa, (unsigned long)file, file, GFP_KERNEL);
}
rcu_read_unlock();
current->io_uring->last = file;
tctx->last = file;
}

return 0;
Expand Down

0 comments on commit a05b5f7

Please sign in to comment.