58   u32 normalized_start = (seg->
start + f->nitems - f->tail) % f->nitems;
    59   s = 
format (s, 
"[%u, %u], len %u, next %d, prev %d", normalized_start,
    60               (normalized_start + seg->
length) % f->nitems, seg->
length,
   110   memset (f->data, 0xFF, f->nitems);
   113   for (i = 0; i < 
vec_len (data); i++)
   116   for (i = 0; i < trace_len; i++)
   119       if (trace[i].action == 1)
   122             s = 
format (s, 
"adding [%u, %u]:", trace[i].offset,
   124                          trace[i].len) % dummy_fifo->nitems);
   126                                         trace[i].len, &data[offset]);
   128       else if (trace[i].action == 2)
   131             s = 
format (s, 
"adding [%u, %u]:", 0, trace[i].len);
   137             s = 
format (s, 
"read: %u", trace[i].len);
   153   u32 ooo_segment_index = f->ooos_list_head;
   160       ooo_segment_index = seg->
next;
   170   int verbose = va_arg (*args, 
int);
   172   s = 
format (s, 
"cursize %u nitems %u has_event %d\n",
   173               f->cursize, f->nitems, f->has_event);
   174   s = 
format (s, 
" head %d tail %d\n", f->head, f->tail);
   178       (s, 
" server session %d thread %d client session %d thread %d\n",
   179        f->master_session_index, f->master_thread_index,
   180        f->client_session_index, f->client_thread_index);
   184       s = 
format (s, 
" ooo pool %d active elts newest %u\n",
   185                   pool_elts (f->ooo_segments), f->ooos_newest);
   197   u32 rounded_data_size;
   200   rounded_data_size = (1 << (
max_log2 (data_size_in_bytes)));
   206   memset (f, 0, 
sizeof (*f));
   207   f->nitems = data_size_in_bytes;
   218   if (--f->refcnt == 0)
   249       next->prev = cur->
prev;
   259       f->ooos_list_head = cur->
next;
   273   u32 new_index, s_end_pos, s_index;
   274   u32 normalized_position, normalized_end_position;
   277   normalized_position = (f->tail + 
offset) % f->nitems;
   278   normalized_end_position = (f->tail + offset + length) % f->nitems;
   285       f->ooos_list_head = s - f->ooo_segments;
   286       f->ooos_newest = f->ooos_list_head;
   310   s_index = s - f->ooo_segments;
   317       new_index = new_s - f->ooo_segments;
   325           prev->
next = new_index;
   330           f->ooos_list_head = new_index;
   333       new_s->
next = s_index;
   335       f->ooos_newest = new_index;
   339   else if (
position_gt (f, normalized_position, s_end_pos))
   342       new_index = new_s - f->ooo_segments;
   350       new_s->
prev = s_index;
   352       f->ooos_newest = new_index;
   364       s->
