mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-02 07:13:54 -06:00
Block pull request
-----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQEcBAABAgAGBQJTIbe5AAoJEJykq7OBq3PI0/AH/jglW+rsN+0/kAr5wIY4BQUn qvzYHBP9xvPtd0cGubTt2N3bZPlb/zTVFe/OOpJmaENrVJY/rPZNF3nKqdrr5uOP WrBbQZi2hkOG82uuiN2BiGh324zOMBNSh3d2WaHmJpxjabZcjN28UOkx5WGlBImu VAHY/HaImOLMkS120GxbMsFziXRLM66o9XWEbA/l0EBOISM6KloX6mtZd7y6MGzA a35JbUueyRo+hvndFEHY2P2XvUc5ZycOYF2yOz59X5QrqhmP6I+Ym4ohtyO2Olga jpoAO7jDpV/7PJwo1xC8j+1jhl3VGdX2L7iWgDUTJ0jY8mV3hQWhy58fOEwYY+k= =0llk -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/stefanha/tags/block-pull-request' into staging Block pull request # gpg: Signature made Thu 13 Mar 2014 13:50:49 GMT using RSA key ID 81AB73C8 # gpg: Good signature from "Stefan Hajnoczi <stefanha@redhat.com>" # gpg: aka "Stefan Hajnoczi <stefanha@gmail.com>" # gpg: WARNING: This key is not certified with a trusted signature! # gpg: There is no indication that the signature belongs to the owner. # Primary key fingerprint: 8695 A8BF D3F9 7CDA AC35 775A 9CA4 ABB3 81AB 73C8 * remotes/stefanha/tags/block-pull-request: (24 commits) block/raw-win32: bdrv_parse_filename() for hdev block/raw-posix: Strip protocol prefix on creation block/raw-posix: bdrv_parse_filename() for cdrom block/raw-posix: bdrv_parse_filename() for floppy block/raw-posix: bdrv_parse_filename() for hdev qemu-io: Fix warnings from static code analysis block: Unlink temporary file qcow2: Don't write with BDRV_O_INCOMING qcow2: Keep option in qcow2_invalidate_cache() qmp: add query-iothreads command iothread: stash thread ID away dataplane: replace internal thread with IOThread iothread: add "iothread" qdev property type qdev: make get_pointer() handle temporary strings iothread: add I/O thread object aio: add aio_context_acquire() and aio_context_release() rfifolock: add recursive FIFO lock object: add object_get_canonical_path_component() block: Rewrite the snapshot authorization mechanism for block filters. iotests: Test corruption during COW request ... Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
57fac92c2d
35 changed files with 943 additions and 138 deletions
|
@ -14,3 +14,4 @@ util-obj-y += crc32c.o
|
|||
util-obj-y += throttle.o
|
||||
util-obj-y += getauxval.o
|
||||
util-obj-y += readline.o
|
||||
util-obj-y += rfifolock.o
|
||||
|
|
78
util/rfifolock.c
Normal file
78
util/rfifolock.c
Normal file
|
@ -0,0 +1,78 @@
|
|||
/*
|
||||
* Recursive FIFO lock
|
||||
*
|
||||
* Copyright Red Hat, Inc. 2013
|
||||
*
|
||||
* Authors:
|
||||
* Stefan Hajnoczi <stefanha@redhat.com>
|
||||
*
|
||||
* This work is licensed under the terms of the GNU LGPL, version 2 or later.
|
||||
* See the COPYING.LIB file in the top-level directory.
|
||||
*
|
||||
*/
|
||||
|
||||
#include <assert.h>
|
||||
#include "qemu/rfifolock.h"
|
||||
|
||||
void rfifolock_init(RFifoLock *r, void (*cb)(void *), void *opaque)
|
||||
{
|
||||
qemu_mutex_init(&r->lock);
|
||||
r->head = 0;
|
||||
r->tail = 0;
|
||||
qemu_cond_init(&r->cond);
|
||||
r->nesting = 0;
|
||||
r->cb = cb;
|
||||
r->cb_opaque = opaque;
|
||||
}
|
||||
|
||||
void rfifolock_destroy(RFifoLock *r)
|
||||
{
|
||||
qemu_cond_destroy(&r->cond);
|
||||
qemu_mutex_destroy(&r->lock);
|
||||
}
|
||||
|
||||
/*
|
||||
* Theory of operation:
|
||||
*
|
||||
* In order to ensure FIFO ordering, implement a ticketlock. Threads acquiring
|
||||
* the lock enqueue themselves by incrementing the tail index. When the lock
|
||||
* is unlocked, the head is incremented and waiting threads are notified.
|
||||
*
|
||||
* Recursive locking does not take a ticket since the head is only incremented
|
||||
* when the outermost recursive caller unlocks.
|
||||
*/
|
||||
void rfifolock_lock(RFifoLock *r)
|
||||
{
|
||||
qemu_mutex_lock(&r->lock);
|
||||
|
||||
/* Take a ticket */
|
||||
unsigned int ticket = r->tail++;
|
||||
|
||||
if (r->nesting > 0 && qemu_thread_is_self(&r->owner_thread)) {
|
||||
r->tail--; /* put ticket back, we're nesting */
|
||||
} else {
|
||||
while (ticket != r->head) {
|
||||
/* Invoke optional contention callback */
|
||||
if (r->cb) {
|
||||
r->cb(r->cb_opaque);
|
||||
}
|
||||
qemu_cond_wait(&r->cond, &r->lock);
|
||||
}
|
||||
}
|
||||
|
||||
qemu_thread_get_self(&r->owner_thread);
|
||||
r->nesting++;
|
||||
qemu_mutex_unlock(&r->lock);
|
||||
}
|
||||
|
||||
void rfifolock_unlock(RFifoLock *r)
|
||||
{
|
||||
qemu_mutex_lock(&r->lock);
|
||||
assert(r->nesting > 0);
|
||||
assert(qemu_thread_is_self(&r->owner_thread));
|
||||
if (--r->nesting == 0) {
|
||||
r->head++;
|
||||
qemu_cond_broadcast(&r->cond);
|
||||
}
|
||||
qemu_mutex_unlock(&r->lock);
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue