| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293 |
- // SPDX-License-Identifier: GPL-2.0
- #include <linux/kernel.h>
- #include <linux/errno.h>
- #include <linux/fs.h>
- #include <linux/file.h>
- #include <linux/mm.h>
- #include <linux/slab.h>
- #include <linux/namei.h>
- #include <linux/io_uring.h>
- #include <uapi/linux/io_uring.h>
- #include "../fs/internal.h"
- #include "io_uring.h"
- #include "fs.h"
- struct io_rename {
- struct file *file;
- int old_dfd;
- int new_dfd;
- struct filename *oldpath;
- struct filename *newpath;
- int flags;
- };
- struct io_unlink {
- struct file *file;
- int dfd;
- int flags;
- struct filename *filename;
- };
- struct io_mkdir {
- struct file *file;
- int dfd;
- umode_t mode;
- struct filename *filename;
- };
- struct io_link {
- struct file *file;
- int old_dfd;
- int new_dfd;
- struct filename *oldpath;
- struct filename *newpath;
- int flags;
- };
- int io_renameat_prep(struct io_kiocb *req, const struct io_uring_sqe *sqe)
- {
- struct io_rename *ren = io_kiocb_to_cmd(req, struct io_rename);
- const char __user *oldf, *newf;
- if (sqe->buf_index || sqe->splice_fd_in)
- return -EINVAL;
- if (unlikely(req->flags & REQ_F_FIXED_FILE))
- return -EBADF;
- ren->old_dfd = READ_ONCE(sqe->fd);
- oldf = u64_to_user_ptr(READ_ONCE(sqe->addr));
- newf = u64_to_user_ptr(READ_ONCE(sqe->addr2));
- ren->new_dfd = READ_ONCE(sqe->len);
- ren->flags = READ_ONCE(sqe->rename_flags);
- ren->oldpath = getname(oldf);
- if (IS_ERR(ren->oldpath))
- return PTR_ERR(ren->oldpath);
- ren->newpath = getname(newf);
- if (IS_ERR(ren->newpath)) {
- putname(ren->oldpath);
- return PTR_ERR(ren->newpath);
- }
- req->flags |= REQ_F_NEED_CLEANUP;
- req->flags |= REQ_F_FORCE_ASYNC;
- return 0;
- }
- int io_renameat(struct io_kiocb *req, unsigned int issue_flags)
- {
- struct io_rename *ren = io_kiocb_to_cmd(req, struct io_rename);
- int ret;
- WARN_ON_ONCE(issue_flags & IO_URING_F_NONBLOCK);
- ret = do_renameat2(ren->old_dfd, ren->oldpath, ren->new_dfd,
- ren->newpath, ren->flags);
- req->flags &= ~REQ_F_NEED_CLEANUP;
- io_req_set_res(req, ret, 0);
- return IOU_OK;
- }
- void io_renameat_cleanup(struct io_kiocb *req)
- {
- struct io_rename *ren = io_kiocb_to_cmd(req, struct io_rename);
- putname(ren->oldpath);
- putname(ren->newpath);
- }
- int io_unlinkat_prep(struct io_kiocb *req, const struct io_uring_sqe *sqe)
- {
- struct io_unlink *un = io_kiocb_to_cmd(req, struct io_unlink);
- const char __user *fname;
- if (sqe->off || sqe->len || sqe->buf_index || sqe->splice_fd_in)
- return -EINVAL;
- if (unlikely(req->flags & REQ_F_FIXED_FILE))
- return -EBADF;
- un->dfd = READ_ONCE(sqe->fd);
- un->flags = READ_ONCE(sqe->unlink_flags);
- if (un->flags & ~AT_REMOVEDIR)
- return -EINVAL;
- fname = u64_to_user_ptr(READ_ONCE(sqe->addr));
- un->filename = getname(fname);
- if (IS_ERR(un->filename))
- return PTR_ERR(un->filename);
- req->flags |= REQ_F_NEED_CLEANUP;
- req->flags |= REQ_F_FORCE_ASYNC;
- return 0;
- }
- int io_unlinkat(struct io_kiocb *req, unsigned int issue_flags)
- {
- struct io_unlink *un = io_kiocb_to_cmd(req, struct io_unlink);
- int ret;
- WARN_ON_ONCE(issue_flags & IO_URING_F_NONBLOCK);
- if (un->flags & AT_REMOVEDIR)
- ret = do_rmdir(un->dfd, un->filename);
- else
- ret = do_unlinkat(un->dfd, un->filename);
- req->flags &= ~REQ_F_NEED_CLEANUP;
- io_req_set_res(req, ret, 0);
- return IOU_OK;
- }
- void io_unlinkat_cleanup(struct io_kiocb *req)
- {
- struct io_unlink *ul = io_kiocb_to_cmd(req, struct io_unlink);
- putname(ul->filename);
- }
- int io_mkdirat_prep(struct io_kiocb *req, const struct io_uring_sqe *sqe)
- {
- struct io_mkdir *mkd = io_kiocb_to_cmd(req, struct io_mkdir);
- const char __user *fname;
- if (sqe->off || sqe->rw_flags || sqe->buf_index || sqe->splice_fd_in)
- return -EINVAL;
- if (unlikely(req->flags & REQ_F_FIXED_FILE))
- return -EBADF;
- mkd->dfd = READ_ONCE(sqe->fd);
- mkd->mode = READ_ONCE(sqe->len);
- fname = u64_to_user_ptr(READ_ONCE(sqe->addr));
- mkd->filename = getname(fname);
- if (IS_ERR(mkd->filename))
- return PTR_ERR(mkd->filename);
- req->flags |= REQ_F_NEED_CLEANUP;
- req->flags |= REQ_F_FORCE_ASYNC;
- return 0;
- }
- int io_mkdirat(struct io_kiocb *req, unsigned int issue_flags)
- {
- struct io_mkdir *mkd = io_kiocb_to_cmd(req, struct io_mkdir);
- int ret;
- WARN_ON_ONCE(issue_flags & IO_URING_F_NONBLOCK);
- ret = do_mkdirat(mkd->dfd, mkd->filename, mkd->mode);
- req->flags &= ~REQ_F_NEED_CLEANUP;
- io_req_set_res(req, ret, 0);
- return IOU_OK;
- }
- void io_mkdirat_cleanup(struct io_kiocb *req)
- {
- struct io_mkdir *md = io_kiocb_to_cmd(req, struct io_mkdir);
- putname(md->filename);
- }
- int io_symlinkat_prep(struct io_kiocb *req, const struct io_uring_sqe *sqe)
- {
- struct io_link *sl = io_kiocb_to_cmd(req, struct io_link);
- const char __user *oldpath, *newpath;
- if (sqe->len || sqe->rw_flags || sqe->buf_index || sqe->splice_fd_in)
- return -EINVAL;
- if (unlikely(req->flags & REQ_F_FIXED_FILE))
- return -EBADF;
- sl->new_dfd = READ_ONCE(sqe->fd);
- oldpath = u64_to_user_ptr(READ_ONCE(sqe->addr));
- newpath = u64_to_user_ptr(READ_ONCE(sqe->addr2));
- sl->oldpath = getname(oldpath);
- if (IS_ERR(sl->oldpath))
- return PTR_ERR(sl->oldpath);
- sl->newpath = getname(newpath);
- if (IS_ERR(sl->newpath)) {
- putname(sl->oldpath);
- return PTR_ERR(sl->newpath);
- }
- req->flags |= REQ_F_NEED_CLEANUP;
- req->flags |= REQ_F_FORCE_ASYNC;
- return 0;
- }
- int io_symlinkat(struct io_kiocb *req, unsigned int issue_flags)
- {
- struct io_link *sl = io_kiocb_to_cmd(req, struct io_link);
- int ret;
- WARN_ON_ONCE(issue_flags & IO_URING_F_NONBLOCK);
- ret = do_symlinkat(sl->oldpath, sl->new_dfd, sl->newpath);
- req->flags &= ~REQ_F_NEED_CLEANUP;
- io_req_set_res(req, ret, 0);
- return IOU_OK;
- }
- int io_linkat_prep(struct io_kiocb *req, const struct io_uring_sqe *sqe)
- {
- struct io_link *lnk = io_kiocb_to_cmd(req, struct io_link);
- const char __user *oldf, *newf;
- if (sqe->buf_index || sqe->splice_fd_in)
- return -EINVAL;
- if (unlikely(req->flags & REQ_F_FIXED_FILE))
- return -EBADF;
- lnk->old_dfd = READ_ONCE(sqe->fd);
- lnk->new_dfd = READ_ONCE(sqe->len);
- oldf = u64_to_user_ptr(READ_ONCE(sqe->addr));
- newf = u64_to_user_ptr(READ_ONCE(sqe->addr2));
- lnk->flags = READ_ONCE(sqe->hardlink_flags);
- lnk->oldpath = getname_uflags(oldf, lnk->flags);
- if (IS_ERR(lnk->oldpath))
- return PTR_ERR(lnk->oldpath);
- lnk->newpath = getname(newf);
- if (IS_ERR(lnk->newpath)) {
- putname(lnk->oldpath);
- return PTR_ERR(lnk->newpath);
- }
- req->flags |= REQ_F_NEED_CLEANUP;
- req->flags |= REQ_F_FORCE_ASYNC;
- return 0;
- }
- int io_linkat(struct io_kiocb *req, unsigned int issue_flags)
- {
- struct io_link *lnk = io_kiocb_to_cmd(req, struct io_link);
- int ret;
- WARN_ON_ONCE(issue_flags & IO_URING_F_NONBLOCK);
- ret = do_linkat(lnk->old_dfd, lnk->oldpath, lnk->new_dfd,
- lnk->newpath, lnk->flags);
- req->flags &= ~REQ_F_NEED_CLEANUP;
- io_req_set_res(req, ret, 0);
- return IOU_OK;
- }
- void io_link_cleanup(struct io_kiocb *req)
- {
- struct io_link *sl = io_kiocb_to_cmd(req, struct io_link);
- putname(sl->oldpath);
- putname(sl->newpath);
- }
|