1 #include <stdlib.h>
2 #include <stdio.h>
3 #include <string.h>
4 #include <unistd.h>
5 #include <errno.h>
6 #include <ctype.h>
7 #include <limits.h>
8 
9 #define STRINGIFY_ARG(a)        #a
10 #define STRINGIFY(a)            STRINGIFY_ARG(a)
11 
12 #define DEF_SORT_FUNC		sort_nr_objs
13 #define SLABINFO_LINE_LEN	512	/* size of longest line */
14 #define SLABINFO_NAME_LEN	32	/* cache name size (will truncate) */
15 #define SLABINFO_FILE		"/proc/slabinfo"
16 #define DEF_NR_ROWS		15	/* default nr of caches to show */
17 
18 /* object representing a slab cache (each line of slabinfo) */
19 struct slab_info {
20 	char name[SLABINFO_NAME_LEN];	/* name of this cache */
21 	struct slab_info *next;
22 	unsigned long nr_pages;		/* size of cache in pages */
23 	unsigned long nr_objs;		/* number of objects in this cache */
24 	unsigned long nr_active_objs;	/* number of active objects */
25 	unsigned long obj_size;		/* size of each object */
26 	unsigned long objs_per_slab;	/* number of objects per slab */
27 	unsigned long nr_slabs;		/* number of slabs in this cache */
28 	unsigned long use;		/* percent full: total / active */
29 };
30 
31 /* object representing system-wide statistics */
32 struct slab_stat {
33 	unsigned long total_size;	/* size of all objects */
34 	unsigned long active_size;	/* size of all active objects */
35 	unsigned long nr_objs;		/* total number of objects */
36 	unsigned long nr_active_objs;	/* total number of active objects */
37 	unsigned long nr_slabs;		/* total number of slabs */
38 	unsigned long nr_active_slabs;	/* total number of active slabs*/
39 	unsigned long nr_caches;	/* number of caches */
40 	unsigned long nr_active_caches;	/* number of active caches */
41 	unsigned long avg_obj_size;	/* average object size */
42 	unsigned long min_obj_size;	/* size of smallest object */
43 	unsigned long max_obj_size;	/* size of largest object */
44 };
45 
46 typedef int (*sort_t)(const struct slab_info *, const struct slab_info *);
47 static sort_t sort_func;
48 
49 /*
50  * get_slabinfo - open, read, and parse a slabinfo 2.x file, which has the
51  * following format:
52  *
53  * slabinfo - version: 2.1
54  * <name>  <active_objs> <num_objs> <objsize> <objperslab> <pagesperslab>
55  * : tunables <limit> <batchcount> <sharedfactor>
56  * : slabdata <active_slabs> <num_slabs> <sharedavail>
57  *
58  * Returns the head of the new list of slab_info structures, or NULL on error.
59  */
60 static struct slab_info * get_slabinfo(struct slab_stat *stats)
61 {
62 	struct slab_info *head = NULL, *p = NULL, *prev = NULL;
63 	FILE *slabfile;
64 	char line[SLABINFO_LINE_LEN];
65 	unsigned int major, minor;
66 
67 	slabfile = fopen(SLABINFO_FILE, "r");
68 	if (!slabfile) {
69 		perror("fopen");
70 		return NULL;
71 	}
72 
73 	if (!fgets(line, SLABINFO_LINE_LEN, slabfile)) {
74 		fprintf(stderr, "cannot read from " SLABINFO_FILE "\n");
75 		return NULL;
76 	}
77 
78 	if (sscanf(line, "slabinfo - version: %u.%u", &major, &minor) != 2) {
79 		fprintf(stderr, "unable to parse slabinfo version!\n");
80 		return NULL;
81 	}
82 
83 	if (major != 2 || minor > 1) {
84 		fprintf(stderr, "we only support slabinfo 2.0 and 2.1!\n");
85 		return NULL;
86 	}
87 
88 	stats->min_obj_size = INT_MAX;
89 
90 	while (fgets(line, SLABINFO_LINE_LEN, slabfile)) {
91 		unsigned long nr_active_slabs, pages_per_slab;
92 		int ret;
93 
94 		if (line[0] == '#')
95 			continue;
96 
97 		p = malloc(sizeof (struct slab_info));
98 		if (!p) {
99 			perror("malloc");
100 			head = NULL;
101 			break;
102 		}
103 		if (stats->nr_caches++ == 0)
104 			head = prev = p;
105 
106 		ret = sscanf(line, "%" STRINGIFY(SLABINFO_NAME_LEN) "s"
107 			     " %lu %lu %lu %lu %lu : tunables %*d %*d %*d : \
108 			     slabdata %lu %lu %*d", p->name,
109 			     &p->nr_active_objs, &p->nr_objs,
110 			     &p->obj_size, &p->objs_per_slab,
111 			     &pages_per_slab,
112 			     &nr_active_slabs,
113 			     &p->nr_slabs);
114 
115 		if (ret != 8) {
116 			fprintf(stderr, "unrecognizable data in slabinfo!\n");
117 			head = NULL;
118 			break;
119 		}
120 
121 		if (p->obj_size < stats->min_obj_size)
122 			stats->min_obj_size = p->obj_size;
123 		if (p->obj_size > stats->max_obj_size)
124 			stats->max_obj_size = p->obj_size;
125 
126 		p->nr_pages = p->nr_slabs * pages_per_slab;
127 
128 		if (p->nr_objs) {
129 			p->use = 100 * p->nr_active_objs / p->nr_objs;
130 			stats->nr_active_caches++;
131 		} else
132 			p->use = 0;
133 
134 		stats->nr_objs += p->nr_objs;
135 		stats->nr_active_objs += p->nr_active_objs;
136 		stats->total_size += p->nr_objs * p->obj_size;
137 		stats->active_size += p->nr_active_objs * p->obj_size;
138 		stats->nr_slabs += p->nr_slabs;
139 		stats->nr_active_slabs += nr_active_slabs;
140 
141 		prev->next = p;
142 		prev = p;
143 	}
144 
145 	if (fclose(slabfile))
146 		perror("fclose");
147 
148 	if (p)
149 		p->next = NULL;
150 	if (stats->nr_objs)
151 		stats->avg_obj_size = stats->total_size / stats->nr_objs;
152 
153 	return head;
154 }
155 
156 /*
157  * free_slablist - deallocate the memory associated with each node in the
158  * provided slab_info linked list
159  */
160 static void free_slablist(struct slab_info *list)
161 {
162 	while (list) {
163 		struct slab_info *temp = list->next;
164 		free(list);
165 		list = temp;
166 	}
167 }
168 
169 static struct slab_info *merge_objs(struct slab_info *a, struct slab_info *b)
170 {
171 	struct slab_info list;
172 	struct slab_info *p = &list;
173 
174 	while (a && b) {
175 		if (sort_func(a, b)) {
176 			p->next = a;
177 			p = a;
178 			a = a->next;
179 		} else {
180 			p->next = b;
181 			p = b;
182 			b = b->next;
183 		}
184 	}
185 
186 	p->next = (a == NULL) ? b : a;
187 	return list.next;
188 }
189 
190 /*
191  * slabsort - merge sort the slab_info linked list based on sort_func
192  */
193 static struct slab_info *slabsort(struct slab_info *list)
194 {
195 	struct slab_info *a, *b;
196 
197 	if (!list || !list->next)
198 		return list;
199 
200 	a = list;
201 	b = list->next;
202 
203 	while (b && b->next) {
204 		list = list->next;
205 		b = b->next->next;
206 	}
207 
208 	b = list->next;
209 	list->next = NULL;
210 
211 	return merge_objs(slabsort(a), slabsort(b));
212 }
213 
214 /*
215  * Sort Routines.  Each of these should be associated with a command-line
216  * search option.  The functions should fit the prototype:
217  *
218  *	int sort_foo(const struct slab_info *a, const struct slab_info *b)
219  *
220  * They return zero if the first parameter is smaller than the second.
221  * Otherwise, they return nonzero.
222  */
223 
224 static int sort_name(const struct slab_info *a, const struct slab_info *b)
225 {
226 	return (strcmp(a->name, b->name) < 0 ) ? 1: 0;
227 }
228 
229 #define BUILD_SORT_FUNC(VAL) \
230 	static int sort_ ## VAL \
231 		(const struct slab_info *a, const struct slab_info *b) { \
232 			return (a-> VAL > b-> VAL); }
233 
234 BUILD_SORT_FUNC(nr_objs)
235 BUILD_SORT_FUNC(nr_active_objs)
236 BUILD_SORT_FUNC(obj_size)
237 BUILD_SORT_FUNC(objs_per_slab)
238 BUILD_SORT_FUNC(nr_slabs)
239 BUILD_SORT_FUNC(use)
240 BUILD_SORT_FUNC(nr_pages)
241 
242 /*
243  * set_sort_func - return the slab_sort_func that matches the given key.
244  * On unrecognizable key, the call returns NULL.
245  */
246 static sort_t set_sort_func(char key)
247 {
248 	switch (tolower(key)) {
249 	case 'a':
250 		return sort_nr_active_objs;
251 	case 'c':
252 		return sort_nr_pages;
253 	case 'l':
254 		return sort_nr_slabs;
255 	case 'n':
256 		return sort_name;
257 	case 'o':
258 		return sort_nr_objs;
259 	case 'p':
260 		return sort_objs_per_slab;
261 	case 's':
262 		return sort_obj_size;
263 	case 'u':
264 		return sort_use;
265 	default:
266 		return NULL;
267 	}
268 }
269 
270 int main(int argc, char *argv[])
271 {
272 	struct slab_info *list, *p;
273 	struct slab_stat stats = { .nr_objs = 0 };
274 	unsigned int page_size = getpagesize() / 1024, nr_rows = DEF_NR_ROWS, i;
275 
276 	sort_func = DEF_SORT_FUNC;
277 
278 	if (argc > 1) {
279 		/* FIXME: Ugh. */
280 		if (argc == 3 && !strcmp(argv[1], "-n")) {
281 			errno = 0;
282 			nr_rows = (unsigned int) strtoul(argv[2], NULL, 0);
283 			if (errno) {
284 				perror("strtoul");
285 				exit(EXIT_FAILURE);
286 			}
287 		}
288 		else if (argc == 3 && !strcmp(argv[1], "-s"))
289 			sort_func = set_sort_func(argv[2][0]) ? : DEF_SORT_FUNC;
290 		else {
291 			fprintf(stderr, "usage: %s [options]\n\n", argv[0]);
292 			fprintf(stderr, "options:\n");
293 			fprintf(stderr, "  -s S   specify sort criteria S\n");
294 			fprintf(stderr, "  -h     display this help\n\n");
295 			fprintf(stderr, "Valid sort criteria:\n");
296 			fprintf(stderr, "  a: number of Active objects\n");
297 			fprintf(stderr, "  c: Cache size\n");
298 			fprintf(stderr, "  l: number of sLabs\n");
299 			fprintf(stderr, "  n: Name\n");
300 			fprintf(stderr, "  o: number of Objects\n");
301 			fprintf(stderr, "  p: objects Per slab\n");
302 			fprintf(stderr, "  s: object Size\n");
303 			fprintf(stderr, "  u: cache Utilization\n");
304 			exit(EXIT_FAILURE);
305 		}
306 	}
307 
308 	list = get_slabinfo (&stats);
309 	if (!list)
310 		exit(EXIT_FAILURE);
311 
312 	printf(" Active / Total Objects (%% used) : %lu / %lu (%.1f%%)\n"
313 	       " Active / Total Slabs (%% used)   : %lu / %lu (%.1f%%)\n"
314 	       " Active / Total Caches (%% used)  : %lu / %lu (%.1f%%)\n"
315 	       " Active / Total Size (%% used)    : %.2fK / %.2fK (%.1f%%)\n"
316 	       " Min / Avg / Max Object Size     : %.2fK / %.2fK / %.2fK\n\n",
317 	       stats.nr_active_objs,
318 	       stats.nr_objs,
319 	       100.0 * stats.nr_active_objs / stats.nr_objs,
320 	       stats.nr_active_slabs,
321 	       stats.nr_slabs,
322 	       100.0 * stats.nr_active_slabs / stats.nr_slabs,
323 	       stats.nr_active_caches,
324 	       stats.nr_caches,
325 	       100.0 * stats.nr_active_caches / stats.nr_caches,
326 	       stats.active_size / 1024.0,
327 	       stats.total_size / 1024.0,
328 	       100.0 * stats.active_size / stats.total_size,
329 	       stats.min_obj_size / 1024.0,
330 	       stats.avg_obj_size / 1024.0,
331 	       stats.max_obj_size / 1024.0);
332 
333 	printf("%6s %6s %4s %8s %6s %8s %10s %-23s\n",
334 	       "OBJS", "ACTIVE", "USE", "OBJ SIZE", "SLABS",
335 	       "OBJ/SLAB", "CACHE SIZE", "NAME");
336 
337 	p = list = slabsort(list);
338 	for (i = 0; i < nr_rows && p; i++) {
339 		printf("%6lu %6lu %3lu%% %7.2fK %6lu %8lu %9luK %-23s\n",
340 		       p->nr_objs, p->nr_active_objs, p->use,
341 		       p->obj_size / 1024.0, p->nr_slabs,
342 		       p->objs_per_slab,
343 		       p->nr_pages * page_size,
344 		       p->name);
345 		p = p->next;
346 	}
347 
348 	free_slablist(list);
349 
350 	return 0;
351 }
352