add UNLEASHED_OBJ to unleashed.mk
[unleashed/tickless.git] / usr / src / cmd / vntsd / queue.c
blob4d50428198df95e70b07c6d1932849c9353a3384
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
22 * Copyright 2006 Sun Microsystems, Inc. All rights reserved.
23 * Use is subject to license terms.
25 #pragma ident "%Z%%M% %I% %E% SMI"
28 * utility for vntsd queue handling
30 #include <stdio.h>
31 #include <sys/types.h>
32 #include <sys/ipc.h>
33 #include <stdlib.h>
34 #include <string.h>
35 #include <unistd.h>
36 #include <sys/socket.h>
37 #include <sys/ipc.h>
38 #include <sys/shm.h>
39 #include <sys/sem.h>
40 #include <wait.h>
41 #include <time.h>
42 #include <netinet/in.h>
43 #include <thread.h>
44 #include <signal.h>
45 #include "vntsd.h"
47 /* alloc_que_el() allocates a queue element */
48 static vntsd_que_t *
49 alloc_que_el(void *handle)
51 vntsd_que_t *el;
53 /* allocate a queue element */
54 el = (vntsd_que_t *)malloc(sizeof (vntsd_que_t));
55 if (el == NULL) {
56 return (NULL);
60 el->nextp = NULL;
61 el->prevp = NULL;
62 el->handle = handle;
64 return (el);
67 /* vntsd_que_append() appends a element to a queue */
68 int
69 vntsd_que_append(vntsd_que_t **que_hd, void *handle)
71 vntsd_que_t *p;
72 vntsd_que_t *el;
74 assert(que_hd);
75 assert(handle);
77 /* allocate a queue element */
78 el = alloc_que_el(handle);
80 if (el == NULL) {
81 return (VNTSD_ERR_NO_MEM);
84 p = *que_hd;
86 if (p == NULL) {
87 /* first one */
88 *que_hd = el;
89 } else {
90 /* walk to the last one */
91 while (p->nextp != NULL)
92 p = p->nextp;
93 p->nextp = el;
96 el->prevp = p;
98 return (VNTSD_SUCCESS);
101 /* vntsd_que_insert_after() inserts element arter the handle */
103 vntsd_que_insert_after(vntsd_que_t *que, void *handle, void *next)
105 vntsd_que_t *q, *el;
107 assert(que);
109 q = que;
111 while (q != NULL) {
112 if (q->handle == handle) {
113 break;
116 q = q->nextp;
119 if (q == NULL) {
120 /* not in queue */
121 return (VNTSD_ERR_EL_NOT_FOUND);
124 el = alloc_que_el(next);
126 if (el == NULL) {
127 return (VNTSD_ERR_NO_MEM);
130 el->nextp = q->nextp;
131 q->nextp = el;
132 el->prevp = q;
134 return (VNTSD_SUCCESS);
139 /* vntsd_que_rm() removes an element from a queue */
141 vntsd_que_rm(vntsd_que_t **que_hd, void *handle)
143 vntsd_que_t *p = *que_hd;
144 vntsd_que_t *prevp = NULL;
147 while (p != NULL) {
148 /* match handle */
149 if (p->handle == handle) {
150 break;
152 prevp = p;
153 p = p->nextp;
156 if (p == NULL) {
157 /* not found */
158 return (VNTSD_ERR_EL_NOT_FOUND);
161 /* found */
162 if (p == *que_hd) {
163 /* first one */
164 *que_hd = p->nextp;
165 } else {
166 prevp->nextp = p->nextp;
169 if (p->nextp != NULL) {
170 p->nextp->prevp = prevp;
173 handle = p->handle;
175 free(p);
177 return (VNTSD_SUCCESS);
181 /* vntsd_que_walk() - walk queue and apply function to each element */
182 void *
183 vntsd_que_walk(vntsd_que_t *que_hd, el_func_t el_func)
185 vntsd_que_t *p = que_hd;
187 while (p != NULL) {
188 if ((*el_func)(p->handle)) {
189 return (p->handle);
192 p = p->nextp;
194 return (VNTSD_SUCCESS);
198 /* vntsd_que_find() finds first match */
199 void *
200 vntsd_que_find(vntsd_que_t *que_hd, compare_func_t compare_func, void *data)
202 vntsd_que_t *p = que_hd;
204 assert(compare_func != NULL);
205 while (p != NULL) {
206 if ((*compare_func)(p->handle, data)) {
207 /* found match */
208 return (p->handle);
211 p = p->nextp;
214 /* not found */
215 return (NULL);
218 /* vntsd_free_que() frees entire queue */
219 void
220 vntsd_free_que(vntsd_que_t **q, clean_func_t clean_func)
222 vntsd_que_t *p;
224 while (*q != NULL) {
225 p = *q;
227 *q = p->nextp;
229 if (clean_func) {
230 /* clean func will free the handle */
231 (*clean_func)(p->handle);
232 } else {
233 free(p->handle);
236 free(p);
241 * vntsd_que_pos() matches a handle and returns a handle located at "pos"
242 * relative to the matched handle. pos supported are 1 or -1.
244 void *
245 vntsd_que_pos(vntsd_que_t *que_hd, void *handle, int pos)
247 vntsd_que_t *p = que_hd;
249 assert((pos == 1) || (pos == -1));
252 while (p != NULL) {
253 if (p->handle == handle) {
254 /* find match */
255 if (pos == 1) {
256 /* forward 1 */
257 if (p->nextp != NULL) {
258 return (p->nextp->handle);
261 /* last one go to first */
262 return (que_hd->handle);
264 } else {
265 /* backward 1 */
266 if (p->prevp != NULL) {
267 return (p->prevp->handle);
270 /* first one, return last one */
271 while (p->nextp != NULL) {
272 p = p->nextp;
275 assert(p != NULL);
276 assert(p->handle != NULL);
277 return (p->handle);
281 p = p->nextp;
284 DERR(stderr, "t@%d vntsd_que_pos can not find handle \n",
285 thr_self());
287 return (NULL);