LCOV - code coverage report
Current view: top level - util/tmpl - fd_deque_dynamic.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 291 292 99.7 %
Date: 2025-01-08 12:08:44 Functions: 261 56745 0.5 %

          Line data    Source code
       1             : /* Declares a family of functions implementing a single-threaded
       2             :    run-time fixed-capacity double-ended queue (deque) designed for high
       3             :    performance contexts.  Example usage:
       4             : 
       5             :      #define DEQUE_NAME my_deque
       6             :      #define DEQUE_T    my_ele_t
       7             :      #include "util/tmpl/fd_deque.c"
       8             : 
       9             :    This creates the following API for use in the local compilation unit:
      10             : 
      11             :      // Constructors
      12             : 
      13             :      // IMPORTANT SAFETY TIP!  max==0 is fine for this API.  Calling
      14             :      // most APIs on such a deque will be a violation of the API
      15             :      // pre-requisites (e.g. push_head and pop_tail are not valid
      16             :      // because a max==0 deque is simultaneously full and empty).  The
      17             :      // accessors (max==0, cnt==0, avail==0, empty==true, full==true),
      18             :      // remove_all (no-op) and iterators (will indicate nothing to
      19             :      // iterator over) are still valid to use.
      20             : 
      21             :      ulong      my_deque_align    ( void               ); // required byte alignment of a deque
      22             :      ulong      my_deque_footprint( ulong      max     ); // required byte footprint of a deque with max capacity
      23             :      void     * my_deque_new      ( void     * shmem,     // format memory region into a my_deque, my_deque will be empty
      24             :                                     ulong      max     ); // (caller not joined on return, mem has required align/footprint, etc)
      25             :      my_ele_t * my_deque_join     ( void     * shdeque ); // join a my_deque (unlimited joins, etc) (NOT A CAST OF SHDEQUE)
      26             :                                                           // join can be indexed like a normal array with max elements
      27             :      void     * my_deque_leave    ( my_ele_t * deque   ); // leave a my_deque (matched with join, etc) (NOT A CAST OF DEQUE)
      28             :      void     * my_deque_delete   ( void     * shdeque ); // unformat memory (no active joins, etc)
      29             : 
      30             :      // Accessors
      31             : 
      32             :      ulong my_deque_max  ( my_ele_t const * deque ); // returns the max elements that could be in the deque
      33             :      ulong my_deque_cnt  ( my_ele_t const * deque ); // returns the number of elements in the deque, in [0,max]
      34             :      ulong my_deque_avail( my_ele_t const * deque ); // returns max-cnt
      35             :      int   my_deque_empty( my_ele_t const * deque ); // returns 1 if deque is empty and 0 otherwise
      36             :      int   my_deque_full ( my_ele_t const * deque ); // returns 1 if deque is full and 0 otherwise
      37             : 
      38             :      // Simple API
      39             : 
      40             :      // IMPORTANT SAFETY TIP!  The wrap APIs discard the value popped on
      41             :      // wrapping.  Thus, these should not be used when a my_ele_t might
      42             :      // contain the only reference to a resource (memory, file
      43             :      // descriptors, etc) that might need to be released after popping
      44             :      // (e.g. use these only if my_ele_t is plain-old-data).
      45             : 
      46             :      // IMPORTANT SAFETY TIP!  The pop_idx APIs have an O(cnt) worst
      47             :      // case.
      48             : 
      49             :      my_ele_t * my_deque_push_head     ( my_ele_t * deque, my_ele_t ele ); // push ele at the deque head, returns deque
      50             :      my_ele_t * my_deque_push_tail     ( my_ele_t * deque, my_ele_t ele ); // push ele at the deque tail, returns deque
      51             :      my_ele_t   my_deque_pop_head      ( my_ele_t * deque               ); // pops ele from the head of the deque, returns ele
      52             :      my_ele_t   my_deque_pop_tail      ( my_ele_t * deque               ); // pops ele from the tail of the deque, returns ele
      53             :      my_ele_t * my_deque_push_head_wrap( my_ele_t * deque, my_ele_t ele ); // push ele at the deque head (pops tail first if deque full), returns deque, assumes deque has a positive max
      54             :      my_ele_t * my_deque_push_tail_wrap( my_ele_t * deque, my_ele_t ele ); // push ele at the deque tail (pops head first if deque full), returns deque, assumes deque has a positive max
      55             :      my_ele_t   my_deque_pop_idx_tail  ( my_ele_t * deque, ulong    idx ); // pops ele at idx, returns ele. idx in [0,cnt). shifts head <= tail
      56             : 
      57             :      // Advanced API for zero-copy usage
      58             : 
      59             :      my_ele_t * my_deque_peek_head  ( my_ele_t * deque ); // peeks at head, returned ptr lifetime is until next op on deque
      60             :      my_ele_t * my_deque_peek_tail  ( my_ele_t * deque ); // peeks at tail, returned ptr lifetime is until next op on deque
      61             :      my_ele_t * my_deque_peek_index ( my_ele_t * deque, ulong idx ); // peeks at index, returned ptr lifetime is until next op on deque
      62             :      my_ele_t * my_deque_insert_head( my_ele_t * deque ); // inserts uninitialized element at head, returns deque
      63             :      my_ele_t * my_deque_insert_tail( my_ele_t * deque ); // inserts uninitialized element at tail, returns deque
      64             :      my_ele_t * my_deque_remove_head( my_ele_t * deque ); // removes head, returns deque
      65             :      my_ele_t * my_deque_remove_tail( my_ele_t * deque ); // removes tail, returns deque
      66             :      my_ele_t * my_deque_remove_all ( my_ele_t * deque ); // removes all, returns deque, fast O(1)
      67             : 
      68             :      my_ele_t * my_deque_push_head_nocopy( my_ele_t * deque ); // push the head, returns the new uninitialized element
      69             :      my_ele_t * my_deque_push_tail_nocopy( my_ele_t * deque ); // push the tail, returns the new uninitialized element
      70             :      my_ele_t * my_deque_pop_head_nocopy ( my_ele_t * deque ); // pops the head, returns the deleted element
      71             :      my_ele_t * my_deque_pop_tail_nocopy ( my_ele_t * deque ); // pops the tail, returns the deleted element
      72             : 
      73             :      my_ele_t const * my_deque_peek_head_const( my_ele_t const * deque ); // const version of peek_head
      74             :      my_ele_t const * my_deque_peek_tail_const( my_ele_t const * deque ); // const version of peek_tail
      75             :      my_ele_t const * my_deque_peek_index_const( my_ele_t const * deque, ulong idx ); // const version of peek_index
      76             : 
      77             :      // my_deque_iter_* allow for iteration over all the elements in
      78             :      // a my_deque.  The iteration will be in order from head to tail.
      79             :      // Example usage:
      80             :      //
      81             :      //   for( my_deque_iter_t iter = my_deque_iter_init( deque ); !my_deque_iter_done( deque, iter ); iter = my_deque_iter_next( deque, iter ) ) {
      82             :      //     my_deque_t * ele = my_deque_iter_ele( deque, iter );
      83             :      //
      84             :      //     ... process ele here
      85             :      //   }
      86             : 
      87             :      my_deque_iter_t  my_deque_iter_init     ( my_deque_t const * deque );
      88             :      int              my_deque_iter_done     ( my_deque_t const * deque, my_deque_iter_t iter ); // returns 1 if no more iterations, 0 o.w.
      89             :      my_deque_iter_t  my_deque_iter_next     ( my_deque_t const * deque, my_deque_iter_t iter ); // returns next iter value iter
      90             :      my_ele_t *       my_deque_iter_ele      ( my_deque_t *       deque, my_deque_iter_t iter ); // assumes not done, return non-NULL ele
      91             :      my_ele_t const * my_deque_iter_ele_const( my_deque_t const * deque, my_deque_iter_t iter ); // assumes not done, return non-NULL ele
      92             : 
      93             :      // my_deque_iter_rev_* allow for iteration similar to the above, but
      94             :      // in reverse order from tail to head.
      95             :      // Example usage:
      96             :      //
      97             :      //   for( my_deque_iter_t iter = my_deque_iter_init_rev( deque ); !my_deque_iter_done_rev( deque, iter ); iter = my_deque_iter_prev( deque, iter ) ) {
      98             :      //     my_deque_t * ele = my_deque_iter_ele( deque, iter );
      99             :      //
     100             :      //     ... process ele here
     101             :      //   }
     102             : 
     103             :      my_deque_iter_t  my_deque_iter_init_rev ( my_deque_t const * deque );
     104             :      int              my_deque_iter_done_rev ( my_deque_t const * deque, my_deque_iter_t iter ); // returns 1 if no more iterations, 0 o.w.
     105             :      my_deque_iter_t  my_deque_iter_prev     ( my_deque_t const * deque, my_deque_iter_t iter ); // returns prev iter value iter
     106             : 
     107             :    For performance, none of the functions do any error checking.
     108             :    Specifically, the caller promises that max is such that footprint
     109             :    will not overflow 2^64 (e.g. max << (2^64)/sizeof(my_ele_t)), cnt<max
     110             :    for any push or insert operation and cnt>0 for any pop, peek or
     111             :    remove operation (remove_all is fine on an empty deque). */
     112             : 
     113             : #include "../bits/fd_bits.h"
     114             : #include <stddef.h>
     115             : 
     116             : #ifndef DEQUE_NAME
     117             : #error "Define DEQUE_NAME"
     118             : #endif
     119             : 
     120             : #ifndef DEQUE_T
     121             : #error "Define DEQUE_T"
     122             : #endif
     123             : 
     124             : /* Implementation *****************************************************/
     125             : 
     126  6112661337 : #define DEQUE_(x) FD_EXPAND_THEN_CONCAT3(DEQUE_NAME,_,x)
     127             : 
     128             : struct DEQUE_(private) {
     129             :   ulong   max1;  /* Max elements in deque minus 1 */
     130             :   ulong   cnt;   /* Num elements in deque, in [0,max] */
     131             :   ulong   start; /* Location of current head, in [0,max) */
     132             :   ulong   end;   /* Location of current tail, in [0,max) */
     133             :   DEQUE_T deque[ 1 ]; /* Actually max in size */
     134             : };
     135             : 
     136             : typedef struct DEQUE_(private) DEQUE_(private_t);
     137             : 
     138             : FD_PROTOTYPES_BEGIN
     139             : 
     140             : /* private_from_deque returns a pointer to the deque_private given a
     141             :    pointer to the deque. */
     142             : 
     143             : FD_FN_CONST static inline DEQUE_(private_t) *
     144    85376691 : DEQUE_(private_hdr_from_deque)( DEQUE_T * deque ) {
     145    85376691 :   return (DEQUE_(private_t) *)( (ulong)deque - offsetof(DEQUE_(private_t), deque) );
     146    85376691 : }
     147             : 
     148             : /* const-correct version of above */
     149             : 
     150             : FD_FN_CONST static inline DEQUE_(private_t) const *
     151   234504147 : DEQUE_(private_const_hdr_from_deque)( DEQUE_T const * deque ) {
     152   234504147 :   return (DEQUE_(private_t) const *)( (ulong)deque - offsetof(DEQUE_(private_t), deque) );
     153   234504147 : }
     154             : 
     155             : /* These move i to the previous or next slot to i for given max.
     156             :    Input should be in [0,max) and output will be in [0,max). */
     157             : 
     158   280767468 : FD_FN_CONST static inline ulong DEQUE_(private_prev)( ulong i, ulong max1 ) { return fd_ulong_if( i==0UL,  max1, i-1UL ); }
     159   407669907 : FD_FN_CONST static inline ulong DEQUE_(private_next)( ulong i, ulong max1 ) { return fd_ulong_if( i>=max1, 0UL,  i+1UL ); }
     160             : 
     161           0 : FD_FN_CONST static inline ulong DEQUE_(align)( void ) { return alignof(DEQUE_(private_t)); }
     162             : 
     163             : FD_FN_CONST static inline ulong
     164      474405 : DEQUE_(footprint)( ulong max ) {
     165      474405 :   return fd_ulong_align_up( fd_ulong_align_up( 32UL, alignof(DEQUE_T) ) + sizeof(DEQUE_T)*max, alignof(DEQUE_(private_t)) );
     166      474405 : }
     167             : 
     168             : static inline void *
     169             : DEQUE_(new)( void * shmem,
     170      474255 :              ulong  max ) {
     171      474255 :   DEQUE_(private_t) * hdr = (DEQUE_(private_t) *)shmem;
     172      474255 :   hdr->max1  = max-1UL; /* Note: will wrap to ULONG_MAX if max==0 (and ULONG_MAX+1 will wrap back to 0) */
     173      474255 :   hdr->cnt   = 0UL;
     174      474255 :   hdr->start = 0UL;
     175      474255 :   hdr->end   = 0UL;
     176      474255 :   return hdr;
     177      474255 : }
     178             : 
     179             : static inline DEQUE_T *
     180      474327 : DEQUE_(join)( void * shdeque ) {
     181      474327 :   DEQUE_(private_t) * hdr = (DEQUE_(private_t) *)shdeque;
     182      474327 :   return hdr->deque;
     183      474327 : }
     184             : 
     185      453870 : static inline void * DEQUE_(leave) ( DEQUE_T * deque   ) { return (void *)DEQUE_(private_hdr_from_deque)( deque ); }
     186      453798 : static inline void * DEQUE_(delete)( void *    shdeque ) { return shdeque; }
     187             : 
     188             : FD_FN_PURE static inline ulong
     189   300000006 : DEQUE_(max)( DEQUE_T const * deque ) {
     190   300000006 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     191   300000006 :   return hdr->max1 + 1UL;
     192   300000006 : }
     193             : 
     194             : FD_FN_PURE static inline ulong
     195   301985526 : DEQUE_(cnt)( DEQUE_T const * deque ) {
     196   301985526 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     197   301985526 :   return hdr->cnt;
     198   301985526 : }
     199             : 
     200             : FD_FN_PURE static inline ulong
     201   300000003 : DEQUE_(avail)( DEQUE_T const * deque ) {
     202   300000003 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     203   300000003 :   return hdr->max1 + 1UL - hdr->cnt;
     204   300000003 : }
     205             : 
     206             : FD_FN_PURE static inline int
     207   300065130 : DEQUE_(empty)( DEQUE_T const * deque ) {
     208   300065130 :   return !DEQUE_(private_const_hdr_from_deque)( deque )->cnt;
     209   300065130 : }
     210             : 
     211             : FD_FN_PURE static inline int
     212   300872664 : DEQUE_(full)( DEQUE_T const * deque ) {
     213   300872664 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     214   300872664 :   return (hdr->max1 + 1UL)==hdr->cnt;
     215   300872664 : }
     216             : 
     217             : static inline DEQUE_T *
     218             : DEQUE_(push_head)( DEQUE_T * deque,
     219    14490972 :                    DEQUE_T   ele ) {
     220    14490972 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     221    14490972 :   ulong max1  = hdr->max1;
     222    14490972 :   ulong cnt   = hdr->cnt;
     223    14490972 :   ulong start = hdr->start;
     224    14490972 :   start = DEQUE_(private_prev)( start, max1 );
     225    14490972 :   hdr->deque[ start ] = ele;
     226    14490972 :   hdr->cnt   = cnt+1UL;
     227    14490972 :   hdr->start = start;
     228    14490972 :   return deque;
     229    14490972 : }
     230             : 
     231             : static inline DEQUE_T *
     232             : DEQUE_(push_tail)( DEQUE_T * deque,
     233    14517741 :                    DEQUE_T   ele ) {
     234    14517741 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     235    14517741 :   ulong max1 = hdr->max1;
     236    14517741 :   ulong cnt  = hdr->cnt;
     237    14517741 :   ulong end  = hdr->end;
     238    14517741 :   hdr->deque[ end ] = ele;
     239    14517741 :   end = DEQUE_(private_next)( end, max1 );
     240    14517741 :   hdr->cnt = cnt+1UL;
     241    14517741 :   hdr->end = end;
     242    14517741 :   return deque;
     243    14517741 : }
     244             : 
     245             : static inline DEQUE_T
     246    16553712 : DEQUE_(pop_head)( DEQUE_T * deque ) {
     247    16553712 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     248    16553712 :   ulong max1  = hdr->max1;
     249    16553712 :   ulong cnt   = hdr->cnt;
     250    16553712 :   ulong start = hdr->start;
     251    16553712 :   DEQUE_T ele = hdr->deque[ start ];
     252    16553712 :   start = DEQUE_(private_next)( start, max1 );
     253    16553712 :   hdr->cnt   = cnt-1UL;
     254    16553712 :   hdr->start = start;
     255    16553712 :   return ele;
     256    16553712 : }
     257             : 
     258             : static inline DEQUE_T
     259    16565442 : DEQUE_(pop_tail)( DEQUE_T * deque ) {
     260    16565442 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     261    16565442 :   ulong max1 = hdr->max1;
     262    16565442 :   ulong cnt  = hdr->cnt;
     263    16565442 :   ulong end  = hdr->end;
     264    16565442 :   end = DEQUE_(private_prev)( end, max1 );
     265    16565442 :   DEQUE_T ele = hdr->deque[ end ];
     266    16565442 :   hdr->cnt = cnt-1UL;
     267    16565442 :   hdr->end = end;
     268    16565442 :   return ele;
     269    16565442 : }
     270             : 
     271             : static inline DEQUE_T *
     272             : DEQUE_(push_head_wrap)( DEQUE_T * deque,
     273    17635728 :                         DEQUE_T   ele ) {
     274    17635728 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     275    17635728 :   ulong max1  = hdr->max1;
     276    17635728 :   ulong cnt   = hdr->cnt;
     277    17635728 :   ulong start = hdr->start;
     278    17635728 :   ulong end   = hdr->end;
     279             : 
     280             : # if 0 /* handholding check */
     281             :   if( FD_UNLIKELY( max1==ULONG_MAX ) ) FD_LOG_CRIT(( "cannot push_head_wrap when max is zero" ));
     282             : # endif
     283             : 
     284             :   /* If the deque is full, pop and discard the tail. */
     285             : 
     286    17635728 :   int is_full = (cnt>max1);
     287    17635728 :   end  = fd_ulong_if( !is_full, end, DEQUE_(private_prev)( end, max1 ) );
     288    17635728 :   cnt -= (ulong)is_full;
     289             : 
     290             :   /* Push the head */
     291             : 
     292    17635728 :   start = DEQUE_(private_prev)( start, max1 );
     293    17635728 :   hdr->deque[ start ] = ele;
     294             : 
     295    17635728 :   hdr->cnt   = cnt + 1UL;
     296    17635728 :   hdr->start = start;
     297    17635728 :   hdr->end   = end;
     298    17635728 :   return deque;
     299    17635728 : }
     300             : 
     301             : static inline DEQUE_T *
     302             : DEQUE_(push_tail_wrap)( DEQUE_T * deque,
     303    17650173 :                         DEQUE_T   ele ) {
     304    17650173 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     305    17650173 :   ulong max1  = hdr->max1;
     306    17650173 :   ulong cnt   = hdr->cnt;
     307    17650173 :   ulong start = hdr->start;
     308    17650173 :   ulong end   = hdr->end;
     309             : 
     310             : # if 0 /* handholding check */
     311             :   if( FD_UNLIKELY( max1==ULONG_MAX ) ) FD_LOG_CRIT(( "cannot push_tail_wrap when max is zero" ));
     312             : # endif
     313             : 
     314             :   /* If the deque is full, pop and discard the head. */
     315             : 
     316    17650173 :   int is_full = (cnt>max1);
     317    17650173 :   start  = fd_ulong_if( !is_full, start, DEQUE_(private_next)( start, max1 ) );
     318    17650173 :   cnt   -= (ulong)is_full;
     319             : 
     320             :   /* Push the head */
     321             : 
     322    17650173 :   hdr->deque[ end ] = ele;
     323    17650173 :   end = DEQUE_(private_next)( end, max1 );
     324             : 
     325    17650173 :   hdr->cnt   = cnt + 1UL;
     326    17650173 :   hdr->start = start;
     327    17650173 :   hdr->end   = end;
     328    17650173 :   return deque;
     329    17650173 : }
     330             : 
     331             : static inline DEQUE_T
     332    16566861 : DEQUE_(pop_idx_tail)( DEQUE_T * deque, ulong idx ) {
     333    16566861 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     334             : 
     335    16566861 :   ulong max1 = hdr->max1;
     336             : 
     337    16566861 :   ulong slot = hdr->start + idx;
     338    16566861 :         slot = fd_ulong_if( slot <= max1, slot, slot - max1 - 1UL );
     339    16566861 :   ulong cnt  = --hdr->cnt;
     340    16566861 :   ulong rem  = cnt - idx;
     341             : 
     342    16566861 :   hdr->end   = DEQUE_(private_prev)( hdr->end, max1 );
     343             : 
     344    16566861 :   DEQUE_T * cur = &deque[ slot ];
     345    16566861 :   DEQUE_T   ele = *cur;
     346    51242253 :   while( rem-- ) {
     347    34675392 :     slot = DEQUE_(private_next)( slot, max1 );
     348    34675392 :     DEQUE_T * next = &deque[ slot ];
     349    34675392 :     *cur = *next;
     350    34675392 :      cur =  next;
     351    34675392 :   }
     352    16566861 :   return ele;
     353    16566861 : }
     354             : 
     355             : FD_FN_PURE static inline DEQUE_T *
     356    14489394 : DEQUE_(peek_head)( DEQUE_T * deque ) {
     357    14489394 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     358    14489394 :   return hdr->deque + hdr->start;
     359    14489394 : }
     360             : 
     361             : FD_FN_PURE static inline DEQUE_T *
     362    14532342 : DEQUE_(peek_tail)( DEQUE_T * deque ) {
     363    14532342 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     364    14532342 :   return hdr->deque + DEQUE_(private_prev)( hdr->end, hdr->max1 );
     365    14532342 : }
     366             : 
     367             : FD_FN_PURE static inline DEQUE_T *
     368   171963318 : DEQUE_(peek_index)( DEQUE_T * deque, ulong idx ) {
     369   171963318 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     370   171963318 :   ulong slot = hdr->start + idx;
     371   171963318 :         slot = fd_ulong_if( slot <= hdr->max1, slot, slot - hdr->max1 - 1UL );
     372   171963318 :   return hdr->deque + slot;
     373   171963318 : }
     374             : 
     375             : FD_FN_PURE static inline DEQUE_T const *
     376    16560099 : DEQUE_(peek_head_const)( DEQUE_T const * deque ) {
     377    16560099 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     378    16560099 :   return hdr->deque + hdr->start;
     379    16560099 : }
     380             : 
     381             : FD_FN_PURE static inline DEQUE_T const *
     382    16574013 : DEQUE_(peek_tail_const)( DEQUE_T const * deque ) {
     383    16574013 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     384    16574013 :   return hdr->deque + DEQUE_(private_prev)( hdr->end, hdr->max1 );
     385    16574013 : }
     386             : 
     387             : FD_FN_PURE static inline DEQUE_T const *
     388   172057671 : DEQUE_(peek_index_const)( DEQUE_T const * deque, ulong idx ) {
     389   172057671 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     390   172057671 :   ulong slot = hdr->start + idx;
     391   172057671 :         slot = fd_ulong_if( slot <= hdr->max1, slot, slot - hdr->max1 - 1UL );
     392   172057671 :   return hdr->deque + slot;
     393   172057671 : }
     394             : 
     395             : static inline DEQUE_T *
     396    14489388 : DEQUE_(insert_head)( DEQUE_T * deque ) {
     397    14489388 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     398    14489388 :   ulong max1  = hdr->max1;
     399    14489388 :   ulong cnt   = hdr->cnt;
     400    14489388 :   ulong start = hdr->start;
     401    14489388 :   hdr->cnt    = cnt + 1UL;
     402    14489388 :   hdr->start  = DEQUE_(private_prev)( start, max1 );
     403    14489388 :   return deque;
     404    14489388 : }
     405             : 
     406             : static inline DEQUE_T *
     407    14492730 : DEQUE_(insert_tail)( DEQUE_T * deque ) {
     408    14492730 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     409    14492730 :   ulong max1 = hdr->max1;
     410    14492730 :   ulong cnt  = hdr->cnt;
     411    14492730 :   ulong end  = hdr->end;
     412    14492730 :   hdr->cnt   = cnt + 1UL;
     413    14492730 :   hdr->end   = DEQUE_(private_next)( end, max1 );
     414    14492730 :   return deque;
     415    14492730 : }
     416             : 
     417             : static inline DEQUE_T *
     418    16557435 : DEQUE_(remove_head)( DEQUE_T * deque ) {
     419    16557435 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     420    16557435 :   ulong max1  = hdr->max1;
     421    16557435 :   ulong cnt   = hdr->cnt;
     422    16557435 :   ulong start = hdr->start;
     423    16557435 :   hdr->cnt    = cnt - 1UL;
     424    16557435 :   hdr->start  = DEQUE_(private_next)( start, max1 );
     425    16557435 :   return deque;
     426    16557435 : }
     427             : 
     428             : static inline DEQUE_T *
     429    16567389 : DEQUE_(remove_tail)( DEQUE_T * deque ) {
     430    16567389 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     431    16567389 :   ulong max1 = hdr->max1;
     432    16567389 :   ulong cnt  = hdr->cnt;
     433    16567389 :   ulong end  = hdr->end;
     434    16567389 :   hdr->cnt   = cnt - 1UL;
     435    16567389 :   hdr->end   = DEQUE_(private_prev)( end, max1 );
     436    16567389 :   return deque;
     437    16567389 : }
     438             : 
     439             : static inline DEQUE_T *
     440    15357660 : DEQUE_(push_head_nocopy)( DEQUE_T * deque ) {
     441    15357660 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     442    15357660 :   ulong max1  = hdr->max1;
     443    15357660 :   ulong cnt   = hdr->cnt;
     444    15357660 :   ulong start = hdr->start;
     445    15357660 :   hdr->cnt    = cnt + 1UL;
     446    15357660 :   hdr->start  = DEQUE_(private_prev)( start, max1 );
     447    15357660 :   return hdr->deque + hdr->start;
     448    15357660 : }
     449             : 
     450             : static inline DEQUE_T *
     451    18644619 : DEQUE_(push_tail_nocopy)( DEQUE_T * deque ) {
     452    18644619 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     453    18644619 :   ulong max1 = hdr->max1;
     454    18644619 :   ulong cnt  = hdr->cnt;
     455    18644619 :   ulong end  = hdr->end;
     456    18644619 :   hdr->cnt   = cnt + 1UL;
     457    18644619 :   DEQUE_T * res = hdr->deque + hdr->end;
     458    18644619 :   hdr->end   = DEQUE_(private_next)( end, max1 );
     459    18644619 :   return res;
     460    18644619 : }
     461             : 
     462             : static inline DEQUE_T *
     463    16557555 : DEQUE_(pop_head_nocopy)( DEQUE_T * deque ) {
     464    16557555 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     465    16557555 :   ulong max1  = hdr->max1;
     466    16557555 :   ulong cnt   = hdr->cnt;
     467    16557555 :   ulong start = hdr->start;
     468    16557555 :   hdr->cnt    = cnt - 1UL;
     469    16557555 :   DEQUE_T * res = hdr->deque + hdr->start;
     470    16557555 :   hdr->start  = DEQUE_(private_next)( start, max1 );
     471    16557555 :   return res;
     472    16557555 : }
     473             : 
     474             : static inline DEQUE_T *
     475    16773192 : DEQUE_(pop_tail_nocopy)( DEQUE_T * deque ) {
     476    16773192 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     477    16773192 :   ulong max1 = hdr->max1;
     478    16773192 :   ulong cnt  = hdr->cnt;
     479    16773192 :   ulong end  = hdr->end;
     480    16773192 :   hdr->cnt   = cnt - 1UL;
     481    16773192 :   hdr->end   = DEQUE_(private_prev)( end, max1 );
     482    16773192 :   return hdr->deque + hdr->end;
     483    16773192 : }
     484             : 
     485             : static inline DEQUE_T *
     486        4527 : DEQUE_(remove_all)( DEQUE_T * deque ) {
     487        4527 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     488        4527 :   hdr->cnt   = 0UL;
     489        4527 :   hdr->start = 0UL;
     490        4527 :   hdr->end   = 0UL;
     491        4527 :   return deque;
     492        4527 : }
     493             : 
     494             : typedef struct { ulong rem; ulong idx; } DEQUE_(iter_t);
     495             : 
     496             : static inline DEQUE_(iter_t)
     497    19896750 : DEQUE_(iter_init)( DEQUE_T const * deque ) {
     498    19896750 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     499    19896750 :   DEQUE_(iter_t) iter;
     500    19896750 :   iter.rem = hdr->cnt;
     501    19896750 :   iter.idx = hdr->start;
     502    19896750 :   return iter;
     503    19896750 : }
     504             : 
     505             : static inline DEQUE_(iter_t)
     506    17651076 : DEQUE_(iter_init_rev)( DEQUE_T const * deque ) {
     507    17651076 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     508    17651076 :   DEQUE_(iter_t) iter;
     509    17651076 :   iter.rem = hdr->cnt;
     510    17651076 :   iter.idx = hdr->end;
     511    17651076 :   iter.idx = DEQUE_(private_prev)( iter.idx, hdr->max1 );
     512    17651076 :   return iter;
     513    17651076 : }
     514             : 
     515             : static inline int
     516   260267127 : DEQUE_(iter_done)( DEQUE_T const * deque, DEQUE_(iter_t) iter ) {
     517   260267127 :   (void)deque;
     518   260267127 :   return !iter.rem;
     519   260267127 : }
     520             : 
     521             : static inline int
     522   103578753 : DEQUE_(iter_done_rev)( DEQUE_T const * deque, DEQUE_(iter_t) iter ) {
     523   103578753 :   (void)deque;
     524   103578753 :   return !iter.rem;
     525   103578753 : }
     526             : 
     527             : static inline DEQUE_(iter_t)
     528   240370377 : DEQUE_(iter_next)( DEQUE_T const * deque, DEQUE_(iter_t) iter ) {
     529   240370377 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     530   240370377 :   iter.rem--;
     531   240370377 :   iter.idx = DEQUE_(private_next)( iter.idx, hdr->max1 );
     532   240370377 :   return iter;
     533   240370377 : }
     534             : 
     535             : static inline DEQUE_(iter_t)
     536    85927677 : DEQUE_(iter_prev)( DEQUE_T const * deque, DEQUE_(iter_t) iter ) {
     537    85927677 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     538    85927677 :   iter.rem--;
     539    85927677 :   iter.idx = DEQUE_(private_prev)( iter.idx, hdr->max1 );
     540    85927677 :   return iter;
     541    85927677 : }
     542             : 
     543             : static inline DEQUE_T *
     544   251494116 : DEQUE_(iter_ele)( DEQUE_T * deque, DEQUE_(iter_t) iter ) {
     545   251494116 :   DEQUE_(private_t) * hdr = DEQUE_(private_hdr_from_deque)( deque );
     546   251494116 :   return hdr->deque + iter.idx;
     547   251494116 : }
     548             : 
     549             : static inline DEQUE_T const *
     550    74804121 : DEQUE_(iter_ele_const)( DEQUE_T const * deque, DEQUE_(iter_t) iter ) {
     551    74804121 :   DEQUE_(private_t) const * hdr = DEQUE_(private_const_hdr_from_deque)( deque );
     552    74804121 :   return hdr->deque + iter.idx;
     553    74804121 : }
     554             : 
     555             : FD_PROTOTYPES_END
     556             : 
     557             : #undef DEQUE_
     558             : 
     559             : #undef DEQUE_T
     560             : #undef DEQUE_NAME
     561             : 

Generated by: LCOV version 1.14