132 for (i = 0; i < l; i++)
138 if (t->
format[i + 1] ==
'%')
227 digits[0] = digits[1] = 0;
228 while (p[i] >=
'0' && p[i] <=
'9')
232 digits[
i] = p[
i] -
'0';
236 if (i >= 1 && i <= 2)
241 *number = 10 * digits[0] + digits[1];
259 if (f[0] ==
'%' && f[1] !=
'%')
273 f += f[0] ==
'+' || f[0] ==
'-' || f[0] ==
'=';
276 while ((f[0] >=
'0' && f[0] <=
'9') || f[0] ==
'.')
280 f += f[0] ==
'w' || f[0] ==
'l' || f[0] ==
'L';
285 ASSERT (*result_len > f - percent);
286 l =
clib_min (f - percent, *result_len - 1);
291 *result_len = f -
fmt;
311 uword n_bytes = 0, n_digits, f_bytes = 0;
313 f_bytes =
sizeof (arg_format);
317 if (a == 0 || a[0] == 0)
339 else if (n_bytes == 2)
341 else if (n_bytes == 4)
343 else if (n_bytes == 8)
351 s =
format (s, arg_format, e);
353 else if (a[0] ==
'T')
357 s =
format (s, arg_format, e);
359 else if (n_bytes == 8)
360 s =
format (s, arg_format, l);
362 s =
format (s, arg_format, i);
371 else if (n_bytes == 8)
375 s =
format (s, arg_format, x);
380 s =
format (s, arg_format, d);
382 n_bytes = strlen (d) + 1;
390 ASSERT (n_digits > 0 && n_digits <= 2);
411 f64 dt = va_arg (*args,
f64);
412 int track_index = va_arg (*args,
int);
421 if (e->
track != track_index)
442 u64 cpu_time_now, os_time_now_nsec;
447 #include <sys/syscall.h> 449 clock_gettime (CLOCK_REALTIME, &ts);
455 os_time_now_nsec = 1e9 * (ts.tv_sec - 1490885108) + ts.tv_nsec;
459 os_time_now_nsec = 0;
462 et->
cpu = cpu_time_now;
463 et->
os_nsec = os_time_now_nsec;
563 for (i = 0; i < n; i++)
647 uword n_bytes = 0, n_digits;
674 ASSERT (n_digits > 0 && n_digits <= 2);
704 u32 string_table_offset_for_src_events;
705 u32 track_offset_for_src_tracks;
729 new_name = (
char *)
format (0,
"%s:%s%c", dst_tag, t->
name, 0);
763 string_table_offset_for_src_events);
766 e->
track += track_offset_for_src_tracks;
771 f64 dt_event, dt_os_nsec, dt_clock_nsec;
784 dt_event = dt_os_nsec;
802 &&
fabs (dt_os_nsec - dt_clock_nsec) < 100)
803 dt_event = dt_clock_nsec;
831 e->
time += dt_event + align_tweak;
836 dt_event = -dt_event;
837 for (e = dst->
events + 0; e < dst->events + l; e++)
838 e->
time += dt_event + align_tweak;
881 uword n_digits, n_bytes = 0;
892 else if (n_bytes == 2)
894 else if (n_bytes == 4)
896 else if (n_bytes == 8)
905 n_bytes = strlen ((
char *) d) + 1;
911 else if (n_bytes == 8)
958 uword n_digits, n_bytes = 0;
973 else if (n_bytes == 2)
978 else if (n_bytes == 4)
983 else if (n_bytes == 8)
998 n_bytes = strlen (t) + 1;
1011 else if (n_bytes == 8)
1035 int n = va_arg (*va,
int);
1037 for (i = 0; i < n; i++)
1044 sizeof (t[i].n_enum_strings));
1054 int n = va_arg (*va,
int);
1056 for (i = 0; i < n; i++)
1063 sizeof (t[i].n_enum_strings));
1064 vec_resize (t[i].enum_strings_vector, t[i].n_enum_strings);
1074 int n = va_arg (*va,
int);
1076 for (i = 0; i < n; i++)
1086 int n = va_arg (*va,
int);
1088 for (i = 0; i < n; i++)
1116 int flush_ring = va_arg (*va,
int);
__clib_export elog_event_t * elog_get_events(elog_main_t *em)
convert event ring events to events, and return them as a vector.
uword * string_table_hash
char ** enum_strings_vector
String table as a vector constructed when type is registered.
__clib_export void * elog_event_data(elog_main_t *em, elog_event_type_t *type, elog_track_t *track, u64 cpu_time)
f64 time
Absolute time as floating point number in seconds.
elog_time_stamp_t serialize_time
__clib_export void elog_alloc(elog_main_t *em, u32 n_events)
void unserialize_check_magic(serialize_main_t *m, void *magic, u32 magic_bytes)
static void maybe_fix_string_table_offset(elog_event_t *e, elog_event_type_t *t, u32 offset)
u16 event_type
Event type index.
static i64 elog_time_stamp_diff_os_nsec(elog_time_stamp_t *t1, elog_time_stamp_t *t2)
void elog_time_now(elog_time_stamp_t *et)
#define vec_serialize(m, v, f)
elog_time_stamp_t init_time
Timestamps.
clib_memset(h->entries, 0, sizeof(h->entries[0]) *entries)
#define vec_unserialize(m, v, f)
serialize_function_t unserialize_64
u64 time_cycles
Absolute time stamp in CPU clock cycles.
__clib_export u8 * format_elog_track(u8 *s, va_list *args)
#define vec_add1(V, E)
Add 1 element to end of vector (unspecified alignment).
static u64 clib_cpu_time_now(void)
char * function
Function name generating event.
#define vec_add2(V, P, N)
Add N elements to end of vector V, return pointer to new elements in P.
#define hash_set_mem(h, key, value)
static void elog_lock(elog_main_t *em)
char * format
Format string.
static uword find_or_create_type(elog_main_t *em, elog_event_type_t *t)
#define vec_validate_aligned(V, I, A)
Make sure vector is long enough for given index (no header, specified alignment)
serialize_function_t serialize_64
u8 data[20]
20-bytes of data follows, pads to 32 bytes.
static void serialize_elog_time_stamp(serialize_main_t *m, va_list *va)
char * enum_strings[]
String table for enum/number to string formatting.
#define vec_reset_length(v)
Reset vector length to zero NULL-pointer tolerant.
static void unserialize_elog_event_type(serialize_main_t *m, va_list *va)
#define clib_memcpy(d, s, n)
static void new_event_type(elog_main_t *em, uword i)
#define vec_add(V, E, N)
Add N elements to end of vector V (no header, unspecified alignment)
void unserialize_elog_main(serialize_main_t *m, va_list *va)
u32 n_enum_strings
Number of elements in string enum table.
serialize_function_t unserialize_f32
static void unserialize_elog_event(serialize_main_t *m, va_list *va)
word elog_event_type_register(elog_main_t *em, elog_event_type_t *t)
register an event type
u32 type_index_plus_one
Type index plus one assigned to this type.
u8 * format_elog_track_name(u8 *s, va_list *va)
static i64 elog_time_stamp_diff_cpu(elog_time_stamp_t *t1, elog_time_stamp_t *t2)
description fragment has unexpected format
const char *const const double number
__clib_export void serialize_close(serialize_main_t *m)
static void unserialize_elog_time_stamp(serialize_main_t *m, va_list *va)
#define vec_elt_at_index(v, i)
Get vector value at index i checking that i is in bounds.
uword * lock
SMP lock, non-zero means locking required.
static uword elog_event_range(elog_main_t *em, uword *lo)
char * format_args
Specifies how arguments to format are parsed from event data.
#define vec_resize(V, N)
Resize a vector (no header, unspecified alignment) Add N elements to end of given vector V...
__clib_export void serialize_cstring(serialize_main_t *m, char *s)
#define vec_end(v)
End (last data address) of vector.
uword * event_type_by_format
Hash table mapping type format to type index.
__clib_export clib_error_t * elog_read_file_not_inline(elog_main_t *em, char *clib_file)
#define clib_atomic_test_and_set(a)
#define hash_create_string(elts, value_bytes)
static int elog_cmp(void *a1, void *a2)
vl_api_fib_path_type_t type
elog_event_type_t * event_types
Vector of event types.
__clib_export void serialize_elog_main(serialize_main_t *m, va_list *va)
static uword parse_2digit_decimal(char *p, uword *number)
#define clib_atomic_release(a)
static char * elog_serialize_magic
char * name
Track name vector.
elog_event_t * event_ring
Vector of events (circular buffer).
static void serialize_elog_event_type(serialize_main_t *m, va_list *va)
f64 nsec_per_cpu_clock
Use serialize_time and init_time to give estimate for cpu clock frequency.
static void unserialize_elog_track(serialize_main_t *m, va_list *va)
serialize_function_t unserialize_f64
static void elog_alloc_internal(elog_main_t *em, u32 n_events, int free_ring)
static void elog_unlock(elog_main_t *em)
__clib_export void elog_init(elog_main_t *em, u32 n_events)
The fine-grained event logger allows lightweight, thread-safe event logging at minimum cost...
__clib_export void unserialize_close(serialize_main_t *m)
sll srl srl sll sra u16x4 i
#define vec_free(V)
Free vector's memory (no header).
__clib_export clib_error_t * serialize(serialize_main_t *m,...)
static f64 elog_nsec_per_clock(elog_main_t *em)
static void serialize_elog_event(serialize_main_t *m, va_list *va)
__clib_export clib_error_t * unserialize_open_clib_file(serialize_main_t *m, char *file)
static uword max_pow2(uword x)
u32 track_index_plus_one
Set to one when track has been added to main structure.
static void unserialize_integer(serialize_main_t *m, void *x, u32 n_bytes)
static void serialize_integer(serialize_main_t *m, u64 x, u32 n_bytes)
__clib_export void elog_resize(elog_main_t *em, u32 n_events)
u32 n_total_events_disable_limit
When count reaches limit logging is disabled.
static void serialize_elog_track(serialize_main_t *m, va_list *va)
u64 os_nsec
OS timer in nano secs since epoch 3/30/2017, see elog_time_now()
serialize_function_t unserialize_vec_8
void serialize_magic(serialize_main_t *m, void *magic, u32 magic_bytes)
uword event_ring_size
Power of 2 number of elements in ring.
__clib_export void clib_time_init(clib_time_t *c)
u64 cpu
CPU cycle counter.
#define vec_append(v1, v2)
Append v2 after v1.
u16 track
Track for this event.
#define clib_mem_unaligned(pointer, type)
clib_error_t * elog_write_file_not_inline(elog_main_t *em, char *clib_file, int flush_ring)
clib_time_t cpu_timer
Place holder for CPU clock frequency.
serialize_function_t serialize_f32
#define vec_elt(v, i)
Get vector value at index i.
elog_event_t * events
Vector of events converted to generic form after collection.
template key/value backing page structure
#define hash_create_vec(elts, key_bytes, value_bytes)
serialize_function_t serialize_vec_8
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
#define vec_sort_with_function(vec, f)
Sort a vector using the supplied element comparison function.
__clib_export void unserialize_cstring(serialize_main_t *m, char **s)
__clib_export u32 elog_string(elog_main_t *em, char *fmt,...)
add a string to the event-log string table
elog_track_t * tracks
Vector of tracks.
#define hash_get_mem(h, key)
struct clib_bihash_value offset
template key/value backing page structure
__clib_export elog_event_t * elog_peek_events(elog_main_t *em)
convert event ring events to events, and return them as a vector.
static u8 * fixed_format(u8 *s, char *fmt, char *result, uword *result_len)
char * string_table
Events may refer to strings in string table.
#define vec_foreach(var, vec)
Vector iterator.
#define SYS_clock_gettime
elog_track_t default_track
Default track.
__clib_export char * format_one_elog_event(void *em_arg, void *ep_arg)
void elog_merge(elog_main_t *dst, u8 *dst_tag, elog_main_t *src, u8 *src_tag, f64 align_tweak)
u32 n_total_events
Total number of events in buffer.
#define CLIB_CACHE_LINE_BYTES
__clib_export word elog_track_register(elog_main_t *em, elog_track_t *t)
register an event track
__clib_export clib_error_t * unserialize(serialize_main_t *m,...)
static void * elog_event_data_inline(elog_main_t *em, elog_event_type_t *type, elog_track_t *track, u64 cpu_time)
Allocate an event to be filled in by the caller.
__clib_export clib_error_t * serialize_open_clib_file(serialize_main_t *m, char *file)
__clib_export u8 * format_elog_event(u8 *s, va_list *va)
serialize_function_t serialize_f64