start = normalized_position;
   366       f->ooos_newest = s - f->ooo_segments;
   372   if (
position_gt (f, normalized_end_position, s_end_pos))
   379                                  normalized_end_position))
   393       f->ooos_newest = s - f->ooo_segments;
   405   u32 index, bytes = 0;
   411   ASSERT (diff != n_bytes_enqueued);
   413   if (diff > n_bytes_enqueued)
   417   while (0 <= diff && diff < n_bytes_enqueued)
   419       index = s - f->ooo_segments;
   426           f->tail %= f->nitems;
   446   ASSERT (bytes <= f->nitems);
   453   u32 total_copy_bytes, first_copy_bytes, second_copy_bytes;
   466   total_copy_bytes = (nitems - cursize) < max_bytes ?
   467     (nitems - cursize) : max_bytes;
   472       first_copy_bytes = ((nitems - f->tail) < total_copy_bytes)
   473         ? (nitems - f->tail) : total_copy_bytes;
   475       clib_memcpy (&f->data[f->tail], copy_from_here, first_copy_bytes);
   476       f->tail += first_copy_bytes;
   477       f->tail = (f->tail == nitems) ? 0 : f->tail;
   480       second_copy_bytes = total_copy_bytes - first_copy_bytes;
   481       if (second_copy_bytes)
   483           clib_memcpy (&f->data[f->tail], copy_from_here + first_copy_bytes,
   485           f->tail += second_copy_bytes;
   486           f->tail = (f->tail == nitems) ? 0 : f->tail;
   494       ASSERT (max_bytes <= (nitems - cursize));
   495       f->tail += max_bytes;
   496       f->tail = f->tail % nitems;
   497       total_copy_bytes = max_bytes;
   507   ASSERT (cursize + total_copy_bytes <= nitems);
   508   __sync_fetch_and_add (&f->cursize, total_copy_bytes);
   510   return (total_copy_bytes);
   513 #define SVM_ENQUEUE_CLONE_TEMPLATE(arch, fn, tgt)                       \   515   __attribute__ ((flatten))                                             \   516   __attribute__ ((target (tgt)))                                        \   518   fn ## _ ## arch ( svm_fifo_t * f, u32 max_bytes, u8 * copy_from_here) \   519   { return fn (f, max_bytes, copy_from_here);}   541     fp = (
void *) svm_fifo_enqueue_nowait_ma_multiarch_select ();
   543   return (*fp) (f, max_bytes, copy_from_here);
   560   u32 total_copy_bytes, first_copy_bytes, second_copy_bytes;
   561   u32 cursize, nitems, normalized_offset;
   569   ASSERT (required_bytes < nitems);
   571   normalized_offset = (f->tail + 
offset) % nitems;
   574   if ((required_bytes + offset) > (nitems - cursize))
   582   total_copy_bytes = required_bytes;
   585   first_copy_bytes = ((nitems - normalized_offset) < total_copy_bytes)
   586     ? (nitems - normalized_offset) : total_copy_bytes;
   588   clib_memcpy (&f->data[normalized_offset], copy_from_here, first_copy_bytes);
   591   second_copy_bytes = total_copy_bytes - first_copy_bytes;
   592   if (second_copy_bytes)
   594       normalized_offset += first_copy_bytes;
   595       normalized_offset %= nitems;
   597       ASSERT (normalized_offset == 0);
   600                    copy_from_here + first_copy_bytes, second_copy_bytes);
   610                               u32 required_bytes, 
u8 * copy_from_here)
   620   u32 total_copy_bytes, first_copy_bytes, second_copy_bytes;
   631   total_copy_bytes = (cursize < max_bytes) ? cursize : max_bytes;
   636       first_copy_bytes = ((nitems - f->head) < total_copy_bytes)
   637         ? (nitems - f->head) : total_copy_bytes;
   638       clib_memcpy (copy_here, &f->data[f->head], first_copy_bytes);
   639       f->head += first_copy_bytes;
   640       f->head = (f->head == nitems) ? 0 : f->head;
   643       second_copy_bytes = total_copy_bytes - first_copy_bytes;
   644       if (second_copy_bytes)
   647                        &f->data[f->head], second_copy_bytes);
   648           f->head += second_copy_bytes;
   649           f->head = (f->head == nitems) ? 0 : f->head;
   656       ASSERT (max_bytes <= cursize);
   657       f->head += max_bytes;
   658       f->head = f->head % nitems;
   659       cursize -= max_bytes;
   660       total_copy_bytes = max_bytes;
   663   ASSERT (f->head <= nitems);
   664   ASSERT (cursize >= total_copy_bytes);
   665   __sync_fetch_and_sub (&f->cursize, total_copy_bytes);
   667   return (total_copy_bytes);
   676 #define SVM_FIFO_DEQUEUE_CLONE_TEMPLATE(arch, fn, tgt)          \   678   __attribute__ ((flatten))                                     \   679   __attribute__ ((target (tgt)))                                \   681   fn ## _ ## arch ( svm_fifo_t * f, u32 max_bytes,              \   683   { return fn (f, max_bytes, copy_here);}   698     fp = (
