You've already forked linux-apfs
mirror of
https://github.com/linux-apfs/linux-apfs.git
synced 2026-05-01 15:00:59 -07:00
Merge branch 'iocb' into for-davem
trivial conflict in net/socket.c and non-trivial one in crypto - that one had evaded aio_complete() removal. Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
This commit is contained in:
@@ -21,7 +21,7 @@
|
|||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
#include <linux/mount.h>
|
#include <linux/mount.h>
|
||||||
#include <linux/aio.h>
|
#include <linux/uio.h>
|
||||||
#include <asm/ebcdic.h>
|
#include <asm/ebcdic.h>
|
||||||
#include "hypfs.h"
|
#include "hypfs.h"
|
||||||
|
|
||||||
|
|||||||
@@ -106,7 +106,7 @@ static void skcipher_async_cb(struct crypto_async_request *req, int err)
|
|||||||
atomic_dec(&ctx->inflight);
|
atomic_dec(&ctx->inflight);
|
||||||
skcipher_free_async_sgls(sreq);
|
skcipher_free_async_sgls(sreq);
|
||||||
kfree(req);
|
kfree(req);
|
||||||
aio_complete(iocb, err, err);
|
iocb->ki_complete(iocb, err, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int skcipher_sndbuf(struct sock *sk)
|
static inline int skcipher_sndbuf(struct sock *sk)
|
||||||
|
|||||||
+1
-1
@@ -26,7 +26,7 @@
|
|||||||
#include <linux/pfn.h>
|
#include <linux/pfn.h>
|
||||||
#include <linux/export.h>
|
#include <linux/export.h>
|
||||||
#include <linux/io.h>
|
#include <linux/io.h>
|
||||||
#include <linux/aio.h>
|
#include <linux/uio.h>
|
||||||
|
|
||||||
#include <linux/uaccess.h>
|
#include <linux/uaccess.h>
|
||||||
|
|
||||||
|
|||||||
@@ -27,7 +27,6 @@
|
|||||||
#include <linux/types.h> /* size_t */
|
#include <linux/types.h> /* size_t */
|
||||||
#include <linux/proc_fs.h>
|
#include <linux/proc_fs.h>
|
||||||
#include <linux/fcntl.h> /* O_ACCMODE */
|
#include <linux/fcntl.h> /* O_ACCMODE */
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/pagemap.h>
|
#include <linux/pagemap.h>
|
||||||
#include <linux/hugetlb.h>
|
#include <linux/hugetlb.h>
|
||||||
#include <linux/uaccess.h>
|
#include <linux/uaccess.h>
|
||||||
|
|||||||
@@ -40,7 +40,6 @@
|
|||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/highmem.h>
|
#include <linux/highmem.h>
|
||||||
#include <linux/io.h>
|
#include <linux/io.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/jiffies.h>
|
#include <linux/jiffies.h>
|
||||||
#include <linux/cpu.h>
|
#include <linux/cpu.h>
|
||||||
#include <asm/pgtable.h>
|
#include <asm/pgtable.h>
|
||||||
|
|||||||
@@ -39,7 +39,6 @@
|
|||||||
#include <linux/vmalloc.h>
|
#include <linux/vmalloc.h>
|
||||||
#include <linux/highmem.h>
|
#include <linux/highmem.h>
|
||||||
#include <linux/io.h>
|
#include <linux/io.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/jiffies.h>
|
#include <linux/jiffies.h>
|
||||||
#include <asm/pgtable.h>
|
#include <asm/pgtable.h>
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
|
|||||||
@@ -19,7 +19,6 @@
|
|||||||
#include <linux/errno.h>
|
#include <linux/errno.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
#include <linux/fcntl.h>
|
#include <linux/fcntl.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/ioctl.h>
|
#include <linux/ioctl.h>
|
||||||
#include <linux/cdev.h>
|
#include <linux/cdev.h>
|
||||||
#include <linux/list.h>
|
#include <linux/list.h>
|
||||||
|
|||||||
@@ -22,7 +22,6 @@
|
|||||||
#include <linux/errno.h>
|
#include <linux/errno.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
#include <linux/fcntl.h>
|
#include <linux/fcntl.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/poll.h>
|
#include <linux/poll.h>
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/ioctl.h>
|
#include <linux/ioctl.h>
|
||||||
|
|||||||
@@ -21,7 +21,6 @@
|
|||||||
#include <linux/errno.h>
|
#include <linux/errno.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
#include <linux/fcntl.h>
|
#include <linux/fcntl.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
#include <linux/poll.h>
|
#include <linux/poll.h>
|
||||||
#include <linux/ioctl.h>
|
#include <linux/ioctl.h>
|
||||||
|
|||||||
+1
-1
@@ -33,7 +33,6 @@ static int sg_version_num = 30536; /* 2 digits for each component */
|
|||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
#include <linux/mm.h>
|
#include <linux/mm.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/errno.h>
|
#include <linux/errno.h>
|
||||||
#include <linux/mtio.h>
|
#include <linux/mtio.h>
|
||||||
#include <linux/ioctl.h>
|
#include <linux/ioctl.h>
|
||||||
@@ -51,6 +50,7 @@ static int sg_version_num = 30536; /* 2 digits for each component */
|
|||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
#include <linux/atomic.h>
|
#include <linux/atomic.h>
|
||||||
#include <linux/ratelimit.h>
|
#include <linux/ratelimit.h>
|
||||||
|
#include <linux/uio.h>
|
||||||
|
|
||||||
#include "scsi.h"
|
#include "scsi.h"
|
||||||
#include <scsi/scsi_dbg.h>
|
#include <scsi/scsi_dbg.h>
|
||||||
|
|||||||
@@ -46,7 +46,6 @@
|
|||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/moduleparam.h>
|
#include <linux/moduleparam.h>
|
||||||
#include <linux/fcntl.h>
|
#include <linux/fcntl.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/workqueue.h>
|
#include <linux/workqueue.h>
|
||||||
#include <linux/kthread.h>
|
#include <linux/kthread.h>
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
|
|||||||
@@ -23,6 +23,7 @@
|
|||||||
#include <linux/export.h>
|
#include <linux/export.h>
|
||||||
#include <linux/hid.h>
|
#include <linux/hid.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
|
#include <linux/uio.h>
|
||||||
#include <asm/unaligned.h>
|
#include <asm/unaligned.h>
|
||||||
|
|
||||||
#include <linux/usb/composite.h>
|
#include <linux/usb/composite.h>
|
||||||
@@ -655,9 +656,10 @@ static void ffs_user_copy_worker(struct work_struct *work)
|
|||||||
unuse_mm(io_data->mm);
|
unuse_mm(io_data->mm);
|
||||||
}
|
}
|
||||||
|
|
||||||
aio_complete(io_data->kiocb, ret, ret);
|
io_data->kiocb->ki_complete(io_data->kiocb, ret, ret);
|
||||||
|
|
||||||
if (io_data->ffs->ffs_eventfd && !io_data->kiocb->ki_eventfd)
|
if (io_data->ffs->ffs_eventfd &&
|
||||||
|
!(io_data->kiocb->ki_flags & IOCB_EVENTFD))
|
||||||
eventfd_signal(io_data->ffs->ffs_eventfd, 1);
|
eventfd_signal(io_data->ffs->ffs_eventfd, 1);
|
||||||
|
|
||||||
usb_ep_free_request(io_data->ep, io_data->req);
|
usb_ep_free_request(io_data->ep, io_data->req);
|
||||||
|
|||||||
@@ -26,6 +26,7 @@
|
|||||||
#include <linux/poll.h>
|
#include <linux/poll.h>
|
||||||
#include <linux/mmu_context.h>
|
#include <linux/mmu_context.h>
|
||||||
#include <linux/aio.h>
|
#include <linux/aio.h>
|
||||||
|
#include <linux/uio.h>
|
||||||
|
|
||||||
#include <linux/device.h>
|
#include <linux/device.h>
|
||||||
#include <linux/moduleparam.h>
|
#include <linux/moduleparam.h>
|
||||||
@@ -469,7 +470,7 @@ static void ep_user_copy_worker(struct work_struct *work)
|
|||||||
ret = -EFAULT;
|
ret = -EFAULT;
|
||||||
|
|
||||||
/* completing the iocb can drop the ctx and mm, don't touch mm after */
|
/* completing the iocb can drop the ctx and mm, don't touch mm after */
|
||||||
aio_complete(iocb, ret, ret);
|
iocb->ki_complete(iocb, ret, ret);
|
||||||
|
|
||||||
kfree(priv->buf);
|
kfree(priv->buf);
|
||||||
kfree(priv->to_free);
|
kfree(priv->to_free);
|
||||||
@@ -497,7 +498,8 @@ static void ep_aio_complete(struct usb_ep *ep, struct usb_request *req)
|
|||||||
kfree(priv);
|
kfree(priv);
|
||||||
iocb->private = NULL;
|
iocb->private = NULL;
|
||||||
/* aio_complete() reports bytes-transferred _and_ faults */
|
/* aio_complete() reports bytes-transferred _and_ faults */
|
||||||
aio_complete(iocb, req->actual ? req->actual : req->status,
|
|
||||||
|
iocb->ki_complete(iocb, req->actual ? req->actual : req->status,
|
||||||
req->status);
|
req->status);
|
||||||
} else {
|
} else {
|
||||||
/* ep_copy_to_user() won't report both; we hide some faults */
|
/* ep_copy_to_user() won't report both; we hide some faults */
|
||||||
|
|||||||
+1
-1
@@ -33,7 +33,7 @@
|
|||||||
#include <linux/pagemap.h>
|
#include <linux/pagemap.h>
|
||||||
#include <linux/idr.h>
|
#include <linux/idr.h>
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
#include <linux/aio.h>
|
#include <linux/uio.h>
|
||||||
#include <net/9p/9p.h>
|
#include <net/9p/9p.h>
|
||||||
#include <net/9p/client.h>
|
#include <net/9p/client.h>
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -12,7 +12,7 @@
|
|||||||
* affs regular file handling primitives
|
* affs regular file handling primitives
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/aio.h>
|
#include <linux/uio.h>
|
||||||
#include "affs.h"
|
#include "affs.h"
|
||||||
|
|
||||||
static struct buffer_head *affs_get_extblock_slow(struct inode *inode, u32 ext);
|
static struct buffer_head *affs_get_extblock_slow(struct inode *inode, u32 ext);
|
||||||
|
|||||||
@@ -14,7 +14,6 @@
|
|||||||
#include <linux/pagemap.h>
|
#include <linux/pagemap.h>
|
||||||
#include <linux/writeback.h>
|
#include <linux/writeback.h>
|
||||||
#include <linux/pagevec.h>
|
#include <linux/pagevec.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include "internal.h"
|
#include "internal.h"
|
||||||
|
|
||||||
static int afs_write_back_from_locked_page(struct afs_writeback *wb,
|
static int afs_write_back_from_locked_page(struct afs_writeback *wb,
|
||||||
|
|||||||
@@ -151,6 +151,38 @@ struct kioctx {
|
|||||||
unsigned id;
|
unsigned id;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We use ki_cancel == KIOCB_CANCELLED to indicate that a kiocb has been either
|
||||||
|
* cancelled or completed (this makes a certain amount of sense because
|
||||||
|
* successful cancellation - io_cancel() - does deliver the completion to
|
||||||
|
* userspace).
|
||||||
|
*
|
||||||
|
* And since most things don't implement kiocb cancellation and we'd really like
|
||||||
|
* kiocb completion to be lockless when possible, we use ki_cancel to
|
||||||
|
* synchronize cancellation and completion - we only set it to KIOCB_CANCELLED
|
||||||
|
* with xchg() or cmpxchg(), see batch_complete_aio() and kiocb_cancel().
|
||||||
|
*/
|
||||||
|
#define KIOCB_CANCELLED ((void *) (~0ULL))
|
||||||
|
|
||||||
|
struct aio_kiocb {
|
||||||
|
struct kiocb common;
|
||||||
|
|
||||||
|
struct kioctx *ki_ctx;
|
||||||
|
kiocb_cancel_fn *ki_cancel;
|
||||||
|
|
||||||
|
struct iocb __user *ki_user_iocb; /* user's aiocb */
|
||||||
|
__u64 ki_user_data; /* user's data for completion */
|
||||||
|
|
||||||
|
struct list_head ki_list; /* the aio core uses this
|
||||||
|
* for cancellation */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the aio_resfd field of the userspace iocb is not zero,
|
||||||
|
* this is the underlying eventfd context to deliver events to.
|
||||||
|
*/
|
||||||
|
struct eventfd_ctx *ki_eventfd;
|
||||||
|
};
|
||||||
|
|
||||||
/*------ sysctl variables----*/
|
/*------ sysctl variables----*/
|
||||||
static DEFINE_SPINLOCK(aio_nr_lock);
|
static DEFINE_SPINLOCK(aio_nr_lock);
|
||||||
unsigned long aio_nr; /* current system wide number of aio requests */
|
unsigned long aio_nr; /* current system wide number of aio requests */
|
||||||
@@ -220,7 +252,7 @@ static int __init aio_setup(void)
|
|||||||
if (IS_ERR(aio_mnt))
|
if (IS_ERR(aio_mnt))
|
||||||
panic("Failed to create aio fs mount.");
|
panic("Failed to create aio fs mount.");
|
||||||
|
|
||||||
kiocb_cachep = KMEM_CACHE(kiocb, SLAB_HWCACHE_ALIGN|SLAB_PANIC);
|
kiocb_cachep = KMEM_CACHE(aio_kiocb, SLAB_HWCACHE_ALIGN|SLAB_PANIC);
|
||||||
kioctx_cachep = KMEM_CACHE(kioctx,SLAB_HWCACHE_ALIGN|SLAB_PANIC);
|
kioctx_cachep = KMEM_CACHE(kioctx,SLAB_HWCACHE_ALIGN|SLAB_PANIC);
|
||||||
|
|
||||||
pr_debug("sizeof(struct page) = %zu\n", sizeof(struct page));
|
pr_debug("sizeof(struct page) = %zu\n", sizeof(struct page));
|
||||||
@@ -480,8 +512,9 @@ static int aio_setup_ring(struct kioctx *ctx)
|
|||||||
#define AIO_EVENTS_FIRST_PAGE ((PAGE_SIZE - sizeof(struct aio_ring)) / sizeof(struct io_event))
|
#define AIO_EVENTS_FIRST_PAGE ((PAGE_SIZE - sizeof(struct aio_ring)) / sizeof(struct io_event))
|
||||||
#define AIO_EVENTS_OFFSET (AIO_EVENTS_PER_PAGE - AIO_EVENTS_FIRST_PAGE)
|
#define AIO_EVENTS_OFFSET (AIO_EVENTS_PER_PAGE - AIO_EVENTS_FIRST_PAGE)
|
||||||
|
|
||||||
void kiocb_set_cancel_fn(struct kiocb *req, kiocb_cancel_fn *cancel)
|
void kiocb_set_cancel_fn(struct kiocb *iocb, kiocb_cancel_fn *cancel)
|
||||||
{
|
{
|
||||||
|
struct aio_kiocb *req = container_of(iocb, struct aio_kiocb, common);
|
||||||
struct kioctx *ctx = req->ki_ctx;
|
struct kioctx *ctx = req->ki_ctx;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
@@ -496,7 +529,7 @@ void kiocb_set_cancel_fn(struct kiocb *req, kiocb_cancel_fn *cancel)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL(kiocb_set_cancel_fn);
|
EXPORT_SYMBOL(kiocb_set_cancel_fn);
|
||||||
|
|
||||||
static int kiocb_cancel(struct kiocb *kiocb)
|
static int kiocb_cancel(struct aio_kiocb *kiocb)
|
||||||
{
|
{
|
||||||
kiocb_cancel_fn *old, *cancel;
|
kiocb_cancel_fn *old, *cancel;
|
||||||
|
|
||||||
@@ -514,7 +547,7 @@ static int kiocb_cancel(struct kiocb *kiocb)
|
|||||||
cancel = cmpxchg(&kiocb->ki_cancel, old, KIOCB_CANCELLED);
|
cancel = cmpxchg(&kiocb->ki_cancel, old, KIOCB_CANCELLED);
|
||||||
} while (cancel != old);
|
} while (cancel != old);
|
||||||
|
|
||||||
return cancel(kiocb);
|
return cancel(&kiocb->common);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void free_ioctx(struct work_struct *work)
|
static void free_ioctx(struct work_struct *work)
|
||||||
@@ -550,13 +583,13 @@ static void free_ioctx_reqs(struct percpu_ref *ref)
|
|||||||
static void free_ioctx_users(struct percpu_ref *ref)
|
static void free_ioctx_users(struct percpu_ref *ref)
|
||||||
{
|
{
|
||||||
struct kioctx *ctx = container_of(ref, struct kioctx, users);
|
struct kioctx *ctx = container_of(ref, struct kioctx, users);
|
||||||
struct kiocb *req;
|
struct aio_kiocb *req;
|
||||||
|
|
||||||
spin_lock_irq(&ctx->ctx_lock);
|
spin_lock_irq(&ctx->ctx_lock);
|
||||||
|
|
||||||
while (!list_empty(&ctx->active_reqs)) {
|
while (!list_empty(&ctx->active_reqs)) {
|
||||||
req = list_first_entry(&ctx->active_reqs,
|
req = list_first_entry(&ctx->active_reqs,
|
||||||
struct kiocb, ki_list);
|
struct aio_kiocb, ki_list);
|
||||||
|
|
||||||
list_del_init(&req->ki_list);
|
list_del_init(&req->ki_list);
|
||||||
kiocb_cancel(req);
|
kiocb_cancel(req);
|
||||||
@@ -778,22 +811,6 @@ static int kill_ioctx(struct mm_struct *mm, struct kioctx *ctx,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* wait_on_sync_kiocb:
|
|
||||||
* Waits on the given sync kiocb to complete.
|
|
||||||
*/
|
|
||||||
ssize_t wait_on_sync_kiocb(struct kiocb *req)
|
|
||||||
{
|
|
||||||
while (!req->ki_ctx) {
|
|
||||||
set_current_state(TASK_UNINTERRUPTIBLE);
|
|
||||||
if (req->ki_ctx)
|
|
||||||
break;
|
|
||||||
io_schedule();
|
|
||||||
}
|
|
||||||
__set_current_state(TASK_RUNNING);
|
|
||||||
return req->ki_user_data;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL(wait_on_sync_kiocb);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* exit_aio: called when the last user of mm goes away. At this point, there is
|
* exit_aio: called when the last user of mm goes away. At this point, there is
|
||||||
* no way for any new requests to be submited or any of the io_* syscalls to be
|
* no way for any new requests to be submited or any of the io_* syscalls to be
|
||||||
@@ -948,9 +965,9 @@ static void user_refill_reqs_available(struct kioctx *ctx)
|
|||||||
* Allocate a slot for an aio request.
|
* Allocate a slot for an aio request.
|
||||||
* Returns NULL if no requests are free.
|
* Returns NULL if no requests are free.
|
||||||
*/
|
*/
|
||||||
static inline struct kiocb *aio_get_req(struct kioctx *ctx)
|
static inline struct aio_kiocb *aio_get_req(struct kioctx *ctx)
|
||||||
{
|
{
|
||||||
struct kiocb *req;
|
struct aio_kiocb *req;
|
||||||
|
|
||||||
if (!get_reqs_available(ctx)) {
|
if (!get_reqs_available(ctx)) {
|
||||||
user_refill_reqs_available(ctx);
|
user_refill_reqs_available(ctx);
|
||||||
@@ -971,10 +988,10 @@ out_put:
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void kiocb_free(struct kiocb *req)
|
static void kiocb_free(struct aio_kiocb *req)
|
||||||
{
|
{
|
||||||
if (req->ki_filp)
|
if (req->common.ki_filp)
|
||||||
fput(req->ki_filp);
|
fput(req->common.ki_filp);
|
||||||
if (req->ki_eventfd != NULL)
|
if (req->ki_eventfd != NULL)
|
||||||
eventfd_ctx_put(req->ki_eventfd);
|
eventfd_ctx_put(req->ki_eventfd);
|
||||||
kmem_cache_free(kiocb_cachep, req);
|
kmem_cache_free(kiocb_cachep, req);
|
||||||
@@ -1010,8 +1027,9 @@ out:
|
|||||||
/* aio_complete
|
/* aio_complete
|
||||||
* Called when the io request on the given iocb is complete.
|
* Called when the io request on the given iocb is complete.
|
||||||
*/
|
*/
|
||||||
void aio_complete(struct kiocb *iocb, long res, long res2)
|
static void aio_complete(struct kiocb *kiocb, long res, long res2)
|
||||||
{
|
{
|
||||||
|
struct aio_kiocb *iocb = container_of(kiocb, struct aio_kiocb, common);
|
||||||
struct kioctx *ctx = iocb->ki_ctx;
|
struct kioctx *ctx = iocb->ki_ctx;
|
||||||
struct aio_ring *ring;
|
struct aio_ring *ring;
|
||||||
struct io_event *ev_page, *event;
|
struct io_event *ev_page, *event;
|
||||||
@@ -1025,13 +1043,7 @@ void aio_complete(struct kiocb *iocb, long res, long res2)
|
|||||||
* ref, no other paths have a way to get another ref
|
* ref, no other paths have a way to get another ref
|
||||||
* - the sync task helpfully left a reference to itself in the iocb
|
* - the sync task helpfully left a reference to itself in the iocb
|
||||||
*/
|
*/
|
||||||
if (is_sync_kiocb(iocb)) {
|
BUG_ON(is_sync_kiocb(kiocb));
|
||||||
iocb->ki_user_data = res;
|
|
||||||
smp_wmb();
|
|
||||||
iocb->ki_ctx = ERR_PTR(-EXDEV);
|
|
||||||
wake_up_process(iocb->ki_obj.tsk);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (iocb->ki_list.next) {
|
if (iocb->ki_list.next) {
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
@@ -1057,7 +1069,7 @@ void aio_complete(struct kiocb *iocb, long res, long res2)
|
|||||||
ev_page = kmap_atomic(ctx->ring_pages[pos / AIO_EVENTS_PER_PAGE]);
|
ev_page = kmap_atomic(ctx->ring_pages[pos / AIO_EVENTS_PER_PAGE]);
|
||||||
event = ev_page + pos % AIO_EVENTS_PER_PAGE;
|
event = ev_page + pos % AIO_EVENTS_PER_PAGE;
|
||||||
|
|
||||||
event->obj = (u64)(unsigned long)iocb->ki_obj.user;
|
event->obj = (u64)(unsigned long)iocb->ki_user_iocb;
|
||||||
event->data = iocb->ki_user_data;
|
event->data = iocb->ki_user_data;
|
||||||
event->res = res;
|
event->res = res;
|
||||||
event->res2 = res2;
|
event->res2 = res2;
|
||||||
@@ -1066,7 +1078,7 @@ void aio_complete(struct kiocb *iocb, long res, long res2)
|
|||||||
flush_dcache_page(ctx->ring_pages[pos / AIO_EVENTS_PER_PAGE]);
|
flush_dcache_page(ctx->ring_pages[pos / AIO_EVENTS_PER_PAGE]);
|
||||||
|
|
||||||
pr_debug("%p[%u]: %p: %p %Lx %lx %lx\n",
|
pr_debug("%p[%u]: %p: %p %Lx %lx %lx\n",
|
||||||
ctx, tail, iocb, iocb->ki_obj.user, iocb->ki_user_data,
|
ctx, tail, iocb, iocb->ki_user_iocb, iocb->ki_user_data,
|
||||||
res, res2);
|
res, res2);
|
||||||
|
|
||||||
/* after flagging the request as done, we
|
/* after flagging the request as done, we
|
||||||
@@ -1113,7 +1125,6 @@ void aio_complete(struct kiocb *iocb, long res, long res2)
|
|||||||
|
|
||||||
percpu_ref_put(&ctx->reqs);
|
percpu_ref_put(&ctx->reqs);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(aio_complete);
|
|
||||||
|
|
||||||
/* aio_read_events_ring
|
/* aio_read_events_ring
|
||||||
* Pull an event off of the ioctx's event ring. Returns the number of
|
* Pull an event off of the ioctx's event ring. Returns the number of
|
||||||
@@ -1344,12 +1355,13 @@ typedef ssize_t (rw_iter_op)(struct kiocb *, struct iov_iter *);
|
|||||||
static ssize_t aio_setup_vectored_rw(struct kiocb *kiocb,
|
static ssize_t aio_setup_vectored_rw(struct kiocb *kiocb,
|
||||||
int rw, char __user *buf,
|
int rw, char __user *buf,
|
||||||
unsigned long *nr_segs,
|
unsigned long *nr_segs,
|
||||||
|
size_t *len,
|
||||||
struct iovec **iovec,
|
struct iovec **iovec,
|
||||||
bool compat)
|
bool compat)
|
||||||
{
|
{
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
|
|
||||||
*nr_segs = kiocb->ki_nbytes;
|
*nr_segs = *len;
|
||||||
|
|
||||||
#ifdef CONFIG_COMPAT
|
#ifdef CONFIG_COMPAT
|
||||||
if (compat)
|
if (compat)
|
||||||
@@ -1364,21 +1376,22 @@ static ssize_t aio_setup_vectored_rw(struct kiocb *kiocb,
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
/* ki_nbytes now reflect bytes instead of segs */
|
/* len now reflect bytes instead of segs */
|
||||||
kiocb->ki_nbytes = ret;
|
*len = ret;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t aio_setup_single_vector(struct kiocb *kiocb,
|
static ssize_t aio_setup_single_vector(struct kiocb *kiocb,
|
||||||
int rw, char __user *buf,
|
int rw, char __user *buf,
|
||||||
unsigned long *nr_segs,
|
unsigned long *nr_segs,
|
||||||
|
size_t len,
|
||||||
struct iovec *iovec)
|
struct iovec *iovec)
|
||||||
{
|
{
|
||||||
if (unlikely(!access_ok(!rw, buf, kiocb->ki_nbytes)))
|
if (unlikely(!access_ok(!rw, buf, len)))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
iovec->iov_base = buf;
|
iovec->iov_base = buf;
|
||||||
iovec->iov_len = kiocb->ki_nbytes;
|
iovec->iov_len = len;
|
||||||
*nr_segs = 1;
|
*nr_segs = 1;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -1388,7 +1401,7 @@ static ssize_t aio_setup_single_vector(struct kiocb *kiocb,
|
|||||||
* Performs the initial checks and io submission.
|
* Performs the initial checks and io submission.
|
||||||
*/
|
*/
|
||||||
static ssize_t aio_run_iocb(struct kiocb *req, unsigned opcode,
|
static ssize_t aio_run_iocb(struct kiocb *req, unsigned opcode,
|
||||||
char __user *buf, bool compat)
|
char __user *buf, size_t len, bool compat)
|
||||||
{
|
{
|
||||||
struct file *file = req->ki_filp;
|
struct file *file = req->ki_filp;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
@@ -1423,21 +1436,21 @@ rw_common:
|
|||||||
if (!rw_op && !iter_op)
|
if (!rw_op && !iter_op)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
ret = (opcode == IOCB_CMD_PREADV ||
|
if (opcode == IOCB_CMD_PREADV || opcode == IOCB_CMD_PWRITEV)
|
||||||
opcode == IOCB_CMD_PWRITEV)
|
ret = aio_setup_vectored_rw(req, rw, buf, &nr_segs,
|
||||||
? aio_setup_vectored_rw(req, rw, buf, &nr_segs,
|
&len, &iovec, compat);
|
||||||
&iovec, compat)
|
else
|
||||||
: aio_setup_single_vector(req, rw, buf, &nr_segs,
|
ret = aio_setup_single_vector(req, rw, buf, &nr_segs,
|
||||||
iovec);
|
len, iovec);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
ret = rw_verify_area(rw, file, &req->ki_pos, req->ki_nbytes);
|
ret = rw_verify_area(rw, file, &req->ki_pos, len);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
if (iovec != inline_vecs)
|
if (iovec != inline_vecs)
|
||||||
kfree(iovec);
|
kfree(iovec);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
req->ki_nbytes = ret;
|
len = ret;
|
||||||
|
|
||||||
/* XXX: move/kill - rw_verify_area()? */
|
/* XXX: move/kill - rw_verify_area()? */
|
||||||
/* This matches the pread()/pwrite() logic */
|
/* This matches the pread()/pwrite() logic */
|
||||||
@@ -1450,7 +1463,7 @@ rw_common:
|
|||||||
file_start_write(file);
|
file_start_write(file);
|
||||||
|
|
||||||
if (iter_op) {
|
if (iter_op) {
|
||||||
iov_iter_init(&iter, rw, iovec, nr_segs, req->ki_nbytes);
|
iov_iter_init(&iter, rw, iovec, nr_segs, len);
|
||||||
ret = iter_op(req, &iter);
|
ret = iter_op(req, &iter);
|
||||||
} else {
|
} else {
|
||||||
ret = rw_op(req, iovec, nr_segs, req->ki_pos);
|
ret = rw_op(req, iovec, nr_segs, req->ki_pos);
|
||||||
@@ -1500,7 +1513,7 @@ rw_common:
|
|||||||
static int io_submit_one(struct kioctx *ctx, struct iocb __user *user_iocb,
|
static int io_submit_one(struct kioctx *ctx, struct iocb __user *user_iocb,
|
||||||
struct iocb *iocb, bool compat)
|
struct iocb *iocb, bool compat)
|
||||||
{
|
{
|
||||||
struct kiocb *req;
|
struct aio_kiocb *req;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
|
|
||||||
/* enforce forwards compatibility on users */
|
/* enforce forwards compatibility on users */
|
||||||
@@ -1523,11 +1536,14 @@ static int io_submit_one(struct kioctx *ctx, struct iocb __user *user_iocb,
|
|||||||
if (unlikely(!req))
|
if (unlikely(!req))
|
||||||
return -EAGAIN;
|
return -EAGAIN;
|
||||||
|
|
||||||
req->ki_filp = fget(iocb->aio_fildes);
|
req->common.ki_filp = fget(iocb->aio_fildes);
|
||||||
if (unlikely(!req->ki_filp)) {
|
if (unlikely(!req->common.ki_filp)) {
|
||||||
ret = -EBADF;
|
ret = -EBADF;
|
||||||
goto out_put_req;
|
goto out_put_req;
|
||||||
}
|
}
|
||||||
|
req->common.ki_pos = iocb->aio_offset;
|
||||||
|
req->common.ki_complete = aio_complete;
|
||||||
|
req->common.ki_flags = 0;
|
||||||
|
|
||||||
if (iocb->aio_flags & IOCB_FLAG_RESFD) {
|
if (iocb->aio_flags & IOCB_FLAG_RESFD) {
|
||||||
/*
|
/*
|
||||||
@@ -1542,6 +1558,8 @@ static int io_submit_one(struct kioctx *ctx, struct iocb __user *user_iocb,
|
|||||||
req->ki_eventfd = NULL;
|
req->ki_eventfd = NULL;
|
||||||
goto out_put_req;
|
goto out_put_req;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
req->common.ki_flags |= IOCB_EVENTFD;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = put_user(KIOCB_KEY, &user_iocb->aio_key);
|
ret = put_user(KIOCB_KEY, &user_iocb->aio_key);
|
||||||
@@ -1550,13 +1568,12 @@ static int io_submit_one(struct kioctx *ctx, struct iocb __user *user_iocb,
|
|||||||
goto out_put_req;
|
goto out_put_req;
|
||||||
}
|
}
|
||||||
|
|
||||||
req->ki_obj.user = user_iocb;
|
req->ki_user_iocb = user_iocb;
|
||||||
req->ki_user_data = iocb->aio_data;
|
req->ki_user_data = iocb->aio_data;
|
||||||
req->ki_pos = iocb->aio_offset;
|
|
||||||
req->ki_nbytes = iocb->aio_nbytes;
|
|
||||||
|
|
||||||
ret = aio_run_iocb(req, iocb->aio_lio_opcode,
|
ret = aio_run_iocb(&req->common, iocb->aio_lio_opcode,
|
||||||
(char __user *)(unsigned long)iocb->aio_buf,
|
(char __user *)(unsigned long)iocb->aio_buf,
|
||||||
|
iocb->aio_nbytes,
|
||||||
compat);
|
compat);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto out_put_req;
|
goto out_put_req;
|
||||||
@@ -1643,10 +1660,10 @@ SYSCALL_DEFINE3(io_submit, aio_context_t, ctx_id, long, nr,
|
|||||||
/* lookup_kiocb
|
/* lookup_kiocb
|
||||||
* Finds a given iocb for cancellation.
|
* Finds a given iocb for cancellation.
|
||||||
*/
|
*/
|
||||||
static struct kiocb *lookup_kiocb(struct kioctx *ctx, struct iocb __user *iocb,
|
static struct aio_kiocb *
|
||||||
u32 key)
|
lookup_kiocb(struct kioctx *ctx, struct iocb __user *iocb, u32 key)
|
||||||
{
|
{
|
||||||
struct list_head *pos;
|
struct aio_kiocb *kiocb;
|
||||||
|
|
||||||
assert_spin_locked(&ctx->ctx_lock);
|
assert_spin_locked(&ctx->ctx_lock);
|
||||||
|
|
||||||
@@ -1654,9 +1671,8 @@ static struct kiocb *lookup_kiocb(struct kioctx *ctx, struct iocb __user *iocb,
|
|||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
/* TODO: use a hash or array, this sucks. */
|
/* TODO: use a hash or array, this sucks. */
|
||||||
list_for_each(pos, &ctx->active_reqs) {
|
list_for_each_entry(kiocb, &ctx->active_reqs, ki_list) {
|
||||||
struct kiocb *kiocb = list_kiocb(pos);
|
if (kiocb->ki_user_iocb == iocb)
|
||||||
if (kiocb->ki_obj.user == iocb)
|
|
||||||
return kiocb;
|
return kiocb;
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
@@ -1676,7 +1692,7 @@ SYSCALL_DEFINE3(io_cancel, aio_context_t, ctx_id, struct iocb __user *, iocb,
|
|||||||
struct io_event __user *, result)
|
struct io_event __user *, result)
|
||||||
{
|
{
|
||||||
struct kioctx *ctx;
|
struct kioctx *ctx;
|
||||||
struct kiocb *kiocb;
|
struct aio_kiocb *kiocb;
|
||||||
u32 key;
|
u32 key;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
|||||||
@@ -15,6 +15,7 @@
|
|||||||
#include <linux/buffer_head.h>
|
#include <linux/buffer_head.h>
|
||||||
#include <linux/vfs.h>
|
#include <linux/vfs.h>
|
||||||
#include <linux/writeback.h>
|
#include <linux/writeback.h>
|
||||||
|
#include <linux/uio.h>
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
#include "bfs.h"
|
#include "bfs.h"
|
||||||
|
|
||||||
|
|||||||
@@ -27,7 +27,6 @@
|
|||||||
#include <linux/namei.h>
|
#include <linux/namei.h>
|
||||||
#include <linux/log2.h>
|
#include <linux/log2.h>
|
||||||
#include <linux/cleancache.h>
|
#include <linux/cleancache.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <asm/uaccess.h>
|
#include <asm/uaccess.h>
|
||||||
#include "internal.h"
|
#include "internal.h"
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -24,7 +24,6 @@
|
|||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
#include <linux/backing-dev.h>
|
#include <linux/backing-dev.h>
|
||||||
#include <linux/mpage.h>
|
#include <linux/mpage.h>
|
||||||
#include <linux/aio.h>
|
|
||||||
#include <linux/falloc.h>
|
#include <linux/falloc.h>
|
||||||
#include <linux/swap.h>
|
#include <linux/swap.h>
|
||||||
#include <linux/writeback.h>
|
#include <linux/writeback.h>
|
||||||
@@ -32,6 +31,7 @@
|
|||||||
#include <linux/compat.h>
|
#include <linux/compat.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/btrfs.h>
|
#include <linux/btrfs.h>
|
||||||
|
#include <linux/uio.h>
|
||||||
#include "ctree.h"
|
#include "ctree.h"
|
||||||
#include "disk-io.h"
|
#include "disk-io.h"
|
||||||
#include "transaction.h"
|
#include "transaction.h"
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user