Home | History | Annotate | Line # | Download | only in libpthread
pthread_queue.h revision 1.3
      1  1.3  christos /*	$NetBSD: pthread_queue.h,v 1.3 2003/01/18 18:45:56 christos Exp $	*/
      2  1.2   thorpej 
      3  1.2   thorpej /*-
      4  1.2   thorpej  * Copyright (c) 2001 The NetBSD Foundation, Inc.
      5  1.2   thorpej  * All rights reserved.
      6  1.2   thorpej  *
      7  1.2   thorpej  * This code is derived from software contributed to The NetBSD Foundation
      8  1.2   thorpej  * by Nathan J. Williams.
      9  1.2   thorpej  *
     10  1.2   thorpej  * Redistribution and use in source and binary forms, with or without
     11  1.2   thorpej  * modification, are permitted provided that the following conditions
     12  1.2   thorpej  * are met:
     13  1.2   thorpej  * 1. Redistributions of source code must retain the above copyright
     14  1.2   thorpej  *    notice, this list of conditions and the following disclaimer.
     15  1.2   thorpej  * 2. Redistributions in binary form must reproduce the above copyright
     16  1.2   thorpej  *    notice, this list of conditions and the following disclaimer in the
     17  1.2   thorpej  *    documentation and/or other materials provided with the distribution.
     18  1.2   thorpej  * 3. All advertising materials mentioning features or use of this software
     19  1.2   thorpej  *    must display the following acknowledgement:
     20  1.2   thorpej  *        This product includes software developed by the NetBSD
     21  1.2   thorpej  *        Foundation, Inc. and its contributors.
     22  1.2   thorpej  * 4. Neither the name of The NetBSD Foundation nor the names of its
     23  1.2   thorpej  *    contributors may be used to endorse or promote products derived
     24  1.2   thorpej  *    from this software without specific prior written permission.
     25  1.2   thorpej  *
     26  1.2   thorpej  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     27  1.2   thorpej  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     28  1.2   thorpej  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     29  1.2   thorpej  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     30  1.2   thorpej  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     31  1.2   thorpej  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     32  1.2   thorpej  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     33  1.2   thorpej  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     34  1.2   thorpej  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     35  1.2   thorpej  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     36  1.2   thorpej  * POSSIBILITY OF SUCH DAMAGE.
     37  1.2   thorpej  */
     38  1.2   thorpej 
     39  1.2   thorpej /* pthread_queue.h
     40  1.2   thorpej  * Definition of a queue interface for the pthread library.
     41  1.2   thorpej  * Style modeled on the sys/queue.h macros; implementation taken from
     42  1.2   thorpej  * the tail queue, with the added property of static initializability
     43  1.2   thorpej  * (and a corresponding extra cost in the _INSERT_TAIL() function.
     44  1.2   thorpej */
     45  1.2   thorpej 
     46  1.2   thorpej 
     47  1.2   thorpej 
     48  1.2   thorpej /*
     49  1.2   thorpej  * Queue definitions.
     50  1.2   thorpej  */
     51  1.2   thorpej #define PTQ_HEAD(name, type)						\
     52  1.2   thorpej struct name {								\
     53  1.2   thorpej 	struct type *ptqh_first;/* first element */			\
     54  1.2   thorpej 	struct type **ptqh_last;/* addr of last next element */		\
     55  1.2   thorpej }
     56  1.2   thorpej 
     57  1.2   thorpej #define PTQ_HEAD_INITIALIZER { NULL, NULL }
     58  1.2   thorpej 
     59  1.2   thorpej 
     60  1.2   thorpej #define PTQ_ENTRY(type)						\
     61  1.2   thorpej struct {								\
     62  1.2   thorpej 	struct type *ptqe_next;	/* next element */			\
     63  1.2   thorpej 	struct type **ptqe_prev;/* address of previous next element */	\
     64  1.2   thorpej }
     65  1.2   thorpej 
     66  1.2   thorpej /*
     67  1.2   thorpej  * Queue functions.
     68  1.2   thorpej  */
     69  1.2   thorpej 
     70  1.2   thorpej #define	PTQ_INIT(head) do {						\
     71  1.2   thorpej 	(head)->ptqh_first = NULL;					\
     72  1.2   thorpej 	(head)->ptqh_last = &(head)->ptqh_first;			\
     73  1.2   thorpej } while (/*CONSTCOND*/0)
     74  1.2   thorpej 
     75  1.2   thorpej #define PTQ_INSERT_HEAD(head, elm, field) do {			\
     76  1.2   thorpej 	if (((elm)->field.ptqe_next = (head)->ptqh_first) != NULL)	\
     77  1.2   thorpej 		(head)->ptqh_first->field.ptqe_prev =			\
     78  1.2   thorpej 		    &(elm)->field.ptqe_next;				\
     79  1.2   thorpej 	else								\
     80  1.2   thorpej 		(head)->ptqh_last = &(elm)->field.ptqe_next;		\
     81  1.2   thorpej 	(head)->ptqh_first = (elm);					\
     82  1.2   thorpej 	(elm)->field.ptqe_prev = &(head)->ptqh_first;			\
     83  1.2   thorpej } while (/*CONSTCOND*/0)
     84  1.2   thorpej 
     85  1.2   thorpej #define PTQ_INSERT_TAIL(head, elm, field) do {				\
     86  1.2   thorpej 	(elm)->field.ptqe_next = NULL;					\
     87  1.2   thorpej 	if ((head)->ptqh_last == NULL)					\
     88  1.2   thorpej 		(head)->ptqh_last = &(head)->ptqh_first;		\
     89  1.2   thorpej 	(elm)->field.ptqe_prev = (head)->ptqh_last;			\
     90  1.2   thorpej 	*(head)->ptqh_last = (elm);					\
     91  1.2   thorpej 	(head)->ptqh_last = &(elm)->field.ptqe_next;			\
     92  1.2   thorpej } while (/*CONSTCOND*/0)
     93  1.2   thorpej 
     94  1.2   thorpej #define PTQ_INSERT_AFTER(head, listelm, elm, field) do {		\
     95  1.2   thorpej 	if (((elm)->field.ptqe_next = (listelm)->field.ptqe_next) != NULL)\
     96  1.2   thorpej 		(elm)->field.ptqe_next->field.ptqe_prev = 		\
     97  1.2   thorpej 		    &(elm)->field.ptqe_next;				\
     98  1.2   thorpej 	else								\
     99  1.2   thorpej 		(head)->ptqh_last = &(elm)->field.ptqe_next;		\
    100  1.2   thorpej 	(listelm)->field.ptqe_next = (elm);				\
    101  1.2   thorpej 	(elm)->field.ptqe_prev = &(listelm)->field.ptqe_next;		\
    102  1.2   thorpej } while (/*CONSTCOND*/0)
    103  1.2   thorpej 
    104  1.2   thorpej #define	PTQ_INSERT_BEFORE(listelm, elm, field) do {			\
    105  1.2   thorpej 	(elm)->field.ptqe_prev = (listelm)->field.ptqe_prev;		\
    106  1.2   thorpej 	(elm)->field.ptqe_next = (listelm);				\
    107  1.2   thorpej 	*(listelm)->field.ptqe_prev = (elm);				\
    108  1.2   thorpej 	(listelm)->field.ptqe_prev = &(elm)->field.ptqe_next;		\
    109  1.2   thorpej } while (/*CONSTCOND*/0)
    110  1.2   thorpej 
    111  1.2   thorpej #define PTQ_REMOVE(head, elm, field) do {				\
    112  1.2   thorpej 	if (((elm)->field.ptqe_next) != NULL)				\
    113  1.2   thorpej 		(elm)->field.ptqe_next->field.ptqe_prev = 		\
    114  1.2   thorpej 		    (elm)->field.ptqe_prev;				\
    115  1.2   thorpej 	else								\
    116  1.2   thorpej 		(head)->ptqh_last = (elm)->field.ptqe_prev;		\
    117  1.2   thorpej 	*(elm)->field.ptqe_prev = (elm)->field.ptqe_next;		\
    118  1.2   thorpej } while (/*CONSTCOND*/0)
    119  1.2   thorpej 
    120  1.2   thorpej /*
    121  1.2   thorpej  * Queue access methods.
    122  1.2   thorpej  */
    123  1.2   thorpej #define	PTQ_EMPTY(head)		((head)->ptqh_first == NULL)
    124  1.2   thorpej #define	PTQ_FIRST(head)		((head)->ptqh_first)
    125  1.2   thorpej #define	PTQ_NEXT(elm, field)		((elm)->field.ptqe_next)
    126  1.2   thorpej 
    127  1.2   thorpej #define PTQ_LAST(head, headname) \
    128  1.3  christos 	(*(((struct headname *)(void *)((head)->ptqh_last))->ptqh_last))
    129  1.2   thorpej #define PTQ_PREV(elm, headname, field) \
    130  1.3  christos 	(*(((struct headname *)(void *)((elm)->field.ptqe_prev))->ptqh_last))
    131  1.2   thorpej 
    132  1.2   thorpej #define PTQ_FOREACH(var, head, field)					\
    133  1.2   thorpej 	for ((var) = ((head)->ptqh_first);				\
    134  1.2   thorpej 		(var);							\
    135  1.2   thorpej 		(var) = ((var)->field.ptqe_next))
    136  1.2   thorpej 
    137  1.2   thorpej #define PTQ_FOREACH_REVERSE(var, head, headname, field)		\
    138  1.3  christos 	for ((var) = (*(((struct headname *)(void *)((head)->ptqh_last))->ptqh_last));	\
    139  1.2   thorpej 		(var);							\
    140  1.3  christos 		(var) = (*(((struct headname *)(void *)((var)->field.ptqe_prev))->ptqh_last)))
    141