2009-11-30 18:21:20 +01:00
|
|
|
/* $NetBSD: queue.h,v 1.52 2009/04/20 09:56:08 mschuett Exp $ */
|
2008-12-14 09:53:17 +01:00
|
|
|
|
|
|
|
/*
|
2012-04-07 09:23:39 +02:00
|
|
|
* QEMU version: Copy from netbsd, removed debug code, removed some of
|
2012-01-13 17:34:02 +01:00
|
|
|
* the implementations. Left in singly-linked lists, lists, simple
|
2012-01-13 17:34:03 +01:00
|
|
|
* queues, and tail queues.
|
2008-12-14 09:53:17 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 1991, 1993
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. Neither the name of the University nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* @(#)queue.h 8.5 (Berkeley) 8/20/94
|
|
|
|
*/
|
|
|
|
|
2016-06-29 13:47:03 +02:00
|
|
|
#ifndef QEMU_SYS_QUEUE_H
|
|
|
|
#define QEMU_SYS_QUEUE_H
|
2008-12-14 09:53:17 +01:00
|
|
|
|
|
|
|
/*
|
2012-01-13 17:34:03 +01:00
|
|
|
* This file defines four types of data structures: singly-linked lists,
|
|
|
|
* lists, simple queues, and tail queues.
|
2008-12-14 09:53:17 +01:00
|
|
|
*
|
2012-01-13 17:34:02 +01:00
|
|
|
* A singly-linked list is headed by a single forward pointer. The
|
|
|
|
* elements are singly linked for minimum space and pointer manipulation
|
|
|
|
* overhead at the expense of O(n) removal for arbitrary elements. New
|
|
|
|
* elements can be added to the list after an existing element or at the
|
|
|
|
* head of the list. Elements being removed from the head of the list
|
|
|
|
* should use the explicit macro for this purpose for optimum
|
|
|
|
* efficiency. A singly-linked list may only be traversed in the forward
|
|
|
|
* direction. Singly-linked lists are ideal for applications with large
|
|
|
|
* datasets and few or no removals or for implementing a LIFO queue.
|
|
|
|
*
|
2008-12-14 09:53:17 +01:00
|
|
|
* A list is headed by a single forward pointer (or an array of forward
|
|
|
|
* pointers for a hash table header). The elements are doubly linked
|
|
|
|
* so that an arbitrary element can be removed without a need to
|
|
|
|
* traverse the list. New elements can be added to the list before
|
|
|
|
* or after an existing element or at the head of the list. A list
|
|
|
|
* may only be traversed in the forward direction.
|
|
|
|
*
|
2009-11-30 18:21:20 +01:00
|
|
|
* A simple queue is headed by a pair of pointers, one the head of the
|
|
|
|
* list and the other to the tail of the list. The elements are singly
|
|
|
|
* linked to save space, so elements can only be removed from the
|
|
|
|
* head of the list. New elements can be added to the list after
|
|
|
|
* an existing element, at the head of the list, or at the end of the
|
|
|
|
* list. A simple queue may only be traversed in the forward direction.
|
|
|
|
*
|
2008-12-14 09:53:17 +01:00
|
|
|
* A tail queue is headed by a pair of pointers, one to the head of the
|
|
|
|
* list and the other to the tail of the list. The elements are doubly
|
|
|
|
* linked so that an arbitrary element can be removed without a need to
|
|
|
|
* traverse the list. New elements can be added to the list before or
|
|
|
|
* after an existing element, at the head of the list, or at the end of
|
|
|
|
* the list. A tail queue may be traversed in either direction.
|
|
|
|
*
|
|
|
|
* For details on the use of these macros, see the queue(3) manual page.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* List definitions.
|
|
|
|
*/
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_HEAD(name, type) \
|
2008-12-14 09:53:17 +01:00
|
|
|
struct name { \
|
|
|
|
struct type *lh_first; /* first element */ \
|
|
|
|
}
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_HEAD_INITIALIZER(head) \
|
2008-12-14 09:53:17 +01:00
|
|
|
{ NULL }
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_ENTRY(type) \
|
2008-12-14 09:53:17 +01:00
|
|
|
struct { \
|
|
|
|
struct type *le_next; /* next element */ \
|
|
|
|
struct type **le_prev; /* address of previous next element */ \
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* List functions.
|
|
|
|
*/
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_INIT(head) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
(head)->lh_first = NULL; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
2013-05-13 13:29:47 +02:00
|
|
|
|
|
|
|
#define QLIST_SWAP(dstlist, srclist, field) do { \
|
|
|
|
void *tmplist; \
|
|
|
|
tmplist = (srclist)->lh_first; \
|
|
|
|
(srclist)->lh_first = (dstlist)->lh_first; \
|
|
|
|
if ((srclist)->lh_first != NULL) { \
|
|
|
|
(srclist)->lh_first->field.le_prev = &(srclist)->lh_first; \
|
|
|
|
} \
|
|
|
|
(dstlist)->lh_first = tmplist; \
|
|
|
|
if ((dstlist)->lh_first != NULL) { \
|
|
|
|
(dstlist)->lh_first->field.le_prev = &(dstlist)->lh_first; \
|
|
|
|
} \
|
|
|
|
} while (/*CONSTCOND*/0)
|
2008-12-14 09:53:17 +01:00
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_INSERT_AFTER(listelm, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
if (((elm)->field.le_next = (listelm)->field.le_next) != NULL) \
|
|
|
|
(listelm)->field.le_next->field.le_prev = \
|
|
|
|
&(elm)->field.le_next; \
|
|
|
|
(listelm)->field.le_next = (elm); \
|
|
|
|
(elm)->field.le_prev = &(listelm)->field.le_next; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_INSERT_BEFORE(listelm, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
(elm)->field.le_prev = (listelm)->field.le_prev; \
|
|
|
|
(elm)->field.le_next = (listelm); \
|
|
|
|
*(listelm)->field.le_prev = (elm); \
|
|
|
|
(listelm)->field.le_prev = &(elm)->field.le_next; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_INSERT_HEAD(head, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
if (((elm)->field.le_next = (head)->lh_first) != NULL) \
|
|
|
|
(head)->lh_first->field.le_prev = &(elm)->field.le_next;\
|
|
|
|
(head)->lh_first = (elm); \
|
|
|
|
(elm)->field.le_prev = &(head)->lh_first; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_REMOVE(elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
if ((elm)->field.le_next != NULL) \
|
|
|
|
(elm)->field.le_next->field.le_prev = \
|
|
|
|
(elm)->field.le_prev; \
|
|
|
|
*(elm)->field.le_prev = (elm)->field.le_next; \
|
2020-02-24 11:34:05 +01:00
|
|
|
(elm)->field.le_next = NULL; \
|
|
|
|
(elm)->field.le_prev = NULL; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2020-02-14 18:17:10 +01:00
|
|
|
/*
|
|
|
|
* Like QLIST_REMOVE() but safe to call when elm is not in a list
|
|
|
|
*/
|
|
|
|
#define QLIST_SAFE_REMOVE(elm, field) do { \
|
|
|
|
if ((elm)->field.le_prev != NULL) { \
|
|
|
|
if ((elm)->field.le_next != NULL) \
|
|
|
|
(elm)->field.le_next->field.le_prev = \
|
|
|
|
(elm)->field.le_prev; \
|
|
|
|
*(elm)->field.le_prev = (elm)->field.le_next; \
|
|
|
|
(elm)->field.le_next = NULL; \
|
|
|
|
(elm)->field.le_prev = NULL; \
|
|
|
|
} \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2020-02-14 18:17:11 +01:00
|
|
|
/* Is elm in a list? */
|
|
|
|
#define QLIST_IS_INSERTED(elm, field) ((elm)->field.le_prev != NULL)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_FOREACH(var, head, field) \
|
2008-12-14 09:53:17 +01:00
|
|
|
for ((var) = ((head)->lh_first); \
|
|
|
|
(var); \
|
|
|
|
(var) = ((var)->field.le_next))
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_FOREACH_SAFE(var, head, field, next_var) \
|
2009-09-01 02:12:32 +02:00
|
|
|
for ((var) = ((head)->lh_first); \
|
|
|
|
(var) && ((next_var) = ((var)->field.le_next), 1); \
|
|
|
|
(var) = (next_var))
|
|
|
|
|
2008-12-14 09:53:17 +01:00
|
|
|
/*
|
|
|
|
* List access methods.
|
|
|
|
*/
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QLIST_EMPTY(head) ((head)->lh_first == NULL)
|
|
|
|
#define QLIST_FIRST(head) ((head)->lh_first)
|
|
|
|
#define QLIST_NEXT(elm, field) ((elm)->field.le_next)
|
2008-12-14 09:53:17 +01:00
|
|
|
|
|
|
|
|
2012-01-13 17:34:02 +01:00
|
|
|
/*
|
|
|
|
* Singly-linked List definitions.
|
|
|
|
*/
|
|
|
|
#define QSLIST_HEAD(name, type) \
|
|
|
|
struct name { \
|
|
|
|
struct type *slh_first; /* first element */ \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define QSLIST_HEAD_INITIALIZER(head) \
|
|
|
|
{ NULL }
|
|
|
|
|
|
|
|
#define QSLIST_ENTRY(type) \
|
|
|
|
struct { \
|
|
|
|
struct type *sle_next; /* next element */ \
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Singly-linked List functions.
|
|
|
|
*/
|
|
|
|
#define QSLIST_INIT(head) do { \
|
|
|
|
(head)->slh_first = NULL; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSLIST_INSERT_AFTER(slistelm, elm, field) do { \
|
|
|
|
(elm)->field.sle_next = (slistelm)->field.sle_next; \
|
|
|
|
(slistelm)->field.sle_next = (elm); \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSLIST_INSERT_HEAD(head, elm, field) do { \
|
2014-12-02 12:05:47 +01:00
|
|
|
(elm)->field.sle_next = (head)->slh_first; \
|
|
|
|
(head)->slh_first = (elm); \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
queue: fix QSLIST_INSERT_HEAD_ATOMIC race
There is a not-so-subtle race in QSLIST_INSERT_HEAD_ATOMIC.
Because atomic_cmpxchg returns the old value instead of a success flag,
QSLIST_INSERT_HEAD_ATOMIC was checking for success by comparing against
the second argument to atomic_cmpxchg. Unfortunately, this only works
if the second argument is a local or thread-local variable.
If it is in memory, it can be subject to common subexpression elimination
(and then everything's fine) or reloaded after the atomic_cmpxchg,
depending on the compiler's whims. If the latter happens, the race can
happen. A thread can sneak in, doing something on elm->field.sle_next
after the atomic_cmpxchg and before the comparison. This causes a wrong
failure, and then two threads are using "elm" at the same time. In the
case discovered by Christian, the sequence was likely something like this:
thread 1 | thread 2
QSLIST_INSERT_HEAD_ATOMIC |
atomic_cmpxchg succeeds |
elm added to list |
| steal release_pool
| QSLIST_REMOVE_HEAD
| elm removed from list
| ...
| QSLIST_INSERT_HEAD_ATOMIC
| (overwrites sle_next)
spurious failure |
atomic_cmpxchg succeeds |
elm added to list again |
|
steal release_pool |
QSLIST_REMOVE_HEAD |
elm removed again |
The last three steps could be done by a third thread as well.
A reproducer that failed in a matter of seconds is as follows:
- the guest has 32 VCPUs on a 28 core host (hyperthreading was enabled),
memory was 16G just to err on the safe side (the host has 64G, but hey
at least you need no s390)
- the guest has 24 null-aio virtio-blk devices using dataplane
(-object iothread,id=ioN -drive if=none,id=blkN,driver=null-aio,size=500G
-device virtio-blk-pci,iothread=ioN,drive=blkN)
- the guest also has a single network interface. It's only doing loopback
tests so slirp vs. tap and the model doesn't matter.
- the guest is running fio with the following script:
[global]
rw=randread
blocksize=16k
ioengine=libaio
runtime=10m
buffered=0
fallocate=none
time_based
iodepth=32
[virtio1a]
filename=/dev/block/252\:16
[virtio1b]
filename=/dev/block/252\:16
...
[virtio24a]
filename=/dev/block/252\:384
[virtio24b]
filename=/dev/block/252\:384
[listen1]
protocol=tcp
ioengine=net
port=12345
listen
rw=read
bs=4k
size=1000g
[connect1]
protocol=tcp
hostname=localhost
ioengine=net
port=12345
protocol=tcp
rw=write
startdelay=1
size=1000g
...
[listen8]
protocol=tcp
ioengine=net
port=12352
listen
rw=read
bs=4k
size=1000g
[connect8]
protocol=tcp
hostname=localhost
ioengine=net
port=12352
rw=write
startdelay=1
size=1000g
Moral of the story: I should refrain from writing more clever stuff.
At least it looks like it is not too clever to be undebuggable.
Reported-by: Christian Borntraeger <borntraeger@de.ibm.com>
Tested-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Message-id: 1426002357-6889-1-git-send-email-pbonzini@redhat.com
Fixes: c740ad92d0d958fa785e5d7aa1b67ecaf30a6a54
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
2015-03-10 16:45:57 +01:00
|
|
|
#define QSLIST_INSERT_HEAD_ATOMIC(head, elm, field) do { \
|
|
|
|
typeof(elm) save_sle_next; \
|
|
|
|
do { \
|
|
|
|
save_sle_next = (elm)->field.sle_next = (head)->slh_first; \
|
2020-09-23 12:56:46 +02:00
|
|
|
} while (qatomic_cmpxchg(&(head)->slh_first, save_sle_next, (elm)) !=\
|
queue: fix QSLIST_INSERT_HEAD_ATOMIC race
There is a not-so-subtle race in QSLIST_INSERT_HEAD_ATOMIC.
Because atomic_cmpxchg returns the old value instead of a success flag,
QSLIST_INSERT_HEAD_ATOMIC was checking for success by comparing against
the second argument to atomic_cmpxchg. Unfortunately, this only works
if the second argument is a local or thread-local variable.
If it is in memory, it can be subject to common subexpression elimination
(and then everything's fine) or reloaded after the atomic_cmpxchg,
depending on the compiler's whims. If the latter happens, the race can
happen. A thread can sneak in, doing something on elm->field.sle_next
after the atomic_cmpxchg and before the comparison. This causes a wrong
failure, and then two threads are using "elm" at the same time. In the
case discovered by Christian, the sequence was likely something like this:
thread 1 | thread 2
QSLIST_INSERT_HEAD_ATOMIC |
atomic_cmpxchg succeeds |
elm added to list |
| steal release_pool
| QSLIST_REMOVE_HEAD
| elm removed from list
| ...
| QSLIST_INSERT_HEAD_ATOMIC
| (overwrites sle_next)
spurious failure |
atomic_cmpxchg succeeds |
elm added to list again |
|
steal release_pool |
QSLIST_REMOVE_HEAD |
elm removed again |
The last three steps could be done by a third thread as well.
A reproducer that failed in a matter of seconds is as follows:
- the guest has 32 VCPUs on a 28 core host (hyperthreading was enabled),
memory was 16G just to err on the safe side (the host has 64G, but hey
at least you need no s390)
- the guest has 24 null-aio virtio-blk devices using dataplane
(-object iothread,id=ioN -drive if=none,id=blkN,driver=null-aio,size=500G
-device virtio-blk-pci,iothread=ioN,drive=blkN)
- the guest also has a single network interface. It's only doing loopback
tests so slirp vs. tap and the model doesn't matter.
- the guest is running fio with the following script:
[global]
rw=randread
blocksize=16k
ioengine=libaio
runtime=10m
buffered=0
fallocate=none
time_based
iodepth=32
[virtio1a]
filename=/dev/block/252\:16
[virtio1b]
filename=/dev/block/252\:16
...
[virtio24a]
filename=/dev/block/252\:384
[virtio24b]
filename=/dev/block/252\:384
[listen1]
protocol=tcp
ioengine=net
port=12345
listen
rw=read
bs=4k
size=1000g
[connect1]
protocol=tcp
hostname=localhost
ioengine=net
port=12345
protocol=tcp
rw=write
startdelay=1
size=1000g
...
[listen8]
protocol=tcp
ioengine=net
port=12352
listen
rw=read
bs=4k
size=1000g
[connect8]
protocol=tcp
hostname=localhost
ioengine=net
port=12352
rw=write
startdelay=1
size=1000g
Moral of the story: I should refrain from writing more clever stuff.
At least it looks like it is not too clever to be undebuggable.
Reported-by: Christian Borntraeger <borntraeger@de.ibm.com>
Tested-by: Christian Borntraeger <borntraeger@de.ibm.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Message-id: 1426002357-6889-1-git-send-email-pbonzini@redhat.com
Fixes: c740ad92d0d958fa785e5d7aa1b67ecaf30a6a54
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
2015-03-10 16:45:57 +01:00
|
|
|
save_sle_next); \
|
2014-12-02 12:05:47 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSLIST_MOVE_ATOMIC(dest, src) do { \
|
2020-09-23 12:56:46 +02:00
|
|
|
(dest)->slh_first = qatomic_xchg(&(src)->slh_first, NULL); \
|
2012-01-13 17:34:02 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSLIST_REMOVE_HEAD(head, field) do { \
|
2020-02-24 11:34:05 +01:00
|
|
|
typeof((head)->slh_first) elm = (head)->slh_first; \
|
|
|
|
(head)->slh_first = elm->field.sle_next; \
|
|
|
|
elm->field.sle_next = NULL; \
|
2012-01-13 17:34:02 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2020-02-20 11:38:28 +01:00
|
|
|
#define QSLIST_REMOVE_AFTER(slistelm, field) do { \
|
2020-02-24 11:34:05 +01:00
|
|
|
typeof(slistelm) next = (slistelm)->field.sle_next; \
|
|
|
|
(slistelm)->field.sle_next = next->field.sle_next; \
|
|
|
|
next->field.sle_next = NULL; \
|
2020-02-20 11:38:28 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSLIST_REMOVE(head, elm, type, field) do { \
|
|
|
|
if ((head)->slh_first == (elm)) { \
|
|
|
|
QSLIST_REMOVE_HEAD((head), field); \
|
|
|
|
} else { \
|
|
|
|
struct type *curelm = (head)->slh_first; \
|
|
|
|
while (curelm->field.sle_next != (elm)) \
|
|
|
|
curelm = curelm->field.sle_next; \
|
|
|
|
curelm->field.sle_next = curelm->field.sle_next->field.sle_next; \
|
2020-02-24 11:34:05 +01:00
|
|
|
(elm)->field.sle_next = NULL; \
|
2020-02-20 11:38:28 +01:00
|
|
|
} \
|
2012-01-13 17:34:02 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSLIST_FOREACH(var, head, field) \
|
|
|
|
for((var) = (head)->slh_first; (var); (var) = (var)->field.sle_next)
|
|
|
|
|
|
|
|
#define QSLIST_FOREACH_SAFE(var, head, field, tvar) \
|
|
|
|
for ((var) = QSLIST_FIRST((head)); \
|
|
|
|
(var) && ((tvar) = QSLIST_NEXT((var), field), 1); \
|
|
|
|
(var) = (tvar))
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Singly-linked List access methods.
|
|
|
|
*/
|
|
|
|
#define QSLIST_EMPTY(head) ((head)->slh_first == NULL)
|
|
|
|
#define QSLIST_FIRST(head) ((head)->slh_first)
|
|
|
|
#define QSLIST_NEXT(elm, field) ((elm)->field.sle_next)
|
|
|
|
|
|
|
|
|
2009-11-30 18:21:20 +01:00
|
|
|
/*
|
|
|
|
* Simple queue definitions.
|
|
|
|
*/
|
|
|
|
#define QSIMPLEQ_HEAD(name, type) \
|
|
|
|
struct name { \
|
|
|
|
struct type *sqh_first; /* first element */ \
|
|
|
|
struct type **sqh_last; /* addr of last next element */ \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define QSIMPLEQ_HEAD_INITIALIZER(head) \
|
|
|
|
{ NULL, &(head).sqh_first }
|
|
|
|
|
|
|
|
#define QSIMPLEQ_ENTRY(type) \
|
|
|
|
struct { \
|
|
|
|
struct type *sqe_next; /* next element */ \
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simple queue functions.
|
|
|
|
*/
|
|
|
|
#define QSIMPLEQ_INIT(head) do { \
|
|
|
|
(head)->sqh_first = NULL; \
|
|
|
|
(head)->sqh_last = &(head)->sqh_first; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSIMPLEQ_INSERT_HEAD(head, elm, field) do { \
|
|
|
|
if (((elm)->field.sqe_next = (head)->sqh_first) == NULL) \
|
|
|
|
(head)->sqh_last = &(elm)->field.sqe_next; \
|
|
|
|
(head)->sqh_first = (elm); \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSIMPLEQ_INSERT_TAIL(head, elm, field) do { \
|
|
|
|
(elm)->field.sqe_next = NULL; \
|
|
|
|
*(head)->sqh_last = (elm); \
|
|
|
|
(head)->sqh_last = &(elm)->field.sqe_next; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSIMPLEQ_INSERT_AFTER(head, listelm, elm, field) do { \
|
|
|
|
if (((elm)->field.sqe_next = (listelm)->field.sqe_next) == NULL) \
|
|
|
|
(head)->sqh_last = &(elm)->field.sqe_next; \
|
|
|
|
(listelm)->field.sqe_next = (elm); \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSIMPLEQ_REMOVE_HEAD(head, field) do { \
|
2020-02-24 11:34:05 +01:00
|
|
|
typeof((head)->sqh_first) elm = (head)->sqh_first; \
|
|
|
|
if (((head)->sqh_first = elm->field.sqe_next) == NULL) \
|
2009-11-30 18:21:20 +01:00
|
|
|
(head)->sqh_last = &(head)->sqh_first; \
|
2020-02-24 11:34:05 +01:00
|
|
|
elm->field.sqe_next = NULL; \
|
2009-11-30 18:21:20 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2014-12-11 14:52:30 +01:00
|
|
|
#define QSIMPLEQ_SPLIT_AFTER(head, elm, field, removed) do { \
|
|
|
|
QSIMPLEQ_INIT(removed); \
|
|
|
|
if (((removed)->sqh_first = (head)->sqh_first) != NULL) { \
|
|
|
|
if (((head)->sqh_first = (elm)->field.sqe_next) == NULL) { \
|
|
|
|
(head)->sqh_last = &(head)->sqh_first; \
|
|
|
|
} \
|
|
|
|
(removed)->sqh_last = &(elm)->field.sqe_next; \
|
|
|
|
(elm)->field.sqe_next = NULL; \
|
|
|
|
} \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-11-30 18:21:20 +01:00
|
|
|
#define QSIMPLEQ_REMOVE(head, elm, type, field) do { \
|
|
|
|
if ((head)->sqh_first == (elm)) { \
|
|
|
|
QSIMPLEQ_REMOVE_HEAD((head), field); \
|
|
|
|
} else { \
|
|
|
|
struct type *curelm = (head)->sqh_first; \
|
|
|
|
while (curelm->field.sqe_next != (elm)) \
|
|
|
|
curelm = curelm->field.sqe_next; \
|
|
|
|
if ((curelm->field.sqe_next = \
|
|
|
|
curelm->field.sqe_next->field.sqe_next) == NULL) \
|
|
|
|
(head)->sqh_last = &(curelm)->field.sqe_next; \
|
2020-02-24 11:34:05 +01:00
|
|
|
(elm)->field.sqe_next = NULL; \
|
2009-11-30 18:21:20 +01:00
|
|
|
} \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
|
|
|
#define QSIMPLEQ_FOREACH(var, head, field) \
|
|
|
|
for ((var) = ((head)->sqh_first); \
|
|
|
|
(var); \
|
|
|
|
(var) = ((var)->field.sqe_next))
|
|
|
|
|
|
|
|
#define QSIMPLEQ_FOREACH_SAFE(var, head, field, next) \
|
|
|
|
for ((var) = ((head)->sqh_first); \
|
|
|
|
(var) && ((next = ((var)->field.sqe_next)), 1); \
|
|
|
|
(var) = (next))
|
|
|
|
|
|
|
|
#define QSIMPLEQ_CONCAT(head1, head2) do { \
|
|
|
|
if (!QSIMPLEQ_EMPTY((head2))) { \
|
|
|
|
*(head1)->sqh_last = (head2)->sqh_first; \
|
|
|
|
(head1)->sqh_last = (head2)->sqh_last; \
|
|
|
|
QSIMPLEQ_INIT((head2)); \
|
|
|
|
} \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2018-03-22 16:28:32 +01:00
|
|
|
#define QSIMPLEQ_PREPEND(head1, head2) do { \
|
|
|
|
if (!QSIMPLEQ_EMPTY((head2))) { \
|
|
|
|
*(head2)->sqh_last = (head1)->sqh_first; \
|
|
|
|
(head1)->sqh_first = (head2)->sqh_first; \
|
|
|
|
QSIMPLEQ_INIT((head2)); \
|
|
|
|
} \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-11-30 18:21:20 +01:00
|
|
|
#define QSIMPLEQ_LAST(head, type, field) \
|
|
|
|
(QSIMPLEQ_EMPTY((head)) ? \
|
|
|
|
NULL : \
|
|
|
|
((struct type *)(void *) \
|
|
|
|
((char *)((head)->sqh_last) - offsetof(struct type, field))))
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simple queue access methods.
|
|
|
|
*/
|
2020-09-23 12:56:46 +02:00
|
|
|
#define QSIMPLEQ_EMPTY_ATOMIC(head) \
|
|
|
|
(qatomic_read(&((head)->sqh_first)) == NULL)
|
2009-11-30 18:21:20 +01:00
|
|
|
#define QSIMPLEQ_EMPTY(head) ((head)->sqh_first == NULL)
|
|
|
|
#define QSIMPLEQ_FIRST(head) ((head)->sqh_first)
|
|
|
|
#define QSIMPLEQ_NEXT(elm, field) ((elm)->field.sqe_next)
|
|
|
|
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
typedef struct QTailQLink {
|
|
|
|
void *tql_next;
|
|
|
|
struct QTailQLink *tql_prev;
|
|
|
|
} QTailQLink;
|
2009-11-30 18:21:20 +01:00
|
|
|
|
2008-12-14 09:53:17 +01:00
|
|
|
/*
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
* Tail queue definitions. The union acts as a poor man template, as if
|
|
|
|
* it were QTailQLink<type>.
|
2008-12-14 09:53:17 +01:00
|
|
|
*/
|
2018-12-10 18:03:06 +01:00
|
|
|
#define QTAILQ_HEAD(name, type) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
union name { \
|
|
|
|
struct type *tqh_first; /* first element */ \
|
|
|
|
QTailQLink tqh_circ; /* link for circular backwards list */ \
|
2008-12-14 09:53:17 +01:00
|
|
|
}
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_HEAD_INITIALIZER(head) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
{ .tqh_circ = { NULL, &(head).tqh_circ } }
|
2008-12-14 09:53:17 +01:00
|
|
|
|
2018-12-10 18:03:06 +01:00
|
|
|
#define QTAILQ_ENTRY(type) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
union { \
|
|
|
|
struct type *tqe_next; /* next element */ \
|
|
|
|
QTailQLink tqe_circ; /* link for circular backwards list */ \
|
2008-12-14 09:53:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Tail queue functions.
|
|
|
|
*/
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_INIT(head) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
(head)->tqh_first = NULL; \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(head)->tqh_circ.tql_prev = &(head)->tqh_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_INSERT_HEAD(head, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
if (((elm)->field.tqe_next = (head)->tqh_first) != NULL) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(head)->tqh_first->field.tqe_circ.tql_prev = \
|
|
|
|
&(elm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
else \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(head)->tqh_circ.tql_prev = &(elm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
(head)->tqh_first = (elm); \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(elm)->field.tqe_circ.tql_prev = &(head)->tqh_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_INSERT_TAIL(head, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
(elm)->field.tqe_next = NULL; \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(elm)->field.tqe_circ.tql_prev = (head)->tqh_circ.tql_prev; \
|
|
|
|
(head)->tqh_circ.tql_prev->tql_next = (elm); \
|
|
|
|
(head)->tqh_circ.tql_prev = &(elm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_INSERT_AFTER(head, listelm, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
if (((elm)->field.tqe_next = (listelm)->field.tqe_next) != NULL)\
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(elm)->field.tqe_next->field.tqe_circ.tql_prev = \
|
|
|
|
&(elm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
else \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(head)->tqh_circ.tql_prev = &(elm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
(listelm)->field.tqe_next = (elm); \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(elm)->field.tqe_circ.tql_prev = &(listelm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
#define QTAILQ_INSERT_BEFORE(listelm, elm, field) do { \
|
|
|
|
(elm)->field.tqe_circ.tql_prev = (listelm)->field.tqe_circ.tql_prev; \
|
|
|
|
(elm)->field.tqe_next = (listelm); \
|
|
|
|
(listelm)->field.tqe_circ.tql_prev->tql_next = (elm); \
|
|
|
|
(listelm)->field.tqe_circ.tql_prev = &(elm)->field.tqe_circ; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_REMOVE(head, elm, field) do { \
|
2008-12-14 09:53:17 +01:00
|
|
|
if (((elm)->field.tqe_next) != NULL) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(elm)->field.tqe_next->field.tqe_circ.tql_prev = \
|
|
|
|
(elm)->field.tqe_circ.tql_prev; \
|
2008-12-14 09:53:17 +01:00
|
|
|
else \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(head)->tqh_circ.tql_prev = (elm)->field.tqe_circ.tql_prev; \
|
|
|
|
(elm)->field.tqe_circ.tql_prev->tql_next = (elm)->field.tqe_next; \
|
|
|
|
(elm)->field.tqe_circ.tql_prev = NULL; \
|
2020-02-24 11:34:05 +01:00
|
|
|
(elm)->field.tqe_circ.tql_next = NULL; \
|
|
|
|
(elm)->field.tqe_next = NULL; \
|
2008-12-14 09:53:17 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2018-12-07 01:04:07 +01:00
|
|
|
/* remove @left, @right and all elements in between from @head */
|
|
|
|
#define QTAILQ_REMOVE_SEVERAL(head, left, right, field) do { \
|
|
|
|
if (((right)->field.tqe_next) != NULL) \
|
|
|
|
(right)->field.tqe_next->field.tqe_circ.tql_prev = \
|
|
|
|
(left)->field.tqe_circ.tql_prev; \
|
|
|
|
else \
|
|
|
|
(head)->tqh_circ.tql_prev = (left)->field.tqe_circ.tql_prev; \
|
|
|
|
(left)->field.tqe_circ.tql_prev->tql_next = (right)->field.tqe_next; \
|
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_FOREACH(var, head, field) \
|
2008-12-14 09:53:17 +01:00
|
|
|
for ((var) = ((head)->tqh_first); \
|
|
|
|
(var); \
|
|
|
|
(var) = ((var)->field.tqe_next))
|
|
|
|
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_FOREACH_SAFE(var, head, field, next_var) \
|
2008-12-14 09:53:17 +01:00
|
|
|
for ((var) = ((head)->tqh_first); \
|
|
|
|
(var) && ((next_var) = ((var)->field.tqe_next), 1); \
|
|
|
|
(var) = (next_var))
|
|
|
|
|
2018-12-06 13:10:34 +01:00
|
|
|
#define QTAILQ_FOREACH_REVERSE(var, head, field) \
|
|
|
|
for ((var) = QTAILQ_LAST(head); \
|
2008-12-14 09:53:17 +01:00
|
|
|
(var); \
|
2018-12-06 13:10:34 +01:00
|
|
|
(var) = QTAILQ_PREV(var, field))
|
2008-12-14 09:53:17 +01:00
|
|
|
|
2018-12-06 13:10:34 +01:00
|
|
|
#define QTAILQ_FOREACH_REVERSE_SAFE(var, head, field, prev_var) \
|
|
|
|
for ((var) = QTAILQ_LAST(head); \
|
2019-02-04 16:40:18 +01:00
|
|
|
(var) && ((prev_var) = QTAILQ_PREV(var, field), 1); \
|
2017-11-02 15:19:14 +01:00
|
|
|
(var) = (prev_var))
|
|
|
|
|
2008-12-14 09:53:17 +01:00
|
|
|
/*
|
|
|
|
* Tail queue access methods.
|
|
|
|
*/
|
2009-09-12 09:36:22 +02:00
|
|
|
#define QTAILQ_EMPTY(head) ((head)->tqh_first == NULL)
|
|
|
|
#define QTAILQ_FIRST(head) ((head)->tqh_first)
|
|
|
|
#define QTAILQ_NEXT(elm, field) ((elm)->field.tqe_next)
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
#define QTAILQ_IN_USE(elm, field) ((elm)->field.tqe_circ.tql_prev != NULL)
|
2008-12-14 09:53:17 +01:00
|
|
|
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
#define QTAILQ_LINK_PREV(link) \
|
|
|
|
((link).tql_prev->tql_prev->tql_next)
|
2018-12-06 13:10:34 +01:00
|
|
|
#define QTAILQ_LAST(head) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
((typeof((head)->tqh_first)) QTAILQ_LINK_PREV((head)->tqh_circ))
|
2018-12-06 13:10:34 +01:00
|
|
|
#define QTAILQ_PREV(elm, field) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
((typeof((elm)->field.tqe_next)) QTAILQ_LINK_PREV((elm)->field.tqe_circ))
|
2008-12-14 09:53:17 +01:00
|
|
|
|
2017-01-19 20:00:51 +01:00
|
|
|
#define field_at_offset(base, offset, type) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
((type *) (((char *) (base)) + (offset)))
|
2017-01-19 20:00:51 +01:00
|
|
|
|
|
|
|
/*
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
* Raw access of elements of a tail queue head. Offsets are all zero
|
|
|
|
* because it's a union.
|
2017-01-19 20:00:51 +01:00
|
|
|
*/
|
|
|
|
#define QTAILQ_RAW_FIRST(head) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
field_at_offset(head, 0, void *)
|
|
|
|
#define QTAILQ_RAW_TQH_CIRC(head) \
|
|
|
|
field_at_offset(head, 0, QTailQLink)
|
2017-01-19 20:00:51 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Raw access of elements of a tail entry
|
|
|
|
*/
|
|
|
|
#define QTAILQ_RAW_NEXT(elm, entry) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
field_at_offset(elm, entry, void *)
|
|
|
|
#define QTAILQ_RAW_TQE_CIRC(elm, entry) \
|
|
|
|
field_at_offset(elm, entry, QTailQLink)
|
2017-01-19 20:00:51 +01:00
|
|
|
/*
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
* Tail queue traversal using pointer arithmetic.
|
2017-01-19 20:00:51 +01:00
|
|
|
*/
|
|
|
|
#define QTAILQ_RAW_FOREACH(elm, head, entry) \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
for ((elm) = *QTAILQ_RAW_FIRST(head); \
|
2017-01-19 20:00:51 +01:00
|
|
|
(elm); \
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
(elm) = *QTAILQ_RAW_NEXT(elm, entry))
|
2017-01-19 20:00:51 +01:00
|
|
|
/*
|
|
|
|
* Tail queue insertion using pointer arithmetic.
|
|
|
|
*/
|
qemu/queue.h: reimplement QTAILQ without pointer-to-pointers
QTAILQ is a doubly linked list, with a pointer-to-pointer to the last
element from the head, and the previous element from each node.
But if you squint enough, QTAILQ becomes a combination of a singly-linked
forwards list, and another singly-linked list which goes backwards and
is circular. This is the idea that lets QTAILQ implement reverse
iteration: only, because the backwards list points inside the node,
accessing the previous element needs to go two steps back and one
forwards.
What this patch does is implement it in these terms, without actually
changing the in-memory layout at all. The coexistence of the two lists
is realized by making QTAILQ_HEAD and QTAILQ_ENTRY unions of the forwards
pointer and a generic QTailQLink node. Thq QTailQLink can walk the list in
both directions; the union is needed so that the forwards pointer can
have the correct type, as a sort of poor man's template. While there
are other ways to get the same layout without a union, this one has
the advantage of simpler operation in the debugger, because the fields
tqh_first and tqe_next still exist as before the patch. Those fields are
also used by scripts/qemugdb/mtree.py, so it's a good idea to preserve them.
The advantage of the new representation is that the two-back-one-forward
dance done by backwards accesses can be done all while operating on
QTailQLinks. No casting to the head struct is needed anymore because,
even though the QTailQLink's forward pointer is a void *, we can use
typeof to recover the correct type. This patch only changes the
implementation, not the interface. The next patch will remove the head
struct name from the backwards visit macros.
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2018-12-06 12:01:53 +01:00
|
|
|
#define QTAILQ_RAW_INSERT_TAIL(head, elm, entry) do { \
|
|
|
|
*QTAILQ_RAW_NEXT(elm, entry) = NULL; \
|
|
|
|
QTAILQ_RAW_TQE_CIRC(elm, entry)->tql_prev = QTAILQ_RAW_TQH_CIRC(head)->tql_prev; \
|
|
|
|
QTAILQ_RAW_TQH_CIRC(head)->tql_prev->tql_next = (elm); \
|
|
|
|
QTAILQ_RAW_TQH_CIRC(head)->tql_prev = QTAILQ_RAW_TQE_CIRC(elm, entry); \
|
2017-01-19 20:00:51 +01:00
|
|
|
} while (/*CONSTCOND*/0)
|
|
|
|
|
2020-01-13 14:48:23 +01:00
|
|
|
#define QLIST_RAW_FIRST(head) \
|
|
|
|
field_at_offset(head, 0, void *)
|
|
|
|
|
|
|
|
#define QLIST_RAW_NEXT(elm, entry) \
|
|
|
|
field_at_offset(elm, entry, void *)
|
|
|
|
|
|
|
|
#define QLIST_RAW_PREVIOUS(elm, entry) \
|
|
|
|
field_at_offset(elm, entry + sizeof(void *), void *)
|
|
|
|
|
|
|
|
#define QLIST_RAW_FOREACH(elm, head, entry) \
|
|
|
|
for ((elm) = *QLIST_RAW_FIRST(head); \
|
|
|
|
(elm); \
|
|
|
|
(elm) = *QLIST_RAW_NEXT(elm, entry))
|
|
|
|
|
2020-01-25 18:24:49 +01:00
|
|
|
#define QLIST_RAW_INSERT_AFTER(head, prev, elem, entry) do { \
|
|
|
|
*QLIST_RAW_NEXT(prev, entry) = elem; \
|
|
|
|
*QLIST_RAW_PREVIOUS(elem, entry) = QLIST_RAW_NEXT(prev, entry); \
|
|
|
|
*QLIST_RAW_NEXT(elem, entry) = NULL; \
|
|
|
|
} while (0)
|
|
|
|
|
2020-01-13 14:48:23 +01:00
|
|
|
#define QLIST_RAW_INSERT_HEAD(head, elm, entry) do { \
|
|
|
|
void *first = *QLIST_RAW_FIRST(head); \
|
|
|
|
*QLIST_RAW_FIRST(head) = elm; \
|
|
|
|
*QLIST_RAW_PREVIOUS(elm, entry) = QLIST_RAW_FIRST(head); \
|
|
|
|
if (first) { \
|
|
|
|
*QLIST_RAW_NEXT(elm, entry) = first; \
|
|
|
|
*QLIST_RAW_PREVIOUS(first, entry) = QLIST_RAW_NEXT(elm, entry); \
|
|
|
|
} else { \
|
|
|
|
*QLIST_RAW_NEXT(elm, entry) = NULL; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2016-06-29 13:47:03 +02:00
|
|
|
#endif /* QEMU_SYS_QUEUE_H */
|