queue.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982
  1. static const char rcsid[] = "#(@) $Id$";
  2. /*
  3. * Date last modified: Jan 2001
  4. * Modifications by Dan Libby (dan@libby.com), including:
  5. * - various fixes, null checks, etc
  6. * - addition of Q_Iter funcs, macros
  7. */
  8. /*-**************************************************************
  9. *
  10. * File : q.c
  11. *
  12. * Author: Peter Yard [1993.01.02] -- 02 Jan 1993
  13. *
  14. * Disclaimer: This code is released to the public domain.
  15. *
  16. * Description:
  17. * Generic double ended queue (Deque pronounced DEK) for handling
  18. * any data types, with sorting.
  19. *
  20. * By use of various functions in this module the caller
  21. * can create stacks, queues, lists, doubly linked lists,
  22. * sorted lists, indexed lists. All lists are dynamic.
  23. *
  24. * It is the responsibility of the caller to malloc and free
  25. * memory for insertion into the queue. A pointer to the object
  26. * is used so that not only can any data be used but various kinds
  27. * of data can be pushed on the same queue if one so wished e.g.
  28. * various length string literals mixed with pointers to structures
  29. * or integers etc.
  30. *
  31. * Enhancements:
  32. * A future improvement would be the option of multiple "cursors"
  33. * so that multiple locations could occur in the one queue to allow
  34. * placemarkers and additional flexibility. Perhaps even use queue
  35. * itself to have a list of cursors.
  36. *
  37. * Usage:
  38. *
  39. * /x init queue x/
  40. * queue q;
  41. * Q_Init(&q);
  42. *
  43. * To create a stack :
  44. *
  45. * Q_PushHead(&q, &mydata1); /x push x/
  46. * Q_PushHead(&q, &mydata2);
  47. * .....
  48. * data_ptr = Q_PopHead(&q); /x pop x/
  49. * .....
  50. * data_ptr = Q_Head(&q); /x top of stack x/
  51. *
  52. * To create a FIFO:
  53. *
  54. * Q_PushHead(&q, &mydata1);
  55. * .....
  56. * data_ptr = Q_PopTail(&q);
  57. *
  58. * To create a double list:
  59. *
  60. * data_ptr = Q_Head(&q);
  61. * ....
  62. * data_ptr = Q_Next(&q);
  63. * data_ptr = Q_Tail(&q);
  64. * if (Q_IsEmpty(&q)) ....
  65. * .....
  66. * data_ptr = Q_Previous(&q);
  67. *
  68. * To create a sorted list:
  69. *
  70. * Q_PushHead(&q, &mydata1); /x push x/
  71. * Q_PushHead(&q, &mydata2);
  72. * .....
  73. * if (!Q_Sort(&q, MyFunction))
  74. * .. error ..
  75. *
  76. * /x fill in key field of mydata1.
  77. * * NB: Q_Find does linear search
  78. * x/
  79. *
  80. * if (Q_Find(&q, &mydata1, MyFunction))
  81. * {
  82. * /x found it, queue cursor now at correct record x/
  83. * /x can retrieve with x/
  84. * data_ptr = Q_Get(&q);
  85. *
  86. * /x alter data , write back with x/
  87. * Q_Put(&q, data_ptr);
  88. * }
  89. *
  90. * /x Search with binary search x/
  91. * if (Q_Seek(&q, &mydata, MyFunction))
  92. * /x etc x/
  93. *
  94. *
  95. ****************************************************************/
  96. #ifdef _WIN32
  97. #include "xmlrpc_win32.h"
  98. #endif
  99. #include <stdlib.h>
  100. #include "queue.h"
  101. static void QuickSort(void *list[], int low, int high,
  102. int (*Comp)(const void *, const void *));
  103. static int Q_BSearch(queue *q, void *key,
  104. int (*Comp)(const void *, const void *));
  105. /* The index: a pointer to pointers */
  106. static void **index;
  107. static datanode **posn_index;
  108. /***
  109. *
  110. ** function : Q_Init
  111. *
  112. ** purpose : Initialise queue object and pointers.
  113. *
  114. ** parameters : 'queue' pointer.
  115. *
  116. ** returns : True_ if init successful else False_
  117. *
  118. ** comments :
  119. ***/
  120. int Q_Init(queue *q)
  121. {
  122. if(q) {
  123. q->head = q->tail = NULL;
  124. q->cursor = q->head;
  125. q->size = 0;
  126. q->sorted = False_;
  127. }
  128. return True_;
  129. }
  130. /***
  131. *
  132. ** function : Q_AtHead
  133. *
  134. ** purpose : tests if cursor is at head of queue
  135. *
  136. ** parameters : 'queue' pointer.
  137. *
  138. ** returns : boolean - True_ is at head else False_
  139. *
  140. ** comments :
  141. *
  142. ***/
  143. int Q_AtHead(queue *q)
  144. {
  145. return(q && q->cursor == q->head);
  146. }
  147. /***
  148. *
  149. ** function : Q_AtTail
  150. *
  151. ** purpose : boolean test if cursor at tail of queue
  152. *
  153. ** parameters : 'queue' pointer to test.
  154. *
  155. ** returns : True_ or False_
  156. *
  157. ** comments :
  158. *
  159. ***/
  160. int Q_AtTail(queue *q)
  161. {
  162. return(q && q->cursor == q->tail);
  163. }
  164. /***
  165. *
  166. ** function : Q_IsEmpty
  167. *
  168. ** purpose : test if queue has nothing in it.
  169. *
  170. ** parameters : 'queue' pointer
  171. *
  172. ** returns : True_ if IsEmpty queue, else False_
  173. *
  174. ** comments :
  175. *
  176. ***/
  177. inline int Q_IsEmpty(queue *q)
  178. {
  179. return(!q || q->size == 0);
  180. }
  181. /***
  182. *
  183. ** function : Q_Size
  184. *
  185. ** purpose : return the number of elements in the queue
  186. *
  187. ** parameters : queue pointer
  188. *
  189. ** returns : number of elements
  190. *
  191. ** comments :
  192. *
  193. ***/
  194. int Q_Size(queue *q)
  195. {
  196. return q ? q->size : 0;
  197. }
  198. /***
  199. *
  200. ** function : Q_Head
  201. *
  202. ** purpose : position queue cursor to first element (head) of queue.
  203. *
  204. ** parameters : 'queue' pointer
  205. *
  206. ** returns : pointer to data at head. If queue is IsEmpty returns NULL
  207. *
  208. ** comments :
  209. *
  210. ***/
  211. void *Q_Head(queue *q)
  212. {
  213. if(Q_IsEmpty(q))
  214. return NULL;
  215. q->cursor = q->head;
  216. return q->cursor->data;
  217. }
  218. /***
  219. *
  220. ** function : Q_Tail
  221. *
  222. ** purpose : locate cursor at tail of queue.
  223. *
  224. ** parameters : 'queue' pointer
  225. *
  226. ** returns : pointer to data at tail , if queue IsEmpty returns NULL
  227. *
  228. ** comments :
  229. *
  230. ***/
  231. void *Q_Tail(queue *q)
  232. {
  233. if(Q_IsEmpty(q))
  234. return NULL;
  235. q->cursor = q->tail;
  236. return q->cursor->data;
  237. }
  238. /***
  239. *
  240. ** function : Q_PushHead
  241. *
  242. ** purpose : put a data pointer at the head of the queue
  243. *
  244. ** parameters : 'queue' pointer, void pointer to the data.
  245. *
  246. ** returns : True_ if success else False_ if unable to push data.
  247. *
  248. ** comments :
  249. *
  250. ***/
  251. int Q_PushHead(queue *q, void *d)
  252. {
  253. if(q && d) {
  254. node *n;
  255. datanode *p;
  256. p = malloc(sizeof(datanode));
  257. if(p == NULL)
  258. return False_;
  259. n = q->head;
  260. q->head = (node*)p;
  261. q->head->prev = NULL;
  262. if(q->size == 0) {
  263. q->head->next = NULL;
  264. q->tail = q->head;
  265. }
  266. else {
  267. q->head->next = (datanode*)n;
  268. n->prev = q->head;
  269. }
  270. q->head->data = d;
  271. q->size++;
  272. q->cursor = q->head;
  273. q->sorted = False_;
  274. return True_;
  275. }
  276. return False_;
  277. }
  278. /***
  279. *
  280. ** function : Q_PushTail
  281. *
  282. ** purpose : put a data element pointer at the tail of the queue
  283. *
  284. ** parameters : queue pointer, pointer to the data
  285. *
  286. ** returns : True_ if data pushed, False_ if data not inserted.
  287. *
  288. ** comments :
  289. *
  290. ***/
  291. int Q_PushTail(queue *q, void *d)
  292. {
  293. if(q && d) {
  294. node *p;
  295. datanode *n;
  296. n = malloc(sizeof(datanode));
  297. if(n == NULL)
  298. return False_;
  299. p = q->tail;
  300. q->tail = (node *)n;
  301. if(q->size == 0) {
  302. q->tail->prev = NULL;
  303. q->head = q->tail;
  304. }
  305. else {
  306. q->tail->prev = (datanode *)p;
  307. p->next = q->tail;
  308. }
  309. q->tail->next = NULL;
  310. q->tail->data = d;
  311. q->cursor = q->tail;
  312. q->size++;
  313. q->sorted = False_;
  314. return True_;
  315. }
  316. return False_;
  317. }
  318. /***
  319. *
  320. ** function : Q_PopHead
  321. *
  322. ** purpose : remove and return the top element at the head of the
  323. * queue.
  324. *
  325. ** parameters : queue pointer
  326. *
  327. ** returns : pointer to data element or NULL if queue is IsEmpty.
  328. *
  329. ** comments :
  330. *
  331. ***/
  332. void *Q_PopHead(queue *q)
  333. {
  334. datanode *n;
  335. void *d;
  336. if(Q_IsEmpty(q))
  337. return NULL;
  338. d = q->head->data;
  339. n = q->head->next;
  340. free(q->head);
  341. q->size--;
  342. if(q->size == 0)
  343. q->head = q->tail = q->cursor = NULL;
  344. else {
  345. q->head = (node *)n;
  346. q->head->prev = NULL;
  347. q->cursor = q->head;
  348. }
  349. q->sorted = False_;
  350. return d;
  351. }
  352. /***
  353. *
  354. ** function : Q_PopTail
  355. *
  356. ** purpose : remove element from tail of queue and return data.
  357. *
  358. ** parameters : queue pointer
  359. *
  360. ** returns : pointer to data element that was at tail. NULL if queue
  361. * IsEmpty.
  362. *
  363. ** comments :
  364. *
  365. ***/
  366. void *Q_PopTail(queue *q)
  367. {
  368. datanode *p;
  369. void *d;
  370. if(Q_IsEmpty(q))
  371. return NULL;
  372. d = q->tail->data;
  373. p = q->tail->prev;
  374. free(q->tail);
  375. q->size--;
  376. if(q->size == 0)
  377. q->head = q->tail = q->cursor = NULL;
  378. else {
  379. q->tail = (node *)p;
  380. q->tail->next = NULL;
  381. q->cursor = q->tail;
  382. }
  383. q->sorted = False_;
  384. return d;
  385. }
  386. /***
  387. *
  388. ** function : Q_Next
  389. *
  390. ** purpose : Move to the next element in the queue without popping
  391. *
  392. ** parameters : queue pointer.
  393. *
  394. ** returns : pointer to data element of new element or NULL if end
  395. * of the queue.
  396. *
  397. ** comments : This uses the cursor for the current position. Q_Next
  398. * only moves in the direction from the head of the queue
  399. * to the tail.
  400. ***/
  401. void *Q_Next(queue *q)
  402. {
  403. if(!q)
  404. return NULL;
  405. if(!q->cursor || q->cursor->next == NULL)
  406. return NULL;
  407. q->cursor = (node *)q->cursor->next;
  408. return q->cursor->data ;
  409. }
  410. /***
  411. *
  412. ** function : Q_Previous
  413. *
  414. ** purpose : Opposite of Q_Next. Move to next element closer to the
  415. * head of the queue.
  416. *
  417. ** parameters : pointer to queue
  418. *
  419. ** returns : pointer to data of new element else NULL if queue IsEmpty
  420. *
  421. ** comments : Makes cursor move towards the head of the queue.
  422. *
  423. ***/
  424. void *Q_Previous(queue *q)
  425. {
  426. if(!q)
  427. return NULL;
  428. if(q->cursor->prev == NULL)
  429. return NULL;
  430. q->cursor = (node *)q->cursor->prev;
  431. return q->cursor->data;
  432. }
  433. void *Q_Iter_Del(queue *q, q_iter iter)
  434. {
  435. void *d;
  436. datanode *n, *p;
  437. if(!q)
  438. return NULL;
  439. if(iter == NULL)
  440. return NULL;
  441. if(iter == (q_iter)q->head)
  442. return Q_PopHead(q);
  443. if(iter == (q_iter)q->tail)
  444. return Q_PopTail(q);
  445. n = ((node*)iter)->next;
  446. p = ((node*)iter)->prev;
  447. d = ((node*)iter)->data;
  448. free(iter);
  449. if(p) {
  450. p->next = n;
  451. }
  452. if (q->cursor == (node*)iter) {
  453. if (p) {
  454. q->cursor = p;
  455. } else {
  456. q->cursor = n;
  457. }
  458. }
  459. if (n != NULL) {
  460. n->prev = p;
  461. }
  462. q->size--;
  463. q->sorted = False_;
  464. return d;
  465. }
  466. /***
  467. *
  468. ** function : Q_DelCur
  469. *
  470. ** purpose : Delete the current queue element as pointed to by
  471. * the cursor.
  472. *
  473. ** parameters : queue pointer
  474. *
  475. ** returns : pointer to data element.
  476. *
  477. ** comments : WARNING! It is the responsibility of the caller to
  478. * free any memory. Queue cannot distinguish between
  479. * pointers to literals and malloced memory.
  480. *
  481. ***/
  482. void *Q_DelCur(queue* q) {
  483. if(q) {
  484. return Q_Iter_Del(q, (q_iter)q->cursor);
  485. }
  486. return 0;
  487. }
  488. /***
  489. *
  490. ** function : Q_Destroy
  491. *
  492. ** purpose : Free all queue resources
  493. *
  494. ** parameters : queue pointer
  495. *
  496. ** returns : null.
  497. *
  498. ** comments : WARNING! It is the responsibility of the caller to
  499. * free any memory. Queue cannot distinguish between
  500. * pointers to literals and malloced memory.
  501. *
  502. ***/
  503. void Q_Destroy(queue *q)
  504. {
  505. while(!Q_IsEmpty(q)) {
  506. Q_PopHead(q);
  507. }
  508. }
  509. /***
  510. *
  511. ** function : Q_Get
  512. *
  513. ** purpose : get the pointer to the data at the cursor location
  514. *
  515. ** parameters : queue pointer
  516. *
  517. ** returns : data element pointer
  518. *
  519. ** comments :
  520. *
  521. ***/
  522. void *Q_Get(queue *q)
  523. {
  524. if(!q)
  525. return NULL;
  526. if(q->cursor == NULL)
  527. return NULL;
  528. return q->cursor->data;
  529. }
  530. /***
  531. *
  532. ** function : Q_Put
  533. *
  534. ** purpose : replace pointer to data with new pointer to data.
  535. *
  536. ** parameters : queue pointer, data pointer
  537. *
  538. ** returns : boolean- True_ if successful, False_ if cursor at NULL
  539. *
  540. ** comments :
  541. *
  542. ***/
  543. int Q_Put(queue *q, void *data)
  544. {
  545. if(q && data) {
  546. if(q->cursor == NULL)
  547. return False_;
  548. q->cursor->data = data;
  549. return True_;
  550. }
  551. return False_;
  552. }
  553. /***
  554. *
  555. ** function : Q_Find
  556. *
  557. ** purpose : Linear search of queue for match with key in *data
  558. *
  559. ** parameters : queue pointer q, data pointer with data containing key
  560. * comparison function here called Comp.
  561. *
  562. ** returns : True_ if found , False_ if not in queue.
  563. *
  564. ** comments : Useful for small queues that are constantly changing
  565. * and would otherwise need constant sorting with the
  566. * Q_Seek function.
  567. * For description of Comp see Q_Sort.
  568. * Queue cursor left on position found item else at end.
  569. *
  570. ***/
  571. int Q_Find(queue *q, void *data,
  572. int (*Comp)(const void *, const void *))
  573. {
  574. void *d;
  575. if (q == NULL) {
  576. return False_;
  577. }
  578. d = Q_Head(q);
  579. do {
  580. if(Comp(d, data) == 0)
  581. return True_;
  582. d = Q_Next(q);
  583. } while(!Q_AtTail(q));
  584. if(Comp(d, data) == 0)
  585. return True_;
  586. return False_;
  587. }
  588. /*======== Sorted Queue and Index functions ========= */
  589. static void QuickSort(void *list[], int low, int high,
  590. int (*Comp)(const void *, const void *))
  591. {
  592. int flag = 1, i, j;
  593. void *key, *temp;
  594. if(low < high) {
  595. i = low;
  596. j = high + 1;
  597. key = list[ low ];
  598. while(flag) {
  599. i++;
  600. while(Comp(list[i], key) < 0)
  601. i++;
  602. j--;
  603. while(Comp(list[j], key) > 0)
  604. j--;
  605. if(i < j) {
  606. temp = list[i];
  607. list[i] = list[j];
  608. list[j] = temp;
  609. }
  610. else flag = 0;
  611. }
  612. temp = list[low];
  613. list[low] = list[j];
  614. list[j] = temp;
  615. QuickSort(list, low, j-1, Comp);
  616. QuickSort(list, j+1, high, Comp);
  617. }
  618. }
  619. /***
  620. *
  621. ** function : Q_Sort
  622. *
  623. ** purpose : sort the queue and allow index style access.
  624. *
  625. ** parameters : queue pointer, comparison function compatible with
  626. * with 'qsort'.
  627. *
  628. ** returns : True_ if sort succeeded. False_ if error occurred.
  629. *
  630. ** comments : Comp function supplied by caller must return
  631. * -1 if data1 < data2
  632. * 0 if data1 == data2
  633. * +1 if data1 > data2
  634. *
  635. * for Comp(data1, data2)
  636. *
  637. * If queue is already sorted it frees the memory of the
  638. * old index and starts again.
  639. *
  640. ***/
  641. int Q_Sort(queue *q, int (*Comp)(const void *, const void *))
  642. {
  643. int i;
  644. void *d;
  645. datanode *dn;
  646. /* if already sorted free memory for tag array */
  647. if(q->sorted) {
  648. free(index);
  649. free(posn_index);
  650. q->sorted = False_;
  651. }
  652. /* Now allocate memory of array, array of pointers */
  653. index = malloc(q->size * sizeof(q->cursor->data));
  654. if(index == NULL)
  655. return False_;
  656. posn_index = malloc(q->size * sizeof(q->cursor));
  657. if(posn_index == NULL) {
  658. free(index);
  659. return False_;
  660. }
  661. /* Walk queue putting pointers into array */
  662. d = Q_Head(q);
  663. for(i=0; i < q->size; i++) {
  664. index[i] = d;
  665. posn_index[i] = q->cursor;
  666. d = Q_Next(q);
  667. }
  668. /* Now sort the index */
  669. QuickSort(index, 0, q->size - 1, Comp);
  670. /* Rearrange the actual queue into correct order */
  671. dn = q->head;
  672. i = 0;
  673. while(dn != NULL) {
  674. dn->data = index[i++];
  675. dn = dn->next;
  676. }
  677. /* Re-position to original element */
  678. if(d != NULL)
  679. Q_Find(q, d, Comp);
  680. else Q_Head(q);
  681. q->sorted = True_;
  682. return True_;
  683. }
  684. /***
  685. *
  686. ** function : Q_BSearch
  687. *
  688. ** purpose : binary search of queue index for node containing key
  689. *
  690. ** parameters : queue pointer 'q', data pointer of key 'key',
  691. * Comp comparison function.
  692. *
  693. ** returns : integer index into array of node pointers,
  694. * or -1 if not found.
  695. *
  696. ** comments : see Q_Sort for description of 'Comp' function.
  697. *
  698. ***/
  699. static int Q_BSearch( queue *q, void *key,
  700. int (*Comp)(const void *, const void*))
  701. {
  702. int low, mid, hi, val;
  703. low = 0;
  704. hi = q->size - 1;
  705. while(low <= hi) {
  706. mid = (low + hi) / 2;
  707. val = Comp(key, index[ mid ]);
  708. if(val < 0)
  709. hi = mid - 1;
  710. else if(val > 0)
  711. low = mid + 1;
  712. else /* Success */
  713. return mid;
  714. }
  715. /* Not Found */
  716. return -1;
  717. }
  718. /***
  719. *
  720. ** function : Q_Seek
  721. *
  722. ** purpose : use index to locate data according to key in 'data'
  723. *
  724. ** parameters : queue pointer 'q', data pointer 'data', Comp comparison
  725. * function.
  726. *
  727. ** returns : pointer to data or NULL if could not find it or could
  728. * not sort queue.
  729. *
  730. ** comments : see Q_Sort for description of 'Comp' function.
  731. *
  732. ***/
  733. void *Q_Seek(queue *q, void *data, int (*Comp)(const void *, const void *))
  734. {
  735. int idx;
  736. if (q == NULL) {
  737. return NULL;
  738. }
  739. if(!q->sorted) {
  740. if(!Q_Sort(q, Comp))
  741. return NULL;
  742. }
  743. idx = Q_BSearch(q, data, Comp);
  744. if(idx < 0)
  745. return NULL;
  746. q->cursor = posn_index[idx];
  747. return index[idx];
  748. }
  749. /***
  750. *
  751. ** function : Q_Insert
  752. *
  753. ** purpose : Insert an element into an indexed queue
  754. *
  755. ** parameters : queue pointer 'q', data pointer 'data', Comp comparison
  756. * function.
  757. *
  758. ** returns : pointer to data or NULL if could not find it or could
  759. * not sort queue.
  760. *
  761. ** comments : see Q_Sort for description of 'Comp' function.
  762. * WARNING! This code can be very slow since each new
  763. * element means a new Q_Sort. Should only be used for
  764. * the insertion of the odd element ,not the piecemeal
  765. * building of an entire queue.
  766. ***/
  767. int Q_Insert(queue *q, void *data, int (*Comp)(const void *, const void *))
  768. {
  769. if (q == NULL) {
  770. return False_;
  771. }
  772. Q_PushHead(q, data);
  773. if(!Q_Sort(q, Comp))
  774. return False_;
  775. return True_;
  776. }
  777. /* read only funcs for iterating through queue. above funcs modify queue */
  778. q_iter Q_Iter_Head(queue *q) {
  779. return q ? (q_iter)q->head : NULL;
  780. }
  781. q_iter Q_Iter_Tail(queue *q) {
  782. return q ? (q_iter)q->tail : NULL;
  783. }
  784. q_iter Q_Iter_Next(q_iter qi) {
  785. return qi ? (q_iter)((node*)qi)->next : NULL;
  786. }
  787. q_iter Q_Iter_Prev(q_iter qi) {
  788. return qi ? (q_iter)((node*)qi)->prev : NULL;
  789. }
  790. void * Q_Iter_Get(q_iter qi) {
  791. return qi ? ((node*)qi)->data : NULL;
  792. }
  793. int Q_Iter_Put(q_iter qi, void* data) {
  794. if(qi) {
  795. ((node*)qi)->data = data;
  796. return True_;
  797. }
  798. return False_;
  799. }