blob: b23f3df5713c0df0069d2ecf49f632c4dcee9e6a [file] [log] [blame]
Damjan Marion8973b072022-03-01 15:51:18 +01001/* SPDX-License-Identifier: Apache-2.0
2 * Copyright(c) 2022 Cisco Systems, Inc.
3 */
4
5#include <vlib/vlib.h>
6#include <vlib/unix/unix.h>
7#include <vlib/stats/stats.h>
8
Damjan Marionc5d81b92022-04-15 15:53:17 +02009enum
10{
11 NODE_CLOCKS,
12 NODE_VECTORS,
13 NODE_CALLS,
14 NODE_SUSPENDS,
15 N_NODE_COUNTERS
16};
17
18struct
19{
20 u32 entry_index;
21 char *name;
22} node_counters[] = {
23 [NODE_CLOCKS] = { .name = "clocks" },
24 [NODE_VECTORS] = { .name = "vectors" },
25 [NODE_CALLS] = { .name = "calls" },
26 [NODE_SUSPENDS] = { .name = "suspends" },
27};
28
29static struct
30{
31 u8 *name;
32 u32 symlinks[N_NODE_COUNTERS];
33} *node_data = 0;
34
35static vlib_stats_string_vector_t node_names = 0;
36
Damjan Marion8973b072022-03-01 15:51:18 +010037static inline void
38update_node_counters (vlib_stats_segment_t *sm)
39{
Damjan Marionc5d81b92022-04-15 15:53:17 +020040 clib_bitmap_t *bmp = 0;
Damjan Marion8973b072022-03-01 15:51:18 +010041 vlib_main_t **stat_vms = 0;
42 vlib_node_t ***node_dups = 0;
Damjan Marionc5d81b92022-04-15 15:53:17 +020043 u32 n_nodes;
Damjan Marion8973b072022-03-01 15:51:18 +010044 int i, j;
Damjan Marion8973b072022-03-01 15:51:18 +010045
46 vlib_node_get_nodes (0 /* vm, for barrier sync */,
47 (u32) ~0 /* all threads */, 1 /* include stats */,
48 0 /* barrier sync */, &node_dups, &stat_vms);
49
Damjan Marionc5d81b92022-04-15 15:53:17 +020050 n_nodes = vec_len (node_dups[0]);
Damjan Marion8973b072022-03-01 15:51:18 +010051
Damjan Marionc5d81b92022-04-15 15:53:17 +020052 vec_validate (node_data, n_nodes - 1);
53
54 for (i = 0; i < n_nodes; i++)
Xiaoming Jiang73507dd2022-12-08 08:08:41 +000055 if (vec_is_equal (node_data[i].name, node_dups[0][i]->name) == 0)
Damjan Marionc5d81b92022-04-15 15:53:17 +020056 bmp = clib_bitmap_set (bmp, i, 1);
57
58 if (bmp)
Damjan Marion8973b072022-03-01 15:51:18 +010059 {
Damjan Marionb3a5b392022-03-31 17:56:58 +020060 u32 last_thread = vlib_get_n_threads ();
Damjan Marion8973b072022-03-01 15:51:18 +010061 vlib_stats_segment_lock ();
Damjan Marionc5d81b92022-04-15 15:53:17 +020062 clib_bitmap_foreach (i, bmp)
Damjan Marion8973b072022-03-01 15:51:18 +010063 {
Damjan Marionc5d81b92022-04-15 15:53:17 +020064 if (node_data[i].name)
65 {
66 vec_free (node_data[i].name);
67 for (j = 0; j < ARRAY_LEN (node_data->symlinks); j++)
68 vlib_stats_remove_entry (node_data[i].symlinks[j]);
69 }
Arthur de Kerhoradd6a382022-05-16 19:46:22 +020070 }
71 /* We can't merge the loops because a node index corresponding to a given
72 * node name can change between 2 updates. Otherwise, we could add
73 * already existing symlinks or delete valid ones.
74 */
75 clib_bitmap_foreach (i, bmp)
76 {
77 vlib_node_t *n = node_dups[0][i];
78 node_data[i].name = vec_dup (n->name);
Damjan Marionc5d81b92022-04-15 15:53:17 +020079 vlib_stats_set_string_vector (&node_names, n->index, "%v", n->name);
80
81 for (int j = 0; j < ARRAY_LEN (node_counters); j++)
82 {
83 vlib_stats_validate (node_counters[j].entry_index, last_thread,
84 n_nodes - 1);
85 node_data[i].symlinks[j] = vlib_stats_add_symlink (
86 node_counters[j].entry_index, n->index, "/nodes/%U/%s",
87 format_vlib_stats_symlink, n->name, node_counters[j].name);
88 ASSERT (node_data[i].symlinks[j] != CLIB_U32_MAX);
89 }
Damjan Marion8973b072022-03-01 15:51:18 +010090 }
Damjan Marion8973b072022-03-01 15:51:18 +010091 vlib_stats_segment_unlock ();
Damjan Marionc5d81b92022-04-15 15:53:17 +020092 vec_free (bmp);
Damjan Marion8973b072022-03-01 15:51:18 +010093 }
94
95 for (j = 0; j < vec_len (node_dups); j++)
96 {
97 vlib_node_t **nodes = node_dups[j];
98
99 for (i = 0; i < vec_len (nodes); i++)
100 {
101 counter_t **counters;
102 counter_t *c;
103 vlib_node_t *n = nodes[i];
104
Damjan Marionc5d81b92022-04-15 15:53:17 +0200105 counters = vlib_stats_get_entry_data_pointer (
106 node_counters[NODE_CLOCKS].entry_index);
Damjan Marion8973b072022-03-01 15:51:18 +0100107 c = counters[j];
108 c[n->index] = n->stats_total.clocks - n->stats_last_clear.clocks;
109
Damjan Marionc5d81b92022-04-15 15:53:17 +0200110 counters = vlib_stats_get_entry_data_pointer (
111 node_counters[NODE_VECTORS].entry_index);
Damjan Marion8973b072022-03-01 15:51:18 +0100112 c = counters[j];
113 c[n->index] = n->stats_total.vectors - n->stats_last_clear.vectors;
114
Damjan Marionc5d81b92022-04-15 15:53:17 +0200115 counters = vlib_stats_get_entry_data_pointer (
116 node_counters[NODE_CALLS].entry_index);
Damjan Marion8973b072022-03-01 15:51:18 +0100117 c = counters[j];
118 c[n->index] = n->stats_total.calls - n->stats_last_clear.calls;
119
Damjan Marionc5d81b92022-04-15 15:53:17 +0200120 counters = vlib_stats_get_entry_data_pointer (
121 node_counters[NODE_SUSPENDS].entry_index);
Damjan Marion8973b072022-03-01 15:51:18 +0100122 c = counters[j];
123 c[n->index] = n->stats_total.suspends - n->stats_last_clear.suspends;
124 }
125 vec_free (node_dups[j]);
126 }
127 vec_free (node_dups);
128 vec_free (stat_vms);
129}
130
131static void
132do_stat_segment_updates (vlib_main_t *vm, vlib_stats_segment_t *sm)
133{
134 if (sm->node_counters_enabled)
135 update_node_counters (sm);
136
137 vlib_stats_collector_t *c;
138 pool_foreach (c, sm->collectors)
139 {
140 vlib_stats_collector_data_t data = {
141 .entry_index = c->entry_index,
142 .vector_index = c->vector_index,
143 .private_data = c->private_data,
144 .entry = sm->directory_vector + c->entry_index,
145 };
146 c->fn (&data);
147 }
148
149 /* Heartbeat, so clients detect we're still here */
150 sm->directory_vector[STAT_COUNTER_HEARTBEAT].value++;
151}
152
153static uword
154stat_segment_collector_process (vlib_main_t *vm, vlib_node_runtime_t *rt,
155 vlib_frame_t *f)
156{
157 vlib_stats_segment_t *sm = vlib_stats_get_segment ();
158
Damjan Marionc5d81b92022-04-15 15:53:17 +0200159 if (sm->node_counters_enabled)
160 {
161 node_names = vlib_stats_add_string_vector ("/sys/node/names");
162 ASSERT (node_names);
163
164 for (int x = 0; x < ARRAY_LEN (node_counters); x++)
165 {
166 node_counters[x].entry_index = vlib_stats_add_counter_vector (
167 "/sys/node/%s", node_counters[x].name);
168 ASSERT (node_counters[x].entry_index != CLIB_U32_MAX);
169 }
170 }
171
Ole Troan5010bbd2022-11-28 13:13:24 +0100172 sm->directory_vector[STAT_COUNTER_BOOTTIME].value = unix_time_now ();
173
Damjan Marion8973b072022-03-01 15:51:18 +0100174 while (1)
175 {
176 do_stat_segment_updates (vm, sm);
177 vlib_process_suspend (vm, sm->update_interval);
178 }
179 return 0; /* or not */
180}
181
182VLIB_REGISTER_NODE (stat_segment_collector, static) = {
183 .function = stat_segment_collector_process,
184 .name = "statseg-collector-process",
185 .type = VLIB_NODE_TYPE_PROCESS,
186};