1 /* 2 Copyright (c) 2007-2021, Troy D. Hanson http://troydhanson.github.io/uthash/ 3 All rights reserved. 4 5 Redistribution and use in source and binary forms, with or without 6 modification, are permitted provided that the following conditions are met: 7 8 * Redistributions of source code must retain the above copyright 9 notice, this list of conditions and the following disclaimer. 10 11 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS 12 IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 13 TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A 14 PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER 15 OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 16 EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 17 PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 18 PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF 19 LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING 20 NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 21 SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 22 */ 23 24 #ifndef UTLIST_H 25 #define UTLIST_H 26 27 #define UTLIST_VERSION 2.3.0 28 29 #include <assert.h> 30 31 /* 32 * This file contains macros to manipulate singly and doubly-linked lists. 33 * 34 * 1. LL_ macros: singly-linked lists. 35 * 2. DL_ macros: doubly-linked lists. 36 * 3. CDL_ macros: circular doubly-linked lists. 37 * 38 * To use singly-linked lists, your structure must have a "next" pointer. 39 * To use doubly-linked lists, your structure must "prev" and "next" pointers. 40 * Either way, the pointer to the head of the list must be initialized to NULL. 41 * 42 * ----------------.EXAMPLE ------------------------- 43 * struct item { 44 * int id; 45 * struct item *prev, *next; 46 * } 47 * 48 * struct item *list = NULL: 49 * 50 * int main() { 51 * struct item *item; 52 * ... allocate and populate item ... 53 * DL_APPEND(list, item); 54 * } 55 * -------------------------------------------------- 56 * 57 * For doubly-linked lists, the append and delete macros are O(1) 58 * For singly-linked lists, append and delete are O(n) but prepend is O(1) 59 * The sort macro is O(n log(n)) for all types of single/double/circular lists. 60 */ 61 62 /* These macros use decltype or the earlier __typeof GNU extension. 63 As decltype is only available in newer compilers (VS2010 or gcc 4.3+ 64 when compiling c++ source) this code uses whatever method is needed 65 or, for VS2008 where neither is available, uses casting workarounds. */ 66 #if !defined(LDECLTYPE) && !defined(NO_DECLTYPE) 67 #if defined(_MSC_VER) /* MS compiler */ 68 #if _MSC_VER >= 1600 && defined(__cplusplus) /* VS2010 or newer in C++ mode */ 69 #define LDECLTYPE(x) decltype(x) 70 #else /* VS2008 or older (or VS2010 in C mode) */ 71 #define NO_DECLTYPE 72 #endif 73 #elif defined(__BORLANDC__) || defined(__ICCARM__) || defined(__LCC__) || defined(__WATCOMC__) 74 #define NO_DECLTYPE 75 #else /* GNU, Sun and other compilers */ 76 #define LDECLTYPE(x) __typeof(x) 77 #endif 78 #endif 79 80 /* for VS2008 we use some workarounds to get around the lack of decltype, 81 * namely, we always reassign our tmp variable to the list head if we need 82 * to dereference its prev/next pointers, and save/restore the real head.*/ 83 #ifdef NO_DECLTYPE 84 #define IF_NO_DECLTYPE(x) x 85 #define LDECLTYPE(x) char* 86 #define UTLIST_SV(elt,list) _tmp = (char*)(list); {char **_alias = (char**)&(list); *_alias = (elt); } 87 #define UTLIST_NEXT(elt,list,next) ((char*)((list)->next)) 88 #define UTLIST_NEXTASGN(elt,list,to,next) { char **_alias = (char**)&((list)->next); *_alias=(char*)(to); } 89 /* #define UTLIST_PREV(elt,list,prev) ((char*)((list)->prev)) */ 90 #define UTLIST_PREVASGN(elt,list,to,prev) { char **_alias = (char**)&((list)->prev); *_alias=(char*)(to); } 91 #define UTLIST_RS(list) { char **_alias = (char**)&(list); *_alias=_tmp; } 92 #define UTLIST_CASTASGN(a,b) { char **_alias = (char**)&(a); *_alias=(char*)(b); } 93 #else 94 #define IF_NO_DECLTYPE(x) 95 #define UTLIST_SV(elt,list) 96 #define UTLIST_NEXT(elt,list,next) ((elt)->next) 97 #define UTLIST_NEXTASGN(elt,list,to,next) ((elt)->next)=(to) 98 /* #define UTLIST_PREV(elt,list,prev) ((elt)->prev) */ 99 #define UTLIST_PREVASGN(elt,list,to,prev) ((elt)->prev)=(to) 100 #define UTLIST_RS(list) 101 #define UTLIST_CASTASGN(a,b) (a)=(b) 102 #endif 103 104 /****************************************************************************** 105 * The sort macro is an adaptation of Simon Tatham's O(n log(n)) mergesort * 106 * Unwieldy variable names used here to avoid shadowing passed-in variables. * 107 *****************************************************************************/ 108 #define LL_SORT(list, cmp) \ 109 LL_SORT2(list, cmp, next) 110 111 #define LL_SORT2(list, cmp, next) \ 112 do { \ 113 LDECLTYPE(list) _ls_p; \ 114 LDECLTYPE(list) _ls_q; \ 115 LDECLTYPE(list) _ls_e; \ 116 LDECLTYPE(list) _ls_tail; \ 117 IF_NO_DECLTYPE(LDECLTYPE(list) _tmp;) \ 118 int _ls_insize, _ls_nmerges, _ls_psize, _ls_qsize, _ls_i, _ls_looping; \ 119 if (list) { \ 120 _ls_insize = 1; \ 121 _ls_looping = 1; \ 122 while (_ls_looping) { \ 123 UTLIST_CASTASGN(_ls_p,list); \ 124 (list) = NULL; \ 125 _ls_tail = NULL; \ 126 _ls_nmerges = 0; \ 127 while (_ls_p) { \ 128 _ls_nmerges++; \ 129 _ls_q = _ls_p; \ 130 _ls_psize = 0; \ 131 for (_ls_i = 0; _ls_i < _ls_insize; _ls_i++) { \ 132 _ls_psize++; \ 133 UTLIST_SV(_ls_q,list); _ls_q = UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); \ 134 if (!_ls_q) break; \ 135 } \ 136 _ls_qsize = _ls_insize; \ 137 while (_ls_psize > 0 || (_ls_qsize > 0 && _ls_q)) { \ 138 if (_ls_psize == 0) { \ 139 _ls_e = _ls_q; UTLIST_SV(_ls_q,list); _ls_q = \ 140 UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); _ls_qsize--; \ 141 } else if (_ls_qsize == 0 || !_ls_q) { \ 142 _ls_e = _ls_p; UTLIST_SV(_ls_p,list); _ls_p = \ 143 UTLIST_NEXT(_ls_p,list,next); UTLIST_RS(list); _ls_psize--; \ 144 } else if (cmp(_ls_p,_ls_q) <= 0) { \ 145 _ls_e = _ls_p; UTLIST_SV(_ls_p,list); _ls_p = \ 146 UTLIST_NEXT(_ls_p,list,next); UTLIST_RS(list); _ls_psize--; \ 147 } else { \ 148 _ls_e = _ls_q; UTLIST_SV(_ls_q,list); _ls_q = \ 149 UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); _ls_qsize--; \ 150 } \ 151 if (_ls_tail) { \ 152 UTLIST_SV(_ls_tail,list); UTLIST_NEXTASGN(_ls_tail,list,_ls_e,next); UTLIST_RS(list); \ 153 } else { \ 154 UTLIST_CASTASGN(list,_ls_e); \ 155 } \ 156 _ls_tail = _ls_e; \ 157 } \ 158 _ls_p = _ls_q; \ 159 } \ 160 if (_ls_tail) { \ 161 UTLIST_SV(_ls_tail,list); UTLIST_NEXTASGN(_ls_tail,list,NULL,next); UTLIST_RS(list); \ 162 } \ 163 if (_ls_nmerges <= 1) { \ 164 _ls_looping=0; \ 165 } \ 166 _ls_insize *= 2; \ 167 } \ 168 } \ 169 } while (0) 170 171 172 #define DL_SORT(list, cmp) \ 173 DL_SORT2(list, cmp, prev, next) 174 175 #define DL_SORT2(list, cmp, prev, next) \ 176 do { \ 177 LDECLTYPE(list) _ls_p; \ 178 LDECLTYPE(list) _ls_q; \ 179 LDECLTYPE(list) _ls_e; \ 180 LDECLTYPE(list) _ls_tail; \ 181 IF_NO_DECLTYPE(LDECLTYPE(list) _tmp;) \ 182 int _ls_insize, _ls_nmerges, _ls_psize, _ls_qsize, _ls_i, _ls_looping; \ 183 if (list) { \ 184 _ls_insize = 1; \ 185 _ls_looping = 1; \ 186 while (_ls_looping) { \ 187 UTLIST_CASTASGN(_ls_p,list); \ 188 (list) = NULL; \ 189 _ls_tail = NULL; \ 190 _ls_nmerges = 0; \ 191 while (_ls_p) { \ 192 _ls_nmerges++; \ 193 _ls_q = _ls_p; \ 194 _ls_psize = 0; \ 195 for (_ls_i = 0; _ls_i < _ls_insize; _ls_i++) { \ 196 _ls_psize++; \ 197 UTLIST_SV(_ls_q,list); _ls_q = UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); \ 198 if (!_ls_q) break; \ 199 } \ 200 _ls_qsize = _ls_insize; \ 201 while ((_ls_psize > 0) || ((_ls_qsize > 0) && _ls_q)) { \ 202 if (_ls_psize == 0) { \ 203 _ls_e = _ls_q; UTLIST_SV(_ls_q,list); _ls_q = \ 204 UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); _ls_qsize--; \ 205 } else if ((_ls_qsize == 0) || (!_ls_q)) { \ 206 _ls_e = _ls_p; UTLIST_SV(_ls_p,list); _ls_p = \ 207 UTLIST_NEXT(_ls_p,list,next); UTLIST_RS(list); _ls_psize--; \ 208 } else if (cmp(_ls_p,_ls_q) <= 0) { \ 209 _ls_e = _ls_p; UTLIST_SV(_ls_p,list); _ls_p = \ 210 UTLIST_NEXT(_ls_p,list,next); UTLIST_RS(list); _ls_psize--; \ 211 } else { \ 212 _ls_e = _ls_q; UTLIST_SV(_ls_q,list); _ls_q = \ 213 UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); _ls_qsize--; \ 214 } \ 215 if (_ls_tail) { \ 216 UTLIST_SV(_ls_tail,list); UTLIST_NEXTASGN(_ls_tail,list,_ls_e,next); UTLIST_RS(list); \ 217 } else { \ 218 UTLIST_CASTASGN(list,_ls_e); \ 219 } \ 220 UTLIST_SV(_ls_e,list); UTLIST_PREVASGN(_ls_e,list,_ls_tail,prev); UTLIST_RS(list); \ 221 _ls_tail = _ls_e; \ 222 } \ 223 _ls_p = _ls_q; \ 224 } \ 225 UTLIST_CASTASGN((list)->prev, _ls_tail); \ 226 UTLIST_SV(_ls_tail,list); UTLIST_NEXTASGN(_ls_tail,list,NULL,next); UTLIST_RS(list); \ 227 if (_ls_nmerges <= 1) { \ 228 _ls_looping=0; \ 229 } \ 230 _ls_insize *= 2; \ 231 } \ 232 } \ 233 } while (0) 234 235 #define CDL_SORT(list, cmp) \ 236 CDL_SORT2(list, cmp, prev, next) 237 238 #define CDL_SORT2(list, cmp, prev, next) \ 239 do { \ 240 LDECLTYPE(list) _ls_p; \ 241 LDECLTYPE(list) _ls_q; \ 242 LDECLTYPE(list) _ls_e; \ 243 LDECLTYPE(list) _ls_tail; \ 244 LDECLTYPE(list) _ls_oldhead; \ 245 LDECLTYPE(list) _tmp; \ 246 int _ls_insize, _ls_nmerges, _ls_psize, _ls_qsize, _ls_i, _ls_looping; \ 247 if (list) { \ 248 _ls_insize = 1; \ 249 _ls_looping = 1; \ 250 while (_ls_looping) { \ 251 UTLIST_CASTASGN(_ls_p,list); \ 252 UTLIST_CASTASGN(_ls_oldhead,list); \ 253 (list) = NULL; \ 254 _ls_tail = NULL; \ 255 _ls_nmerges = 0; \ 256 while (_ls_p) { \ 257 _ls_nmerges++; \ 258 _ls_q = _ls_p; \ 259 _ls_psize = 0; \ 260 for (_ls_i = 0; _ls_i < _ls_insize; _ls_i++) { \ 261 _ls_psize++; \ 262 UTLIST_SV(_ls_q,list); \ 263 if (UTLIST_NEXT(_ls_q,list,next) == _ls_oldhead) { \ 264 _ls_q = NULL; \ 265 } else { \ 266 _ls_q = UTLIST_NEXT(_ls_q,list,next); \ 267 } \ 268 UTLIST_RS(list); \ 269 if (!_ls_q) break; \ 270 } \ 271 _ls_qsize = _ls_insize; \ 272 while (_ls_psize > 0 || (_ls_qsize > 0 && _ls_q)) { \ 273 if (_ls_psize == 0) { \ 274 _ls_e = _ls_q; UTLIST_SV(_ls_q,list); _ls_q = \ 275 UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); _ls_qsize--; \ 276 if (_ls_q == _ls_oldhead) { _ls_q = NULL; } \ 277 } else if (_ls_qsize == 0 || !_ls_q) { \ 278 _ls_e = _ls_p; UTLIST_SV(_ls_p,list); _ls_p = \ 279 UTLIST_NEXT(_ls_p,list,next); UTLIST_RS(list); _ls_psize--; \ 280 if (_ls_p == _ls_oldhead) { _ls_p = NULL; } \ 281 } else if (cmp(_ls_p,_ls_q) <= 0) { \ 282 _ls_e = _ls_p; UTLIST_SV(_ls_p,list); _ls_p = \ 283 UTLIST_NEXT(_ls_p,list,next); UTLIST_RS(list); _ls_psize--; \ 284 if (_ls_p == _ls_oldhead) { _ls_p = NULL; } \ 285 } else { \ 286 _ls_e = _ls_q; UTLIST_SV(_ls_q,list); _ls_q = \ 287 UTLIST_NEXT(_ls_q,list,next); UTLIST_RS(list); _ls_qsize--; \ 288 if (_ls_q == _ls_oldhead) { _ls_q = NULL; } \ 289 } \ 290 if (_ls_tail) { \ 291 UTLIST_SV(_ls_tail,list); UTLIST_NEXTASGN(_ls_tail,list,_ls_e,next); UTLIST_RS(list); \ 292 } else { \ 293 UTLIST_CASTASGN(list,_ls_e); \ 294 } \ 295 UTLIST_SV(_ls_e,list); UTLIST_PREVASGN(_ls_e,list,_ls_tail,prev); UTLIST_RS(list); \ 296 _ls_tail = _ls_e; \ 297 } \ 298 _ls_p = _ls_q; \ 299 } \ 300 UTLIST_CASTASGN((list)->prev,_ls_tail); \ 301 UTLIST_CASTASGN(_tmp,list); \ 302 UTLIST_SV(_ls_tail,list); UTLIST_NEXTASGN(_ls_tail,list,_tmp,next); UTLIST_RS(list); \ 303 if (_ls_nmerges <= 1) { \ 304 _ls_looping=0; \ 305 } \ 306 _ls_insize *= 2; \ 307 } \ 308 } \ 309 } while (0) 310 311 /****************************************************************************** 312 * singly linked list macros (non-circular) * 313 *****************************************************************************/ 314 #define LL_PREPEND(head,add) \ 315 LL_PREPEND2(head,add,next) 316 317 #define LL_PREPEND2(head,add,next) \ 318 do { \ 319 (add)->next = (head); \ 320 (head) = (add); \ 321 } while (0) 322 323 #define LL_CONCAT(head1,head2) \ 324 LL_CONCAT2(head1,head2,next) 325 326 #define LL_CONCAT2(head1,head2,next) \ 327 do { \ 328 LDECLTYPE(head1) _tmp; \ 329 if (head1) { \ 330 _tmp = (head1); \ 331 while (_tmp->next) { _tmp = _tmp->next; } \ 332 _tmp->next=(head2); \ 333 } else { \ 334 (head1)=(head2); \ 335 } \ 336 } while (0) 337 338 #define LL_APPEND(head,add) \ 339 LL_APPEND2(head,add,next) 340 341 #define LL_APPEND2(head,add,next) \ 342 do { \ 343 LDECLTYPE(head) _tmp; \ 344 (add)->next=NULL; \ 345 if (head) { \ 346 _tmp = (head); \ 347 while (_tmp->next) { _tmp = _tmp->next; } \ 348 _tmp->next=(add); \ 349 } else { \ 350 (head)=(add); \ 351 } \ 352 } while (0) 353 354 #define LL_INSERT_INORDER(head,add,cmp) \ 355 LL_INSERT_INORDER2(head,add,cmp,next) 356 357 #define LL_INSERT_INORDER2(head,add,cmp,next) \ 358 do { \ 359 LDECLTYPE(head) _tmp; \ 360 if (head) { \ 361 LL_LOWER_BOUND2(head, _tmp, add, cmp, next); \ 362 LL_APPEND_ELEM2(head, _tmp, add, next); \ 363 } else { \ 364 (head) = (add); \ 365 (head)->next = NULL; \ 366 } \ 367 } while (0) 368 369 #define LL_LOWER_BOUND(head,elt,like,cmp) \ 370 LL_LOWER_BOUND2(head,elt,like,cmp,next) 371 372 #define LL_LOWER_BOUND2(head,elt,like,cmp,next) \ 373 do { \ 374 if ((head) == NULL || (cmp(head, like)) >= 0) { \ 375 (elt) = NULL; \ 376 } else { \ 377 for ((elt) = (head); (elt)->next != NULL; (elt) = (elt)->next) { \ 378 if (cmp((elt)->next, like) >= 0) { \ 379 break; \ 380 } \ 381 } \ 382 } \ 383 } while (0) 384 385 #define LL_DELETE(head,del) \ 386 LL_DELETE2(head,del,next) 387 388 #define LL_DELETE2(head,del,next) \ 389 do { \ 390 LDECLTYPE(head) _tmp; \ 391 if ((head) == (del)) { \ 392 (head)=(head)->next; \ 393 } else { \ 394 _tmp = (head); \ 395 while (_tmp->next && (_tmp->next != (del))) { \ 396 _tmp = _tmp->next; \ 397 } \ 398 if (_tmp->next) { \ 399 _tmp->next = (del)->next; \ 400 } \ 401 } \ 402 } while (0) 403 404 #define LL_COUNT(head,el,counter) \ 405 LL_COUNT2(head,el,counter,next) \ 406 407 #define LL_COUNT2(head,el,counter,next) \ 408 do { \ 409 (counter) = 0; \ 410 LL_FOREACH2(head,el,next) { ++(counter); } \ 411 } while (0) 412 413 #define LL_FOREACH(head,el) \ 414 LL_FOREACH2(head,el,next) 415 416 #define LL_FOREACH2(head,el,next) \ 417 for ((el) = (head); el; (el) = (el)->next) 418 419 #define LL_FOREACH_SAFE(head,el,tmp) \ 420 LL_FOREACH_SAFE2(head,el,tmp,next) 421 422 #define LL_FOREACH_SAFE2(head,el,tmp,next) \ 423 for ((el) = (head); (el) && ((tmp) = (el)->next, 1); (el) = (tmp)) 424 425 #define LL_SEARCH_SCALAR(head,out,field,val) \ 426 LL_SEARCH_SCALAR2(head,out,field,val,next) 427 428 #define LL_SEARCH_SCALAR2(head,out,field,val,next) \ 429 do { \ 430 LL_FOREACH2(head,out,next) { \ 431 if ((out)->field == (val)) break; \ 432 } \ 433 } while (0) 434 435 #define LL_SEARCH(head,out,elt,cmp) \ 436 LL_SEARCH2(head,out,elt,cmp,next) 437 438 #define LL_SEARCH2(head,out,elt,cmp,next) \ 439 do { \ 440 LL_FOREACH2(head,out,next) { \ 441 if ((cmp(out,elt))==0) break; \ 442 } \ 443 } while (0) 444 445 #define LL_REPLACE_ELEM2(head, el, add, next) \ 446 do { \ 447 LDECLTYPE(head) _tmp; \ 448 assert((head) != NULL); \ 449 assert((el) != NULL); \ 450 assert((add) != NULL); \ 451 (add)->next = (el)->next; \ 452 if ((head) == (el)) { \ 453 (head) = (add); \ 454 } else { \ 455 _tmp = (head); \ 456 while (_tmp->next && (_tmp->next != (el))) { \ 457 _tmp = _tmp->next; \ 458 } \ 459 if (_tmp->next) { \ 460 _tmp->next = (add); \ 461 } \ 462 } \ 463 } while (0) 464 465 #define LL_REPLACE_ELEM(head, el, add) \ 466 LL_REPLACE_ELEM2(head, el, add, next) 467 468 #define LL_PREPEND_ELEM2(head, el, add, next) \ 469 do { \ 470 if (el) { \ 471 LDECLTYPE(head) _tmp; \ 472 assert((head) != NULL); \ 473 assert((add) != NULL); \ 474 (add)->next = (el); \ 475 if ((head) == (el)) { \ 476 (head) = (add); \ 477 } else { \ 478 _tmp = (head); \ 479 while (_tmp->next && (_tmp->next != (el))) { \ 480 _tmp = _tmp->next; \ 481 } \ 482 if (_tmp->next) { \ 483 _tmp->next = (add); \ 484 } \ 485 } \ 486 } else { \ 487 LL_APPEND2(head, add, next); \ 488 } \ 489 } while (0) \ 490 491 #define LL_PREPEND_ELEM(head, el, add) \ 492 LL_PREPEND_ELEM2(head, el, add, next) 493 494 #define LL_APPEND_ELEM2(head, el, add, next) \ 495 do { \ 496 if (el) { \ 497 assert((head) != NULL); \ 498 assert((add) != NULL); \ 499 (add)->next = (el)->next; \ 500 (el)->next = (add); \ 501 } else { \ 502 LL_PREPEND2(head, add, next); \ 503 } \ 504 } while (0) \ 505 506 #define LL_APPEND_ELEM(head, el, add) \ 507 LL_APPEND_ELEM2(head, el, add, next) 508 509 #ifdef NO_DECLTYPE 510 /* Here are VS2008 / NO_DECLTYPE replacements for a few functions */ 511 512 #undef LL_CONCAT2 513 #define LL_CONCAT2(head1,head2,next) \ 514 do { \ 515 char *_tmp; \ 516 if (head1) { \ 517 _tmp = (char*)(head1); \ 518 while ((head1)->next) { (head1) = (head1)->next; } \ 519 (head1)->next = (head2); \ 520 UTLIST_RS(head1); \ 521 } else { \ 522 (head1)=(head2); \ 523 } \ 524 } while (0) 525 526 #undef LL_APPEND2 527 #define LL_APPEND2(head,add,next) \ 528 do { \ 529 if (head) { \ 530 (add)->next = head; /* use add->next as a temp variable */ \ 531 while ((add)->next->next) { (add)->next = (add)->next->next; } \ 532 (add)->next->next=(add); \ 533 } else { \ 534 (head)=(add); \ 535 } \ 536 (add)->next=NULL; \ 537 } while (0) 538 539 #undef LL_INSERT_INORDER2 540 #define LL_INSERT_INORDER2(head,add,cmp,next) \ 541 do { \ 542 if ((head) == NULL || (cmp(head, add)) >= 0) { \ 543 (add)->next = (head); \ 544 (head) = (add); \ 545 } else { \ 546 char *_tmp = (char*)(head); \ 547 while ((head)->next != NULL && (cmp((head)->next, add)) < 0) { \ 548 (head) = (head)->next; \ 549 } \ 550 (add)->next = (head)->next; \ 551 (head)->next = (add); \ 552 UTLIST_RS(head); \ 553 } \ 554 } while (0) 555 556 #undef LL_DELETE2 557 #define LL_DELETE2(head,del,next) \ 558 do { \ 559 if ((head) == (del)) { \ 560 (head)=(head)->next; \ 561 } else { \ 562 char *_tmp = (char*)(head); \ 563 while ((head)->next && ((head)->next != (del))) { \ 564 (head) = (head)->next; \ 565 } \ 566 if ((head)->next) { \ 567 (head)->next = ((del)->next); \ 568 } \ 569 UTLIST_RS(head); \ 570 } \ 571 } while (0) 572 573 #undef LL_REPLACE_ELEM2 574 #define LL_REPLACE_ELEM2(head, el, add, next) \ 575 do { \ 576 assert((head) != NULL); \ 577 assert((el) != NULL); \ 578 assert((add) != NULL); \ 579 if ((head) == (el)) { \ 580 (head) = (add); \ 581 } else { \ 582 (add)->next = head; \ 583 while ((add)->next->next && ((add)->next->next != (el))) { \ 584 (add)->next = (add)->next->next; \ 585 } \ 586 if ((add)->next->next) { \ 587 (add)->next->next = (add); \ 588 } \ 589 } \ 590 (add)->next = (el)->next; \ 591 } while (0) 592 593 #undef LL_PREPEND_ELEM2 594 #define LL_PREPEND_ELEM2(head, el, add, next) \ 595 do { \ 596 if (el) { \ 597 assert((head) != NULL); \ 598 assert((add) != NULL); \ 599 if ((head) == (el)) { \ 600 (head) = (add); \ 601 } else { \ 602 (add)->next = (head); \ 603 while ((add)->next->next && ((add)->next->next != (el))) { \ 604 (add)->next = (add)->next->next; \ 605 } \ 606 if ((add)->next->next) { \ 607 (add)->next->next = (add); \ 608 } \ 609 } \ 610 (add)->next = (el); \ 611 } else { \ 612 LL_APPEND2(head, add, next); \ 613 } \ 614 } while (0) \ 615 616 #endif /* NO_DECLTYPE */ 617 618 /****************************************************************************** 619 * doubly linked list macros (non-circular) * 620 *****************************************************************************/ 621 #define DL_PREPEND(head,add) \ 622 DL_PREPEND2(head,add,prev,next) 623 624 #define DL_PREPEND2(head,add,prev,next) \ 625 do { \ 626 (add)->next = (head); \ 627 if (head) { \ 628 (add)->prev = (head)->prev; \ 629 (head)->prev = (add); \ 630 } else { \ 631 (add)->prev = (add); \ 632 } \ 633 (head) = (add); \ 634 } while (0) 635 636 #define DL_APPEND(head,add) \ 637 DL_APPEND2(head,add,prev,next) 638 639 #define DL_APPEND2(head,add,prev,next) \ 640 do { \ 641 if (head) { \ 642 (add)->prev = (head)->prev; \ 643 (head)->prev->next = (add); \ 644 (head)->prev = (add); \ 645 (add)->next = NULL; \ 646 } else { \ 647 (head)=(add); \ 648 (head)->prev = (head); \ 649 (head)->next = NULL; \ 650 } \ 651 } while (0) 652 653 #define DL_INSERT_INORDER(head,add,cmp) \ 654 DL_INSERT_INORDER2(head,add,cmp,prev,next) 655 656 #define DL_INSERT_INORDER2(head,add,cmp,prev,next) \ 657 do { \ 658 LDECLTYPE(head) _tmp; \ 659 if (head) { \ 660 DL_LOWER_BOUND2(head, _tmp, add, cmp, next); \ 661 DL_APPEND_ELEM2(head, _tmp, add, prev, next); \ 662 } else { \ 663 (head) = (add); \ 664 (head)->prev = (head); \ 665 (head)->next = NULL; \ 666 } \ 667 } while (0) 668 669 #define DL_LOWER_BOUND(head,elt,like,cmp) \ 670 DL_LOWER_BOUND2(head,elt,like,cmp,next) 671 672 #define DL_LOWER_BOUND2(head,elt,like,cmp,next) \ 673 do { \ 674 if ((head) == NULL || (cmp(head, like)) >= 0) { \ 675 (elt) = NULL; \ 676 } else { \ 677 for ((elt) = (head); (elt)->next != NULL; (elt) = (elt)->next) { \ 678 if ((cmp((elt)->next, like)) >= 0) { \ 679 break; \ 680 } \ 681 } \ 682 } \ 683 } while (0) 684 685 #define DL_CONCAT(head1,head2) \ 686 DL_CONCAT2(head1,head2,prev,next) 687 688 #define DL_CONCAT2(head1,head2,prev,next) \ 689 do { \ 690 LDECLTYPE(head1) _tmp; \ 691 if (head2) { \ 692 if (head1) { \ 693 UTLIST_CASTASGN(_tmp, (head2)->prev); \ 694 (head2)->prev = (head1)->prev; \ 695 (head1)->prev->next = (head2); \ 696 UTLIST_CASTASGN((head1)->prev, _tmp); \ 697 } else { \ 698 (head1)=(head2); \ 699 } \ 700 } \ 701 } while (0) 702 703 #define DL_DELETE(head,del) \ 704 DL_DELETE2(head,del,prev,next) 705 706 #define DL_DELETE2(head,del,prev,next) \ 707 do { \ 708 assert((head) != NULL); \ 709 assert((del)->prev != NULL); \ 710 if ((del)->prev == (del)) { \ 711 (head)=NULL; \ 712 } else if ((del)==(head)) { \ 713 (del)->next->prev = (del)->prev; \ 714 (head) = (del)->next; \ 715 } else { \ 716 (del)->prev->next = (del)->next; \ 717 if ((del)->next) { \ 718 (del)->next->prev = (del)->prev; \ 719 } else { \ 720 (head)->prev = (del)->prev; \ 721 } \ 722 } \ 723 } while (0) 724 725 #define DL_COUNT(head,el,counter) \ 726 DL_COUNT2(head,el,counter,next) \ 727 728 #define DL_COUNT2(head,el,counter,next) \ 729 do { \ 730 (counter) = 0; \ 731 DL_FOREACH2(head,el,next) { ++(counter); } \ 732 } while (0) 733 734 #define DL_FOREACH(head,el) \ 735 DL_FOREACH2(head,el,next) 736 737 #define DL_FOREACH2(head,el,next) \ 738 for ((el) = (head); el; (el) = (el)->next) 739 740 /* this version is safe for deleting the elements during iteration */ 741 #define DL_FOREACH_SAFE(head,el,tmp) \ 742 DL_FOREACH_SAFE2(head,el,tmp,next) 743 744 #define DL_FOREACH_SAFE2(head,el,tmp,next) \ 745 for ((el) = (head); (el) && ((tmp) = (el)->next, 1); (el) = (tmp)) 746 747 /* these are identical to their singly-linked list counterparts */ 748 #define DL_SEARCH_SCALAR LL_SEARCH_SCALAR 749 #define DL_SEARCH LL_SEARCH 750 #define DL_SEARCH_SCALAR2 LL_SEARCH_SCALAR2 751 #define DL_SEARCH2 LL_SEARCH2 752 753 #define DL_REPLACE_ELEM2(head, el, add, prev, next) \ 754 do { \ 755 assert((head) != NULL); \ 756 assert((el) != NULL); \ 757 assert((add) != NULL); \ 758 if ((head) == (el)) { \ 759 (head) = (add); \ 760 (add)->next = (el)->next; \ 761 if ((el)->next == NULL) { \ 762 (add)->prev = (add); \ 763 } else { \ 764 (add)->prev = (el)->prev; \ 765 (add)->next->prev = (add); \ 766 } \ 767 } else { \ 768 (add)->next = (el)->next; \ 769 (add)->prev = (el)->prev; \ 770 (add)->prev->next = (add); \ 771 if ((el)->next == NULL) { \ 772 (head)->prev = (add); \ 773 } else { \ 774 (add)->next->prev = (add); \ 775 } \ 776 } \ 777 } while (0) 778 779 #define DL_REPLACE_ELEM(head, el, add) \ 780 DL_REPLACE_ELEM2(head, el, add, prev, next) 781 782 #define DL_PREPEND_ELEM2(head, el, add, prev, next) \ 783 do { \ 784 if (el) { \ 785 assert((head) != NULL); \ 786 assert((add) != NULL); \ 787 (add)->next = (el); \ 788 (add)->prev = (el)->prev; \ 789 (el)->prev = (add); \ 790 if ((head) == (el)) { \ 791 (head) = (add); \ 792 } else { \ 793 (add)->prev->next = (add); \ 794 } \ 795 } else { \ 796 DL_APPEND2(head, add, prev, next); \ 797 } \ 798 } while (0) \ 799 800 #define DL_PREPEND_ELEM(head, el, add) \ 801 DL_PREPEND_ELEM2(head, el, add, prev, next) 802 803 #define DL_APPEND_ELEM2(head, el, add, prev, next) \ 804 do { \ 805 if (el) { \ 806 assert((head) != NULL); \ 807 assert((add) != NULL); \ 808 (add)->next = (el)->next; \ 809 (add)->prev = (el); \ 810 (el)->next = (add); \ 811 if ((add)->next) { \ 812 (add)->next->prev = (add); \ 813 } else { \ 814 (head)->prev = (add); \ 815 } \ 816 } else { \ 817 DL_PREPEND2(head, add, prev, next); \ 818 } \ 819 } while (0) \ 820 821 #define DL_APPEND_ELEM(head, el, add) \ 822 DL_APPEND_ELEM2(head, el, add, prev, next) 823 824 #ifdef NO_DECLTYPE 825 /* Here are VS2008 / NO_DECLTYPE replacements for a few functions */ 826 827 #undef DL_INSERT_INORDER2 828 #define DL_INSERT_INORDER2(head,add,cmp,prev,next) \ 829 do { \ 830 if ((head) == NULL) { \ 831 (add)->prev = (add); \ 832 (add)->next = NULL; \ 833 (head) = (add); \ 834 } else if ((cmp(head, add)) >= 0) { \ 835 (add)->prev = (head)->prev; \ 836 (add)->next = (head); \ 837 (head)->prev = (add); \ 838 (head) = (add); \ 839 } else { \ 840 char *_tmp = (char*)(head); \ 841 while ((head)->next && (cmp((head)->next, add)) < 0) { \ 842 (head) = (head)->next; \ 843 } \ 844 (add)->prev = (head); \ 845 (add)->next = (head)->next; \ 846 (head)->next = (add); \ 847 UTLIST_RS(head); \ 848 if ((add)->next) { \ 849 (add)->next->prev = (add); \ 850 } else { \ 851 (head)->prev = (add); \ 852 } \ 853 } \ 854 } while (0) 855 #endif /* NO_DECLTYPE */ 856 857 /****************************************************************************** 858 * circular doubly linked list macros * 859 *****************************************************************************/ 860 #define CDL_APPEND(head,add) \ 861 CDL_APPEND2(head,add,prev,next) 862 863 #define CDL_APPEND2(head,add,prev,next) \ 864 do { \ 865 if (head) { \ 866 (add)->prev = (head)->prev; \ 867 (add)->next = (head); \ 868 (head)->prev = (add); \ 869 (add)->prev->next = (add); \ 870 } else { \ 871 (add)->prev = (add); \ 872 (add)->next = (add); \ 873 (head) = (add); \ 874 } \ 875 } while (0) 876 877 #define CDL_PREPEND(head,add) \ 878 CDL_PREPEND2(head,add,prev,next) 879 880 #define CDL_PREPEND2(head,add,prev,next) \ 881 do { \ 882 if (head) { \ 883 (add)->prev = (head)->prev; \ 884 (add)->next = (head); \ 885 (head)->prev = (add); \ 886 (add)->prev->next = (add); \ 887 } else { \ 888 (add)->prev = (add); \ 889 (add)->next = (add); \ 890 } \ 891 (head) = (add); \ 892 } while (0) 893 894 #define CDL_INSERT_INORDER(head,add,cmp) \ 895 CDL_INSERT_INORDER2(head,add,cmp,prev,next) 896 897 #define CDL_INSERT_INORDER2(head,add,cmp,prev,next) \ 898 do { \ 899 LDECLTYPE(head) _tmp; \ 900 if (head) { \ 901 CDL_LOWER_BOUND2(head, _tmp, add, cmp, next); \ 902 CDL_APPEND_ELEM2(head, _tmp, add, prev, next); \ 903 } else { \ 904 (head) = (add); \ 905 (head)->next = (head); \ 906 (head)->prev = (head); \ 907 } \ 908 } while (0) 909 910 #define CDL_LOWER_BOUND(head,elt,like,cmp) \ 911 CDL_LOWER_BOUND2(head,elt,like,cmp,next) 912 913 #define CDL_LOWER_BOUND2(head,elt,like,cmp,next) \ 914 do { \ 915 if ((head) == NULL || (cmp(head, like)) >= 0) { \ 916 (elt) = NULL; \ 917 } else { \ 918 for ((elt) = (head); (elt)->next != (head); (elt) = (elt)->next) { \ 919 if ((cmp((elt)->next, like)) >= 0) { \ 920 break; \ 921 } \ 922 } \ 923 } \ 924 } while (0) 925 926 #define CDL_DELETE(head,del) \ 927 CDL_DELETE2(head,del,prev,next) 928 929 #define CDL_DELETE2(head,del,prev,next) \ 930 do { \ 931 if (((head)==(del)) && ((head)->next == (head))) { \ 932 (head) = NULL; \ 933 } else { \ 934 (del)->next->prev = (del)->prev; \ 935 (del)->prev->next = (del)->next; \ 936 if ((del) == (head)) (head)=(del)->next; \ 937 } \ 938 } while (0) 939 940 #define CDL_COUNT(head,el,counter) \ 941 CDL_COUNT2(head,el,counter,next) \ 942 943 #define CDL_COUNT2(head, el, counter,next) \ 944 do { \ 945 (counter) = 0; \ 946 CDL_FOREACH2(head,el,next) { ++(counter); } \ 947 } while (0) 948 949 #define CDL_FOREACH(head,el) \ 950 CDL_FOREACH2(head,el,next) 951 952 #define CDL_FOREACH2(head,el,next) \ 953 for ((el)=(head);el;(el)=(((el)->next==(head)) ? NULL : (el)->next)) 954 955 #define CDL_FOREACH_SAFE(head,el,tmp1,tmp2) \ 956 CDL_FOREACH_SAFE2(head,el,tmp1,tmp2,prev,next) 957 958 #define CDL_FOREACH_SAFE2(head,el,tmp1,tmp2,prev,next) \ 959 for ((el) = (head), (tmp1) = (head) ? (head)->prev : NULL; \ 960 (el) && ((tmp2) = (el)->next, 1); \ 961 (el) = ((el) == (tmp1) ? NULL : (tmp2))) 962 963 #define CDL_SEARCH_SCALAR(head,out,field,val) \ 964 CDL_SEARCH_SCALAR2(head,out,field,val,next) 965 966 #define CDL_SEARCH_SCALAR2(head,out,field,val,next) \ 967 do { \ 968 CDL_FOREACH2(head,out,next) { \ 969 if ((out)->field == (val)) break; \ 970 } \ 971 } while (0) 972 973 #define CDL_SEARCH(head,out,elt,cmp) \ 974 CDL_SEARCH2(head,out,elt,cmp,next) 975 976 #define CDL_SEARCH2(head,out,elt,cmp,next) \ 977 do { \ 978 CDL_FOREACH2(head,out,next) { \ 979 if ((cmp(out,elt))==0) break; \ 980 } \ 981 } while (0) 982 983 #define CDL_REPLACE_ELEM2(head, el, add, prev, next) \ 984 do { \ 985 assert((head) != NULL); \ 986 assert((el) != NULL); \ 987 assert((add) != NULL); \ 988 if ((el)->next == (el)) { \ 989 (add)->next = (add); \ 990 (add)->prev = (add); \ 991 (head) = (add); \ 992 } else { \ 993 (add)->next = (el)->next; \ 994 (add)->prev = (el)->prev; \ 995 (add)->next->prev = (add); \ 996 (add)->prev->next = (add); \ 997 if ((head) == (el)) { \ 998 (head) = (add); \ 999 } \ 1000 } \ 1001 } while (0) 1002 1003 #define CDL_REPLACE_ELEM(head, el, add) \ 1004 CDL_REPLACE_ELEM2(head, el, add, prev, next) 1005 1006 #define CDL_PREPEND_ELEM2(head, el, add, prev, next) \ 1007 do { \ 1008 if (el) { \ 1009 assert((head) != NULL); \ 1010 assert((add) != NULL); \ 1011 (add)->next = (el); \ 1012 (add)->prev = (el)->prev; \ 1013 (el)->prev = (add); \ 1014 (add)->prev->next = (add); \ 1015 if ((head) == (el)) { \ 1016 (head) = (add); \ 1017 } \ 1018 } else { \ 1019 CDL_APPEND2(head, add, prev, next); \ 1020 } \ 1021 } while (0) 1022 1023 #define CDL_PREPEND_ELEM(head, el, add) \ 1024 CDL_PREPEND_ELEM2(head, el, add, prev, next) 1025 1026 #define CDL_APPEND_ELEM2(head, el, add, prev, next) \ 1027 do { \ 1028 if (el) { \ 1029 assert((head) != NULL); \ 1030 assert((add) != NULL); \ 1031 (add)->next = (el)->next; \ 1032 (add)->prev = (el); \ 1033 (el)->next = (add); \ 1034 (add)->next->prev = (add); \ 1035 } else { \ 1036 CDL_PREPEND2(head, add, prev, next); \ 1037 } \ 1038 } while (0) 1039 1040 #define CDL_APPEND_ELEM(head, el, add) \ 1041 CDL_APPEND_ELEM2(head, el, add, prev, next) 1042 1043 #ifdef NO_DECLTYPE 1044 /* Here are VS2008 / NO_DECLTYPE replacements for a few functions */ 1045 1046 #undef CDL_INSERT_INORDER2 1047 #define CDL_INSERT_INORDER2(head,add,cmp,prev,next) \ 1048 do { \ 1049 if ((head) == NULL) { \ 1050 (add)->prev = (add); \ 1051 (add)->next = (add); \ 1052 (head) = (add); \ 1053 } else if ((cmp(head, add)) >= 0) { \ 1054 (add)->prev = (head)->prev; \ 1055 (add)->next = (head); \ 1056 (add)->prev->next = (add); \ 1057 (head)->prev = (add); \ 1058 (head) = (add); \ 1059 } else { \ 1060 char *_tmp = (char*)(head); \ 1061 while ((char*)(head)->next != _tmp && (cmp((head)->next, add)) < 0) { \ 1062 (head) = (head)->next; \ 1063 } \ 1064 (add)->prev = (head); \ 1065 (add)->next = (head)->next; \ 1066 (add)->next->prev = (add); \ 1067 (head)->next = (add); \ 1068 UTLIST_RS(head); \ 1069 } \ 1070 } while (0) 1071 #endif /* NO_DECLTYPE */ 1072 1073 #endif /* UTLIST_H */ 1074