Linux Audio

Check our new training course

Loading...
Note: File does not exist in v4.10.11.
  1// SPDX-License-Identifier: GPL-2.0
  2#define _GNU_SOURCE
  3
  4#include <stdio.h>
  5#include <string.h>
  6
  7#include <ynl.h>
  8
  9#include <net/if.h>
 10
 11#include "netdev-user.h"
 12
 13struct stat {
 14	unsigned int ifc;
 15
 16	struct {
 17		unsigned int cnt;
 18		size_t refs, bytes;
 19	} live[2];
 20
 21	size_t alloc_slow, alloc_fast, recycle_ring, recycle_cache;
 22};
 23
 24struct stats_array {
 25	unsigned int i, max;
 26	struct stat *s;
 27};
 28
 29static struct stat *find_ifc(struct stats_array *a, unsigned int ifindex)
 30{
 31	unsigned int i;
 32
 33	for (i = 0; i < a->i; i++) {
 34		if (a->s[i].ifc == ifindex)
 35			return &a->s[i];
 36	}
 37
 38	a->i++;
 39	if (a->i == a->max) {
 40		a->max *= 2;
 41		a->s = reallocarray(a->s, a->max, sizeof(*a->s));
 42	}
 43	a->s[i].ifc = ifindex;
 44	return &a->s[i];
 45}
 46
 47static void count(struct stat *s, unsigned int l,
 48		  struct netdev_page_pool_get_rsp *pp)
 49{
 50	s->live[l].cnt++;
 51	if (pp->_present.inflight)
 52		s->live[l].refs += pp->inflight;
 53	if (pp->_present.inflight_mem)
 54		s->live[l].bytes += pp->inflight_mem;
 55}
 56
 57int main(int argc, char **argv)
 58{
 59	struct netdev_page_pool_stats_get_list *pp_stats;
 60	struct netdev_page_pool_get_list *pools;
 61	struct stats_array a = {};
 62	struct ynl_error yerr;
 63	struct ynl_sock *ys;
 64
 65	ys = ynl_sock_create(&ynl_netdev_family, &yerr);
 66	if (!ys) {
 67		fprintf(stderr, "YNL: %s\n", yerr.msg);
 68		return 1;
 69	}
 70
 71	a.max = 128;
 72	a.s = calloc(a.max, sizeof(*a.s));
 73	if (!a.s)
 74		goto err_close;
 75
 76	pools = netdev_page_pool_get_dump(ys);
 77	if (!pools)
 78		goto err_free;
 79
 80	ynl_dump_foreach(pools, pp) {
 81		struct stat *s = find_ifc(&a, pp->ifindex);
 82
 83		count(s, 1, pp);
 84		if (pp->_present.detach_time)
 85			count(s, 0, pp);
 86	}
 87	netdev_page_pool_get_list_free(pools);
 88
 89	pp_stats = netdev_page_pool_stats_get_dump(ys);
 90	if (!pp_stats)
 91		goto err_free;
 92
 93	ynl_dump_foreach(pp_stats, pp) {
 94		struct stat *s = find_ifc(&a, pp->info.ifindex);
 95
 96		if (pp->_present.alloc_fast)
 97			s->alloc_fast += pp->alloc_fast;
 98		if (pp->_present.alloc_slow)
 99			s->alloc_slow += pp->alloc_slow;
100		if (pp->_present.recycle_ring)
101			s->recycle_ring += pp->recycle_ring;
102		if (pp->_present.recycle_cached)
103			s->recycle_cache += pp->recycle_cached;
104	}
105	netdev_page_pool_stats_get_list_free(pp_stats);
106
107	for (unsigned int i = 0; i < a.i; i++) {
108		char ifname[IF_NAMESIZE];
109		struct stat *s = &a.s[i];
110		const char *name;
111		double recycle;
112
113		if (!s->ifc) {
114			name = "<orphan>\t";
115		} else {
116			name = if_indextoname(s->ifc, ifname);
117			if (name)
118				printf("%8s", name);
119			printf("[%d]\t", s->ifc);
120		}
121
122		printf("page pools: %u (zombies: %u)\n",
123		       s->live[1].cnt, s->live[0].cnt);
124		printf("\t\trefs: %zu bytes: %zu (refs: %zu bytes: %zu)\n",
125		       s->live[1].refs, s->live[1].bytes,
126		       s->live[0].refs, s->live[0].bytes);
127
128		/* We don't know how many pages are sitting in cache and ring
129		 * so we will under-count the recycling rate a bit.
130		 */
131		recycle = (double)(s->recycle_ring + s->recycle_cache) /
132			(s->alloc_fast + s->alloc_slow) * 100;
133		printf("\t\trecycling: %.1lf%% (alloc: %zu:%zu recycle: %zu:%zu)\n",
134		       recycle, s->alloc_slow, s->alloc_fast,
135		       s->recycle_ring, s->recycle_cache);
136	}
137
138	ynl_sock_destroy(ys);
139	return 0;
140
141err_free:
142	free(a.s);
143err_close:
144	fprintf(stderr, "YNL: %s\n", ys->err.msg);
145	ynl_sock_destroy(ys);
146	return 2;
147}