-
Notifications
You must be signed in to change notification settings - Fork 45
/
Copy pathhoma_metrics.c
425 lines (405 loc) · 17.7 KB
/
homa_metrics.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
// SPDX-License-Identifier: BSD-2-Clause
/* This file contains various functions for managing Homa's performance
* counters.
*/
#include "homa_impl.h"
DEFINE_PER_CPU(struct homa_metrics, homa_metrics);
/**
* homa_metric_append() - Formats a new metric and appends it to homa->metrics.
* @homa: The new data will appended to the @metrics field of
* this structure.
* @format: Standard printf-style format string describing the
* new metric. Arguments after this provide the usual
* values expected for printf-like functions.
*/
void homa_metric_append(struct homa *homa, const char *format, ...)
{
char *new_buffer;
size_t new_chars;
va_list ap;
if (!homa->metrics) {
#ifdef __UNIT_TEST__
homa->metrics_capacity = 30;
#else
homa->metrics_capacity = 4096;
#endif
homa->metrics = kmalloc(homa->metrics_capacity, GFP_KERNEL);
if (!homa->metrics) {
pr_warn("%s couldn't allocate memory\n", __func__);
return;
}
homa->metrics_length = 0;
}
/* May have to execute this loop multiple times if we run out
* of space in homa->metrics; each iteration expands the storage,
* until eventually it is large enough.
*/
while (true) {
va_start(ap, format);
new_chars = vsnprintf(homa->metrics + homa->metrics_length,
homa->metrics_capacity -
homa->metrics_length, format, ap);
va_end(ap);
if ((homa->metrics_length + new_chars) < homa->metrics_capacity)
break;
/* Not enough room; expand buffer capacity. */
homa->metrics_capacity *= 2;
new_buffer = kmalloc(homa->metrics_capacity, GFP_KERNEL);
if (!new_buffer) {
pr_warn("%s couldn't allocate memory\n", __func__);
return;
}
memcpy(new_buffer, homa->metrics, homa->metrics_length);
kfree(homa->metrics);
homa->metrics = new_buffer;
}
homa->metrics_length += new_chars;
}
/**
* homa_metrics_print() - Sample all of the Homa performance metrics and
* generate a human-readable string describing all of them.
* @homa: Overall data about the Homa protocol implementation;
* the formatted string will be stored in homa->metrics.
*
* Return: The formatted string.
*/
char *homa_metrics_print(struct homa *homa)
{
int core, i, lower = 0;
homa->metrics_length = 0;
#define M(...) homa_metric_append(homa, __VA_ARGS__)
M("time_ns %20llu sched_clock() time when metrics were gathered\n",
sched_clock());
for (core = 0; core < nr_cpu_ids; core++) {
struct homa_metrics *m = &per_cpu(homa_metrics, core);
__s64 delta;
M("core %15d Core id for following metrics\n",
core);
for (i = 0; i < HOMA_NUM_SMALL_COUNTS; i++) {
M("msg_bytes_%-9d %15llu Bytes in incoming messages containing %d-%d bytes\n",
(i + 1) * 64, m->small_msg_bytes[i], lower,
(i + 1) * 64);
lower = (i + 1) * 64 + 1;
}
for (i = (HOMA_NUM_SMALL_COUNTS * 64) / 1024;
i < HOMA_NUM_MEDIUM_COUNTS; i++) {
M("msg_bytes_%-9d %15llu Bytes in incoming messages containing %d-%d bytes\n",
(i + 1) * 1024, m->medium_msg_bytes[i], lower,
(i + 1) * 1024);
lower = (i + 1) * 1024 + 1;
}
M("large_msg_count %15llu # of incoming messages >= %d bytes\n",
m->large_msg_count, lower);
M("large_msg_bytes %15llu Bytes in incoming messages >= %d bytes\n",
m->large_msg_bytes, lower);
M("sent_msg_bytes %15llu otal bytes in all outgoing messages\n",
m->sent_msg_bytes);
for (i = DATA; i < BOGUS; i++) {
char *symbol = homa_symbol_for_type(i);
M("packets_sent_%-7s %15llu %s packets sent\n",
symbol, m->packets_sent[i - DATA], symbol);
}
for (i = DATA; i < BOGUS; i++) {
char *symbol = homa_symbol_for_type(i);
M("packets_rcvd_%-7s %15llu %s packets received\n",
symbol, m->packets_received[i - DATA], symbol);
}
for (i = 0; i < HOMA_MAX_PRIORITIES; i++) {
M("priority%d_bytes %15llu Bytes sent at priority %d (including headers)\n",
i, m->priority_bytes[i], i);
}
for (i = 0; i < HOMA_MAX_PRIORITIES; i++) {
M("priority%d_packets %15llu Packets sent at priority %d\n",
i, m->priority_packets[i], i);
}
M("skb_allocs %15llu sk_buffs allocated\n",
m->skb_allocs);
M("skb_alloc_ns %15llu Time spent allocating sk_buffs\n",
m->skb_alloc_ns);
M("skb_frees %15llu Data sk_buffs freed in normal paths\n",
m->skb_frees);
M("skb_free_ns %15llu Time spent freeing data sk_buffs\n",
m->skb_free_ns);
M("skb_page_allocs %15llu Pages allocated for sk_buff frags\n",
m->skb_page_allocs);
M("skb_page_alloc_ns %15llu Time spent allocating pages for sk_buff frags\n",
m->skb_page_alloc_ns);
M("requests_received %15llu Incoming request messages\n",
m->requests_received);
M("requests_queued %15llu Requests for which no thread was waiting\n",
m->requests_queued);
M("responses_received %15llu Incoming response messages\n",
m->responses_received);
M("responses_queued %15llu Responses for which no thread was waiting\n",
m->responses_queued);
M("fast_wakeups %15llu Messages received while polling\n",
m->fast_wakeups);
M("slow_wakeups %15llu Messages received after thread went to sleep\n",
m->slow_wakeups);
M("handoffs_thread_waiting %15llu RPC handoffs to waiting threads (vs. queue)\n",
m->handoffs_thread_waiting);
M("handoffs_alt_thread %15llu RPC handoffs not to first on list (avoid busy core)\n",
m->handoffs_alt_thread);
M("poll_ns %15llu Time spent polling for incoming messages\n",
m->poll_ns);
M("softirq_calls %15llu Calls to homa_softirq (i.e. # GRO pkts received)\n",
m->softirq_calls);
M("softirq_ns %15llu Time spent in homa_softirq during SoftIRQ\n",
m->softirq_ns);
M("bypass_softirq_ns %15llu Time spent in homa_softirq during bypass from GRO\n",
m->bypass_softirq_ns);
M("linux_softirq_ns %15llu Time spent in all Linux SoftIRQ\n",
m->linux_softirq_ns);
M("napi_ns %15llu Time spent in NAPI-level packet handling\n",
m->napi_ns);
M("send_ns %15llu Time spent in homa_sendmsg for requests\n",
m->send_ns);
M("send_calls %15llu Total invocations of homa_sendmsg for equests\n",
m->send_calls);
// It is possible for us to get here at a time when a
// thread has been blocked for a long time and has
// recorded blocked_ns, but hasn't finished the
// system call so recv_nss hasn't been incremented
// yet. If that happens, just record 0 to prevent
// underflow errors.
delta = m->recv_ns - m->blocked_ns;
if (delta < 0)
delta = 0;
M("recv_ns %15llu Unblocked time spent in recvmsg kernel call\n",
delta);
M("recv_calls %15llu Total invocations of recvmsg kernel call\n",
m->recv_calls);
M("blocked_ns %15llu Time spent blocked in homa_recvmsg\n",
m->blocked_ns);
M("reply_ns %15llu Time spent in homa_sendmsg for responses\n",
m->reply_ns);
M("reply_calls %15llu Total invocations of homa_sendmsg for responses\n",
m->reply_calls);
M("abort_ns %15llu Time spent in homa_ioc_abort kernel call\n",
m->reply_ns);
M("abort_calls %15llu Total invocations of abort kernel call\n",
m->reply_calls);
M("so_set_buf_ns %15llu Time spent in setsockopt SO_HOMA_RCVBUF\n",
m->so_set_buf_ns);
M("so_set_buf_calls %15llu Total invocations of setsockopt SO_HOMA_RCVBUF\n",
m->so_set_buf_calls);
M("grantable_lock_ns %15llu Time spent with homa->grantable_lock locked\n",
m->grantable_lock_ns);
M("timer_ns %15llu Time spent in homa_timer\n",
m->timer_ns);
M("timer_reap_ns %15llu Time in homa_timer spent reaping RPCs\n",
m->timer_reap_ns);
M("data_pkt_reap_ns %15llu Time in homa_data_pkt spent reaping RPCs\n",
m->data_pkt_reap_ns);
M("pacer_ns %15llu Time spent in homa_pacer_main\n",
m->pacer_ns);
M("homa_ns %15llu Total time in all Homa-related functions\n",
m->softirq_ns + m->napi_ns +
m->send_ns + m->recv_ns +
m->reply_ns - m->blocked_ns +
m->timer_ns + m->pacer_ns);
M("pacer_lost_ns %15llu Lost transmission time because pacer was slow\n",
m->pacer_lost_ns);
M("pacer_bytes %15llu Bytes transmitted when the pacer was active\n",
m->pacer_bytes);
M("pacer_skipped_rpcs %15llu Pacer aborts because of locked RPCs\n",
m->pacer_skipped_rpcs);
M("pacer_needed_help %15llu homa_pacer_xmit invocations from homa_check_pacer\n",
m->pacer_needed_help);
M("throttled_ns %15llu Time when the throttled queue was nonempty\n",
m->throttled_ns);
M("resent_packets %15llu DATA packets sent in response to RESENDs\n",
m->resent_packets);
M("peer_hash_links %15llu Hash chain link traversals in peer table\n",
m->peer_hash_links);
M("peer_new_entries %15llu New entries created in peer table\n",
m->peer_new_entries);
M("peer_kmalloc_errors %15llu kmalloc failures creating peer table entries\n",
m->peer_kmalloc_errors);
M("peer_route_errors %15llu Routing failures creating peer table entries\n",
m->peer_route_errors);
M("control_xmit_errors %15llu Errors sending control packets\n",
m->control_xmit_errors);
M("data_xmit_errors %15llu Errors sending data packets\n",
m->data_xmit_errors);
M("unknown_rpcs %15llu Non-grant packets discarded because RPC unknown\n",
m->unknown_rpcs);
M("server_cant_create_rpcs %15llu Packets discarded because server couldn't create RPC\n",
m->server_cant_create_rpcs);
M("unknown_packet_types %15llu Packets discarded because of unsupported type\n",
m->unknown_packet_types);
M("short_packets %15llu Packets discarded because too short\n",
m->short_packets);
M("packet_discards %15llu Non-resent packets discarded because data already received\n",
m->packet_discards);
M("resent_discards %15llu Resent packets discarded because data already received\n",
m->resent_discards);
M("resent_packets_used %15llu Retransmitted packets that were actually used\n",
m->resent_packets_used);
M("rpc_timeouts %15llu RPCs aborted because peer was nonresponsive\n",
m->rpc_timeouts);
M("server_rpc_discards %15llu RPCs discarded by server because of errors\n",
m->server_rpc_discards);
M("server_rpcs_unknown %15llu RPCs aborted by server because unknown to client\n",
m->server_rpcs_unknown);
M("client_lock_misses %15llu Bucket lock misses for client RPCs\n",
m->client_lock_misses);
M("client_lock_miss_ns %15llu Time lost waiting for client bucket locks\n",
m->client_lock_miss_ns);
M("server_lock_misses %15llu Bucket lock misses for server RPCs\n",
m->server_lock_misses);
M("server_lock_miss_ns %15llu Time lost waiting for server bucket locks\n",
m->server_lock_miss_ns);
M("socket_lock_misses %15llu Socket lock misses\n",
m->socket_lock_misses);
M("socket_lock_miss_ns %15llu Time lost waiting for socket locks\n",
m->socket_lock_miss_ns);
M("throttle_lock_misses %15llu Throttle lock misses\n",
m->throttle_lock_misses);
M("throttle_lock_miss_ns %15llu Time lost waiting for throttle locks\n",
m->throttle_lock_miss_ns);
M("peer_ack_lock_misses %15llu Misses on peer ack locks\n",
m->peer_ack_lock_misses);
M("peer_ack_lock_miss_ns %15llu Time lost waiting for peer ack locks\n",
m->peer_ack_lock_miss_ns);
M("grantable_lock_misses %15llu Grantable lock misses\n",
m->grantable_lock_misses);
M("grantable_lock_miss_ns %15llu Time lost waiting for grantable lock\n",
m->grantable_lock_miss_ns);
M("grantable_rpcs_integral %15llu Integral of homa->num_grantable_rpcs*dt\n",
m->grantable_rpcs_integral);
M("grant_recalc_calls %15llu Number of calls to homa_grant_recalc\n",
m->grant_recalc_calls);
M("grant_recalc_ns %15llu Time spent in homa_grant_recalc\n",
m->grant_recalc_ns);
M("grant_recalc_skips %15llu Number of times homa_grant_recalc skipped redundant work\n",
m->grant_recalc_skips);
M("grant_recalc_loops %15llu Number of times homa_grant_recalc looped back\n",
m->grant_recalc_loops);
M("grant_priority_bumps %15llu Number of times an RPC moved up in the grant priority order\n",
m->grant_priority_bumps);
M("fifo_grants %15llu Grants issued using FIFO priority\n",
m->fifo_grants);
M("fifo_grants_no_incoming %15llu FIFO grants to messages with no outstanding grants\n",
m->fifo_grants_no_incoming);
M("disabled_reaps %15llu Reaper invocations that were disabled\n",
m->disabled_reaps);
M("disabled_rpc_reaps %15llu Disabled RPCs skipped by reaper\n",
m->disabled_rpc_reaps);
M("reaper_calls %15llu Reaper invocations that were not disabled\n",
m->reaper_calls);
M("reaper_dead_skbs %15llu Sum of hsk->dead_skbs across all reaper calls\n",
m->reaper_dead_skbs);
M("forced_reaps %15llu Reaps forced by accumulation of dead RPCs\n",
m->forced_reaps);
M("throttle_list_adds %15llu Calls to homa_add_to_throttled\n",
m->throttle_list_adds);
M("throttle_list_checks %15llu List elements checked in homa_add_to_throttled\n",
m->throttle_list_checks);
M("ack_overflows %15llu Explicit ACKs sent because peer->acks was full\n",
m->ack_overflows);
M("ignored_need_acks %15llu NEED_ACKs ignored because RPC result not yet received\n",
m->ignored_need_acks);
M("bpage_reuses %15llu Buffer page could be reused because ref count was zero\n",
m->bpage_reuses);
M("buffer_alloc_failures %15llu homa_pool_allocate didn't find enough buffer space for an RPC\n",
m->buffer_alloc_failures);
M("linux_pkt_alloc_bytes %15llu Bytes allocated in new packets by NIC driver due to cache overflows\n",
m->linux_pkt_alloc_bytes);
M("dropped_data_no_bufs %15llu Data bytes dropped because app buffers full\n",
m->dropped_data_no_bufs);
M("gen3_handoffs %15llu GRO->SoftIRQ handoffs made by Gen3 balancer\n",
m->gen3_handoffs);
M("gen3_alt_handoffs %15llu Gen3 handoffs to secondary core (primary was busy)\n",
m->gen3_alt_handoffs);
M("gro_grant_bypasses %15llu Grant packets passed directly to homa_softirq by homa_gro_receive\n",
m->gro_grant_bypasses);
M("gro_data_bypasses %15llu Data packets passed directly to homa_softirq by homa_gro_receive\n",
m->gro_data_bypasses);
for (i = 0; i < NUM_TEMP_METRICS; i++)
M("temp%-2d %15llu Temporary use in testing\n",
i, m->temp[i]);
}
return homa->metrics;
}
/**
* homa_metrics_open() - This function is invoked when /proc/net/homa_metrics is
* opened.
* @inode: The inode corresponding to the file.
* @file: Information about the open file.
*
* Return: always 0.
*/
int homa_metrics_open(struct inode *inode, struct file *file)
{
struct homa *homa = global_homa;
/* Collect all of the metrics when the file is opened, and save
* these for use by subsequent reads (don't want the metrics to
* change between reads). If there are concurrent opens on the
* file, only read the metrics once, during the first open, and
* use this copy for subsequent opens, until the file has been
* completely closed.
*/
spin_lock(&homa->metrics_lock);
if (homa->metrics_active_opens == 0)
homa_metrics_print(homa);
homa->metrics_active_opens++;
spin_unlock(&homa->metrics_lock);
return 0;
}
/**
* homa_metrics_read() - This function is invoked to handle read kernel calls on
* /proc/net/homa_metrics.
* @file: Information about the file being read.
* @buffer: Address in user space of the buffer in which data from the file
* should be returned.
* @length: Number of bytes available at @buffer.
* @offset: Current read offset within the file.
*
* Return: the number of bytes returned at @buffer. 0 means the end of the
* file was reached, and a negative number indicates an error (-errno).
*/
ssize_t homa_metrics_read(struct file *file, char __user *buffer,
size_t length, loff_t *offset)
{
struct homa *homa = global_homa;
size_t copied;
if (*offset >= homa->metrics_length)
return 0;
copied = homa->metrics_length - *offset;
if (copied > length)
copied = length;
if (copy_to_user(buffer, homa->metrics + *offset, copied))
return -EFAULT;
*offset += copied;
return copied;
}
/**
* homa_metrics_lseek() - This function is invoked to handle seeks on
* /proc/net/homa_metrics. Right now seeks are ignored: the file must be
* read sequentially.
* @file: Information about the file being read.
* @offset: Distance to seek, in bytes
* @whence: Starting point from which to measure the distance to seek.
* Return: current position within file.
*/
loff_t homa_metrics_lseek(struct file *file, loff_t offset, int whence)
{
return 0;
}
/**
* homa_metrics_release() - This function is invoked when the last reference to
* an open /proc/net/homa_metrics is closed. It performs cleanup.
* @inode: The inode corresponding to the file.
* @file: Information about the open file.
*
* Return: always 0.
*/
int homa_metrics_release(struct inode *inode, struct file *file)
{
struct homa *homa = global_homa;
spin_lock(&homa->metrics_lock);
homa->metrics_active_opens--;
spin_unlock(&homa->metrics_lock);
return 0;
}