diff options
author | aliguori <aliguori@c046a42c-6fe2-441c-8c8c-71466251a162> | 2008-09-22 19:17:18 +0000 |
---|---|---|
committer | aliguori <aliguori@c046a42c-6fe2-441c-8c8c-71466251a162> | 2008-09-22 19:17:18 +0000 |
commit | a76bab4952a1539266490295fb50b78802c467c2 (patch) | |
tree | ae411e977a38d038186d96cee86affe3e04e1d57 /block-raw-posix.c | |
parent | 1121f87961e87563df65134b15db05c0c2cc67da (diff) |
Refactor AIO to allow multiple AIO implementations
This patch refactors the AIO layer to allow multiple AIO implementations. It's
only possible because of the recent signalfd() patch.
Right now, the AIO infrastructure is pretty specific to the block raw backend.
For other block devices to implement AIO, the qemu_aio_wait function must
support registration. This patch introduces a new function,
qemu_aio_set_fd_handler, which can be used to register a file descriptor to be
called back. qemu_aio_wait() now polls a set of file descriptors registered
with this function until one becomes readable or writable.
This patch should allow the implementation of alternative AIO backends (via a
thread pool or linux-aio) and AIO backends in non-traditional block devices
(like NBD).
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@5297 c046a42c-6fe2-441c-8c8c-71466251a162
Diffstat (limited to 'block-raw-posix.c')
-rw-r--r-- | block-raw-posix.c | 109 |
1 files changed, 44 insertions, 65 deletions
diff --git a/block-raw-posix.c b/block-raw-posix.c index 26819a4d8c..41f997686f 100644 --- a/block-raw-posix.c +++ b/block-raw-posix.c @@ -101,6 +101,8 @@ typedef struct BDRVRawState { #endif } BDRVRawState; +static int posix_aio_init(void); + static int fd_open(BlockDriverState *bs); static int raw_open(BlockDriverState *bs, const char *filename, int flags) @@ -108,6 +110,8 @@ static int raw_open(BlockDriverState *bs, const char *filename, int flags) BDRVRawState *s = bs->opaque; int fd, open_flags, ret; + posix_aio_init(); + s->lseek_err_cnt = 0; open_flags = O_BINARY; @@ -437,13 +441,15 @@ typedef struct RawAIOCB { int ret; } RawAIOCB; -static int aio_sig_fd = -1; -static int aio_sig_num = SIGUSR2; -static RawAIOCB *first_aio; /* AIO issued */ -static int aio_initialized = 0; +typedef struct PosixAioState +{ + int fd; + RawAIOCB *first_aio; +} PosixAioState; -static void qemu_aio_poll(void *opaque) +static void posix_aio_read(void *opaque) { + PosixAioState *s = opaque; RawAIOCB *acb, **pacb; int ret; size_t offset; @@ -457,7 +463,7 @@ static void qemu_aio_poll(void *opaque) while (offset < 128) { ssize_t len; - len = read(aio_sig_fd, sig.buf + offset, 128 - offset); + len = read(s->fd, sig.buf + offset, 128 - offset); if (len == -1 && errno == EINTR) continue; if (len == -1 && errno == EAGAIN) { @@ -472,7 +478,7 @@ static void qemu_aio_poll(void *opaque) } for(;;) { - pacb = &first_aio; + pacb = &s->first_aio; for(;;) { acb = *pacb; if (!acb) @@ -507,25 +513,37 @@ static void qemu_aio_poll(void *opaque) the_end: ; } -void qemu_aio_init(void) +static int posix_aio_flush(void *opaque) { - sigset_t mask; + PosixAioState *s = opaque; + return !!s->first_aio; +} - if (aio_initialized) - return; +static PosixAioState *posix_aio_state; - aio_initialized = 1; +static int posix_aio_init(void) +{ + sigset_t mask; + PosixAioState *s; + + if (posix_aio_state) + return 0; + + s = qemu_malloc(sizeof(PosixAioState)); + if (s == NULL) + return -ENOMEM; /* Make sure to block AIO signal */ sigemptyset(&mask); - sigaddset(&mask, aio_sig_num); + sigaddset(&mask, SIGUSR2); sigprocmask(SIG_BLOCK, &mask, NULL); - aio_sig_fd = qemu_signalfd(&mask); + s->first_aio = NULL; + s->fd = qemu_signalfd(&mask); - fcntl(aio_sig_fd, F_SETFL, O_NONBLOCK); + fcntl(s->fd, F_SETFL, O_NONBLOCK); - qemu_set_fd_handler2(aio_sig_fd, NULL, qemu_aio_poll, NULL, NULL); + qemu_aio_set_fd_handler(s->fd, posix_aio_read, NULL, posix_aio_flush, s); #if defined(__GLIBC__) && defined(__linux__) { @@ -539,39 +557,9 @@ void qemu_aio_init(void) aio_init(&ai); } #endif -} - -/* Wait for all IO requests to complete. */ -void qemu_aio_flush(void) -{ - qemu_aio_poll(NULL); - while (first_aio) { - qemu_aio_wait(); - } -} - -void qemu_aio_wait(void) -{ - int ret; - - if (qemu_bh_poll()) - return; - - if (!first_aio) - return; - - do { - fd_set rdfds; - - FD_ZERO(&rdfds); - FD_SET(aio_sig_fd, &rdfds); + posix_aio_state = s; - ret = select(aio_sig_fd + 1, &rdfds, NULL, NULL, NULL); - if (ret == -1 && errno == EINTR) - continue; - } while (ret == 0); - - qemu_aio_poll(NULL); + return 0; } static RawAIOCB *raw_aio_setup(BlockDriverState *bs, @@ -588,7 +576,7 @@ static RawAIOCB *raw_aio_setup(BlockDriverState *bs, if (!acb) return NULL; acb->aiocb.aio_fildes = s->fd; - acb->aiocb.aio_sigevent.sigev_signo = aio_sig_num; + acb->aiocb.aio_sigevent.sigev_signo = SIGUSR2; acb->aiocb.aio_sigevent.sigev_notify = SIGEV_SIGNAL; acb->aiocb.aio_buf = buf; if (nb_sectors < 0) @@ -596,8 +584,8 @@ static RawAIOCB *raw_aio_setup(BlockDriverState *bs, else acb->aiocb.aio_nbytes = nb_sectors * 512; acb->aiocb.aio_offset = sector_num * 512; - acb->next = first_aio; - first_aio = acb; + acb->next = posix_aio_state->first_aio; + posix_aio_state->first_aio = acb; return acb; } @@ -688,7 +676,7 @@ static void raw_aio_cancel(BlockDriverAIOCB *blockacb) } /* remove the callback from the queue */ - pacb = &first_aio; + pacb = &posix_aio_state->first_aio; for(;;) { if (*pacb == NULL) { break; @@ -701,21 +689,10 @@ static void raw_aio_cancel(BlockDriverAIOCB *blockacb) } } -# else /* CONFIG_AIO */ - -void qemu_aio_init(void) +#else /* CONFIG_AIO */ +static int posix_aio_init(void) { } - -void qemu_aio_flush(void) -{ -} - -void qemu_aio_wait(void) -{ - qemu_bh_poll(); -} - #endif /* CONFIG_AIO */ static void raw_close(BlockDriverState *bs) @@ -921,6 +898,8 @@ static int hdev_open(BlockDriverState *bs, const char *filename, int flags) BDRVRawState *s = bs->opaque; int fd, open_flags, ret; + posix_aio_init(); + #ifdef CONFIG_COCOA if (strstart(filename, "/dev/cdrom", NULL)) { kern_return_t kernResult; |