void *) svm_fifo_dequeue_nowait_ma_multiarch_select ();
   700   return (*fp) (f, max_bytes, copy_here);
   708   u32 total_copy_bytes, first_copy_bytes, second_copy_bytes;
   709   u32 cursize, nitems, real_head;
   717   real_head = f->head + relative_offset;
   718   real_head = real_head >= nitems ? real_head - nitems : real_head;
   721   total_copy_bytes = (cursize - relative_offset < max_bytes) ?
   722     cursize - relative_offset : max_bytes;
   728         ((nitems - real_head) < total_copy_bytes) ?
   729         (nitems - real_head) : total_copy_bytes;
   730       clib_memcpy (copy_here, &f->data[real_head], first_copy_bytes);
   733       second_copy_bytes = total_copy_bytes - first_copy_bytes;
   734       if (second_copy_bytes)
   736           clib_memcpy (copy_here + first_copy_bytes, &f->data[0],
   740   return total_copy_bytes;
   743 #define SVM_FIFO_PEEK_CLONE_TEMPLATE(arch, fn, tgt)                     \   745   __attribute__ ((flatten))                                             \   746   __attribute__ ((target (tgt)))                                        \   748   fn ## _ ## arch ( svm_fifo_t * f, u32 relative_offset, u32 max_bytes, \   750   { return fn (f, relative_offset, max_bytes, copy_here);}   765     fp = (
void *) svm_fifo_peek_ma_multiarch_select ();
   767   return (*fp) (f, relative_offset, max_bytes, copy_here);
   774   u32 total_drop_bytes, first_drop_bytes, second_drop_bytes;
   785   total_drop_bytes = (cursize < max_bytes) ? cursize : max_bytes;
   791     ((nitems - f->head) < total_drop_bytes) ?
   792     (nitems - f->head) : total_drop_bytes;
   793   f->head += first_drop_bytes;
   794   f->head = (f->head == nitems) ? 0 : f->head;
   797   second_drop_bytes = total_drop_bytes - first_drop_bytes;
   798   if (second_drop_bytes)
   800       f->head += second_drop_bytes;
   801       f->head = (f->head == nitems) ? 0 : f->head;
   804   ASSERT (f->head <= nitems);
   805   ASSERT (cursize >= total_drop_bytes);
   806   __sync_fetch_and_sub (&f->cursize, total_drop_bytes);
   808   return total_drop_bytes;
   829   f->head = f->tail = pointer % f->nitems;
 #define vec_validate(V, I)
Make sure vector is long enough for given index (no header, unspecified alignment) ...
sll srl srl sll sra u16x4 i
void svm_fifo_init_pointers(svm_fifo_t *f, u32 pointer)
Set fifo pointers to requested offset. 
static u32 position_diff(svm_fifo_t *f, u32 posa, u32 posb)
static vlib_cli_command_t trace
(constructor) VLIB_CLI_COMMAND (trace) 
#define SVM_ENQUEUE_CLONE_TEMPLATE(arch, fn, tgt)
static u8 svm_fifo_has_ooo_data(svm_fifo_t *f)
void svm_fifo_free(svm_fifo_t *f)
u32 prev
Previous linked-list element pool index. 
static int ooo_segment_try_collect(svm_fifo_t *f, u32 n_bytes_enqueued)
Removes segments that can now be enqueued because the fifo's tail has advanced. 
ooo_segment_t * svm_fifo_first_ooo_segment(svm_fifo_t *f)
#define pool_get(P, E)
Allocate an object E from a pool P (unspecified alignment). 
static int svm_fifo_enqueue_internal(svm_fifo_t *f, u32 max_bytes, u8 *copy_from_here)
static u32 ooo_segment_distance_from_tail(svm_fifo_t *f, u32 pos)
struct _svm_fifo svm_fifo_t
static void ooo_segment_add(svm_fifo_t *f, u32 offset, u32 length)
Add segment to fifo's out-of-order segment list. 
static u32 svm_fifo_max_dequeue(svm_fifo_t *f)
u8 * format_ooo_list(u8 *s, va_list *args)
#define pool_elt_at_index(p, i)
Returns pointer to element at given index. 
static u8 position_gt(svm_fifo_t *f, u32 a, u32 b)
static int svm_fifo_enqueue_nowait_ma(svm_fifo_t *f, u32 max_bytes, u8 *copy_from_here)
u32 svm_fifo_number_ooo_segments(svm_fifo_t *f)
#define pool_put(P, E)
Free an object E in pool P. 
static ooo_segment_t * ooo_segment_next(svm_fifo_t *f, ooo_segment_t *s)
#define svm_fifo_trace_add(_f, _s, _l, _t)
u8 * svm_fifo_replay(u8 *s, svm_fifo_t *f, u8 no_read, u8 verbose)
#define SVM_FIFO_DEQUEUE_CLONE_TEMPLATE(arch, fn, tgt)
static int svm_fifo_peek_ma(svm_fifo_t *f, u32 relative_offset, u32 max_bytes, u8 *copy_here)
#define pool_free(p)
Free a pool. 
static u8 position_leq(svm_fifo_t *f, u32 a, u32 b)
#define clib_memcpy(a, b, c)
static ooo_segment_t * ooo_segment_get_prev(svm_fifo_t *f, ooo_segment_t *s)
int svm_fifo_enqueue_nowait(svm_fifo_t *f, u32 max_bytes, u8 *copy_from_here)
static void ooo_segment_del(svm_fifo_t *f, u32 index)
#define OOO_SEGMENT_INVALID_INDEX
u8 * format_ooo_segment(u8 *s, va_list *args)
static void * clib_mem_alloc_aligned_or_null(uword size, uword align)
u8 * format_svm_fifo(u8 *s, va_list *args)
static u8 position_lt(svm_fifo_t *f, u32 a, u32 b)
static void clib_mem_free(void *p)
static int svm_fifo_dequeue_nowait_ma(svm_fifo_t *f, u32 max_bytes, u8 *copy_here)
int svm_fifo_enqueue_with_offset(svm_fifo_t *f, u32 offset, u32 required_bytes, u8 *copy_from_here)
u32 length
Length of segment. 
CLIB_MULTIARCH_SELECT_FN(svm_fifo_enqueue_nowait_ma)
u8 * svm_fifo_dump_trace(u8 *s, svm_fifo_t *f)
u32 next
Next linked-list element pool index. 
template key/value backing page structure 
int svm_fifo_dequeue_drop(svm_fifo_t *f, u32 max_bytes)
static u32 ooo_segment_end_pos(svm_fifo_t *f, ooo_segment_t *s)
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant) 
static ooo_segment_t * ooo_segment_new(svm_fifo_t *f, u32 start, u32 length)
foreach_march_variant(SVM_ENQUEUE_CLONE_TEMPLATE, svm_fifo_enqueue_nowait_ma)
static uword max_log2(uword x)
static int svm_fifo_dequeue_internal(svm_fifo_t *f, u32 max_bytes, u8 *copy_here)
static int svm_fifo_enqueue_with_offset_internal(svm_fifo_t *f, u32 offset, u32 required_bytes, u8 *copy_from_here)
Enqueue a future segment. 
struct clib_bihash_value offset
template key/value backing page structure 
#define vec_foreach(var, vec)
Vector iterator. 
#define CLIB_CACHE_LINE_BYTES
int svm_fifo_peek(svm_fifo_t *f, u32 relative_offset, u32 max_bytes, u8 *copy_here)
#define SVM_FIFO_PEEK_CLONE_TEMPLATE(arch, fn, tgt)
int svm_fifo_dequeue_nowait(svm_fifo_t *f, u32 max_bytes, u8 *copy_here)
u32 start
Start of segment, normalized. 
static u32 ooo_segment_distance_to_tail(svm_fifo_t *f, u32 pos)
svm_fifo_t * svm_fifo_create(u32 data_size_in_bytes)
create an svm fifo, in the current heap. 
static uword pool_elts(void *v)
Number of active elements in a pool.