1/*-
2 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3 *
4 * Copyright (c) 2005-2007, Joseph Koshy
5 * Copyright (c) 2007 The FreeBSD Foundation
6 * All rights reserved.
7 *
8 * Portions of this software were developed by A. Joseph Koshy under
9 * sponsorship from the FreeBSD Foundation and Google, Inc.
10 *
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 *    notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 *    notice, this list of conditions and the following disclaimer in the
18 *    documentation and/or other materials provided with the distribution.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 * SUCH DAMAGE.
31 */
32
33/*
34 * Transform a hwpmc(4) log into human readable form, and into
35 * gprof(1) compatible profiles.
36 */
37
38#include <sys/cdefs.h>
39__FBSDID("$FreeBSD$");
40
41#include <sys/param.h>
42#include <sys/endian.h>
43#include <sys/gmon.h>
44#include <sys/imgact_aout.h>
45#include <sys/imgact_elf.h>
46#include <sys/mman.h>
47#include <sys/pmc.h>
48#include <sys/queue.h>
49#include <sys/socket.h>
50#include <sys/stat.h>
51#include <sys/wait.h>
52
53#include <netinet/in.h>
54
55#include <assert.h>
56#include <curses.h>
57#include <err.h>
58#include <errno.h>
59#include <fcntl.h>
60#include <gelf.h>
61#include <inttypes.h>
62#include <libgen.h>
63#include <limits.h>
64#include <netdb.h>
65#include <pmc.h>
66#include <pmclog.h>
67#include <sysexits.h>
68#include <stdint.h>
69#include <stdio.h>
70#include <stdlib.h>
71#include <string.h>
72#include <unistd.h>
73
74#include "pmcstat.h"
75#include "pmcstat_log.h"
76#include "pmcstat_top.h"
77#include "pmcpl_callgraph.h"
78
79#define	min(A,B)		((A) < (B) ? (A) : (B))
80#define	max(A,B)		((A) > (B) ? (A) : (B))
81
82/* Get the sample value in percent related to nsamples. */
83#define PMCPL_CG_COUNTP(a) \
84	((a)->pcg_count * 100.0 / nsamples)
85
86/*
87 * The toplevel CG nodes (i.e., with rank == 0) are placed in a hash table.
88 */
89
90struct pmcstat_cgnode_hash_list pmcstat_cgnode_hash[PMCSTAT_NHASH];
91int pmcstat_cgnode_hash_count;
92
93static pmcstat_interned_string pmcstat_previous_filename_printed;
94
95static struct pmcstat_cgnode *
96pmcstat_cgnode_allocate(struct pmcstat_image *image, uintfptr_t pc)
97{
98	struct pmcstat_cgnode *cg;
99
100	if ((cg = malloc(sizeof(*cg))) == NULL)
101		err(EX_OSERR, "ERROR: Cannot allocate callgraph node");
102
103	cg->pcg_image = image;
104	cg->pcg_func = pc;
105
106	cg->pcg_count = 0;
107	cg->pcg_nchildren = 0;
108	LIST_INIT(&cg->pcg_children);
109
110	return (cg);
111}
112
113/*
114 * Free a node and its children.
115 */
116static void
117pmcstat_cgnode_free(struct pmcstat_cgnode *cg)
118{
119	struct pmcstat_cgnode *cgc, *cgtmp;
120
121	LIST_FOREACH_SAFE(cgc, &cg->pcg_children, pcg_sibling, cgtmp)
122		pmcstat_cgnode_free(cgc);
123	free(cg);
124}
125
126/*
127 * Look for a callgraph node associated with pmc `pmcid' in the global
128 * hash table that corresponds to the given `pc' value in the process
129 * `pp'.
130 */
131static struct pmcstat_cgnode *
132pmcstat_cgnode_hash_lookup_pc(struct pmcstat_process *pp, pmc_id_t pmcid,
133    uintfptr_t pc, int usermode)
134{
135	struct pmcstat_pcmap *ppm;
136	struct pmcstat_symbol *sym;
137	struct pmcstat_image *image;
138	struct pmcstat_cgnode *cg;
139	struct pmcstat_cgnode_hash *h;
140	uintfptr_t loadaddress;
141	unsigned int i, hash;
142
143	ppm = pmcstat_process_find_map(usermode ? pp : pmcstat_kernproc, pc);
144	if (ppm == NULL)
145		return (NULL);
146
147	image = ppm->ppm_image;
148
149	loadaddress = ppm->ppm_lowpc + image->pi_vaddr - image->pi_start;
150	pc -= loadaddress;	/* Convert to an offset in the image. */
151
152	/*
153	 * Try determine the function at this offset.  If we can't
154	 * find a function round leave the `pc' value alone.
155	 */
156	if (!(args.pa_flags & (FLAG_SKIP_TOP_FN_RES | FLAG_SHOW_OFFSET))) {
157		if ((sym = pmcstat_symbol_search(image, pc)) != NULL)
158			pc = sym->ps_start;
159		else
160			pmcstat_stats.ps_samples_unknown_function++;
161	}
162
163	for (hash = i = 0; i < sizeof(uintfptr_t); i++)
164		hash += (pc >> i) & 0xFF;
165
166	hash &= PMCSTAT_HASH_MASK;
167
168	cg = NULL;
169	LIST_FOREACH(h, &pmcstat_cgnode_hash[hash], pch_next)
170	{
171		if (h->pch_pmcid != pmcid)
172			continue;
173
174		cg = h->pch_cgnode;
175
176		assert(cg != NULL);
177
178		if (cg->pcg_image == image && cg->pcg_func == pc)
179			return (cg);
180	}
181
182	/*
183	 * We haven't seen this (pmcid, pc) tuple yet, so allocate a
184	 * new callgraph node and a new hash table entry for it.
185	 */
186	cg = pmcstat_cgnode_allocate(image, pc);
187	if ((h = malloc(sizeof(*h))) == NULL)
188		err(EX_OSERR, "ERROR: Could not allocate callgraph node");
189
190	h->pch_pmcid = pmcid;
191	h->pch_cgnode = cg;
192	LIST_INSERT_HEAD(&pmcstat_cgnode_hash[hash], h, pch_next);
193
194	pmcstat_cgnode_hash_count++;
195
196	return (cg);
197}
198
199/*
200 * Compare two callgraph nodes for sorting.
201 */
202static int
203pmcstat_cgnode_compare(const void *a, const void *b)
204{
205	const struct pmcstat_cgnode *const *pcg1, *const *pcg2, *cg1, *cg2;
206
207	pcg1 = (const struct pmcstat_cgnode *const *) a;
208	cg1 = *pcg1;
209	pcg2 = (const struct pmcstat_cgnode *const *) b;
210	cg2 = *pcg2;
211
212	/* Sort in reverse order */
213	if (cg1->pcg_count < cg2->pcg_count)
214		return (1);
215	if (cg1->pcg_count > cg2->pcg_count)
216		return (-1);
217	return (0);
218}
219
220/*
221 * Find (allocating if a needed) a callgraph node in the given
222 * parent with the same (image, pcoffset) pair.
223 */
224
225static struct pmcstat_cgnode *
226pmcstat_cgnode_find(struct pmcstat_cgnode *parent, struct pmcstat_image *image,
227    uintfptr_t pcoffset)
228{
229	struct pmcstat_cgnode *child;
230
231	LIST_FOREACH(child, &parent->pcg_children, pcg_sibling) {
232		if (child->pcg_image == image &&
233		    child->pcg_func == pcoffset)
234			return (child);
235	}
236
237	/*
238	 * Allocate a new structure.
239	 */
240
241	child = pmcstat_cgnode_allocate(image, pcoffset);
242
243	/*
244	 * Link it into the parent.
245	 */
246	LIST_INSERT_HEAD(&parent->pcg_children, child, pcg_sibling);
247	parent->pcg_nchildren++;
248
249	return (child);
250}
251
252/*
253 * Print one callgraph node.  The output format is:
254 *
255 * indentation %(parent's samples) #nsamples function@object
256 */
257static void
258pmcstat_cgnode_print(struct pmcstat_cgnode *cg, int depth, uint32_t total)
259{
260	uint32_t n;
261	const char *space;
262	struct pmcstat_symbol *sym;
263	struct pmcstat_cgnode **sortbuffer, **cgn, *pcg;
264
265	space = " ";
266
267	if (depth > 0)
268		(void) fprintf(args.pa_graphfile, "%*s", depth, space);
269
270	if (cg->pcg_count == total)
271		(void) fprintf(args.pa_graphfile, "100.0%% ");
272	else
273		(void) fprintf(args.pa_graphfile, "%05.2f%% ",
274		    100.0 * cg->pcg_count / total);
275
276	n = fprintf(args.pa_graphfile, " [%u] ", cg->pcg_count);
277
278	/* #samples is a 12 character wide field. */
279	if (n < 12)
280		(void) fprintf(args.pa_graphfile, "%*s", 12 - n, space);
281
282	if (depth > 0)
283		(void) fprintf(args.pa_graphfile, "%*s", depth, space);
284
285	sym = pmcstat_symbol_search(cg->pcg_image, cg->pcg_func);
286	if (sym)
287		(void) fprintf(args.pa_graphfile, "%s",
288		    pmcstat_string_unintern(sym->ps_name));
289	else
290		(void) fprintf(args.pa_graphfile, "%p",
291		    (void *) (cg->pcg_image->pi_vaddr + cg->pcg_func));
292
293	if (pmcstat_previous_filename_printed !=
294	    cg->pcg_image->pi_fullpath) {
295		pmcstat_previous_filename_printed = cg->pcg_image->pi_fullpath;
296		(void) fprintf(args.pa_graphfile, " @ %s\n",
297		    pmcstat_string_unintern(
298		    pmcstat_previous_filename_printed));
299	} else
300		(void) fprintf(args.pa_graphfile, "\n");
301
302	if (cg->pcg_nchildren == 0)
303		return;
304
305	if ((sortbuffer = (struct pmcstat_cgnode **)
306		malloc(sizeof(struct pmcstat_cgnode *) *
307		    cg->pcg_nchildren)) == NULL)
308		err(EX_OSERR, "ERROR: Cannot print callgraph");
309	cgn = sortbuffer;
310
311	LIST_FOREACH(pcg, &cg->pcg_children, pcg_sibling)
312	    *cgn++ = pcg;
313
314	assert(cgn - sortbuffer == (int) cg->pcg_nchildren);
315
316	qsort(sortbuffer, cg->pcg_nchildren, sizeof(struct pmcstat_cgnode *),
317	    pmcstat_cgnode_compare);
318
319	for (cgn = sortbuffer, n = 0; n < cg->pcg_nchildren; n++, cgn++)
320		pmcstat_cgnode_print(*cgn, depth+1, cg->pcg_count);
321
322	free(sortbuffer);
323}
324
325/*
326 * Record a callchain.
327 */
328
329void
330pmcpl_cg_process(struct pmcstat_process *pp, struct pmcstat_pmcrecord *pmcr,
331    uint32_t nsamples, uintfptr_t *cc, int usermode, uint32_t cpu)
332{
333	uintfptr_t pc, loadaddress;
334	uint32_t n;
335	struct pmcstat_image *image;
336	struct pmcstat_pcmap *ppm;
337	struct pmcstat_symbol *sym;
338	struct pmcstat_cgnode *parent, *child;
339	struct pmcstat_process *km;
340	pmc_id_t pmcid;
341
342	(void) cpu;
343
344	/*
345	 * Find the callgraph node recorded in the global hash table
346	 * for this (pmcid, pc).
347	 */
348
349	pc = cc[0];
350	pmcid = pmcr->pr_pmcid;
351	child = parent = pmcstat_cgnode_hash_lookup_pc(pp, pmcid, pc, usermode);
352	if (parent == NULL) {
353		pmcstat_stats.ps_callchain_dubious_frames++;
354		pmcr->pr_dubious_frames++;
355		return;
356	}
357
358	parent->pcg_count++;
359
360	/*
361	 * For each return address in the call chain record, subject
362	 * to the maximum depth desired.
363	 * - Find the image associated with the sample.  Stop if there
364	 *   there is no valid image at that address.
365	 * - Find the function that overlaps the return address.
366	 * - If found: use the start address of the function.
367	 *   If not found (say an object's symbol table is not present or
368	 *   is incomplete), round down to th gprof bucket granularity.
369	 * - Convert return virtual address to an offset in the image.
370	 * - Look for a child with the same {offset,image} tuple,
371	 *   inserting one if needed.
372	 * - Increment the count of occurrences of the child.
373	 */
374	km = pmcstat_kernproc;
375
376	for (n = 1; n < (uint32_t) args.pa_graphdepth && n < nsamples; n++,
377	    parent = child) {
378		pc = cc[n];
379
380		ppm = pmcstat_process_find_map(usermode ? pp : km, pc);
381		if (ppm == NULL) {
382			/* Detect full frame capture (kernel + user). */
383			if (!usermode) {
384				ppm = pmcstat_process_find_map(pp, pc);
385				if (ppm != NULL)
386					km = pp;
387			}
388		}
389		if (ppm == NULL)
390			continue;
391
392		image = ppm->ppm_image;
393		loadaddress = ppm->ppm_lowpc + image->pi_vaddr -
394		    image->pi_start;
395		pc -= loadaddress;
396
397		if ((sym = pmcstat_symbol_search(image, pc)) != NULL)
398			pc = sym->ps_start;
399
400		child = pmcstat_cgnode_find(parent, image, pc);
401		child->pcg_count++;
402	}
403}
404
405/*
406 * Printing a callgraph for a PMC.
407 */
408static void
409pmcstat_callgraph_print_for_pmcid(struct pmcstat_pmcrecord *pmcr)
410{
411	int n, nentries;
412	uint32_t nsamples;
413	pmc_id_t pmcid;
414	struct pmcstat_cgnode **sortbuffer, **cgn;
415	struct pmcstat_cgnode_hash *pch;
416
417	/*
418	 * We pull out all callgraph nodes in the top-level hash table
419	 * with a matching PMC id.  We then sort these based on the
420	 * frequency of occurrence.  Each callgraph node is then
421	 * printed.
422	 */
423
424	nsamples = 0;
425	pmcid = pmcr->pr_pmcid;
426	if ((sortbuffer = (struct pmcstat_cgnode **)
427	    malloc(sizeof(struct pmcstat_cgnode *) *
428	    pmcstat_cgnode_hash_count)) == NULL)
429		err(EX_OSERR, "ERROR: Cannot sort callgraph");
430	cgn = sortbuffer;
431
432	for (n = 0; n < PMCSTAT_NHASH; n++)
433		LIST_FOREACH(pch, &pmcstat_cgnode_hash[n], pch_next)
434		    if (pch->pch_pmcid == pmcid) {
435			    nsamples += pch->pch_cgnode->pcg_count;
436			    *cgn++ = pch->pch_cgnode;
437		    }
438
439	nentries = cgn - sortbuffer;
440	assert(nentries <= pmcstat_cgnode_hash_count);
441
442	if (nentries == 0) {
443		free(sortbuffer);
444		return;
445	}
446
447	qsort(sortbuffer, nentries, sizeof(struct pmcstat_cgnode *),
448	    pmcstat_cgnode_compare);
449
450	(void) fprintf(args.pa_graphfile,
451	    "@ %s [%u samples]\n\n",
452	    pmcstat_string_unintern(pmcr->pr_pmcname),
453	    nsamples);
454
455	for (cgn = sortbuffer, n = 0; n < nentries; n++, cgn++) {
456		pmcstat_previous_filename_printed = NULL;
457		pmcstat_cgnode_print(*cgn, 0, nsamples);
458		(void) fprintf(args.pa_graphfile, "\n");
459	}
460
461	free(sortbuffer);
462}
463
464/*
465 * Print out callgraphs.
466 */
467
468static void
469pmcstat_callgraph_print(void)
470{
471	struct pmcstat_pmcrecord *pmcr;
472
473	LIST_FOREACH(pmcr, &pmcstat_pmcs, pr_next)
474	    pmcstat_callgraph_print_for_pmcid(pmcr);
475}
476
477static void
478pmcstat_cgnode_topprint(struct pmcstat_cgnode *cg,
479    int depth __unused, uint32_t nsamples)
480{
481	int v_attrs, vs_len, ns_len, width, len, n, nchildren;
482	float v;
483	char ns[30], vs[10];
484	struct pmcstat_symbol *sym;
485	struct pmcstat_cgnode **sortbuffer, **cgn, *pcg;
486
487	/* Format value. */
488	v = PMCPL_CG_COUNTP(cg);
489	snprintf(vs, sizeof(vs), "%.1f", v);
490	v_attrs = PMCSTAT_ATTRPERCENT(v);
491
492	/* Format name. */
493	sym = pmcstat_symbol_search(cg->pcg_image, cg->pcg_func);
494	if (sym == NULL) {
495		snprintf(ns, sizeof(ns), "%p",
496		    (void *)(cg->pcg_image->pi_vaddr + cg->pcg_func));
497	} else {
498		switch (args.pa_flags & (FLAG_SKIP_TOP_FN_RES | FLAG_SHOW_OFFSET)) {
499		case FLAG_SKIP_TOP_FN_RES | FLAG_SHOW_OFFSET:
500		case FLAG_SKIP_TOP_FN_RES:
501			snprintf(ns, sizeof(ns), "%p",
502			    (void *)(cg->pcg_image->pi_vaddr + cg->pcg_func));
503			break;
504		case FLAG_SHOW_OFFSET:
505			snprintf(ns, sizeof(ns), "%s+%#0" PRIx64,
506			    pmcstat_string_unintern(sym->ps_name),
507			    cg->pcg_func - sym->ps_start);
508			break;
509		default:
510			snprintf(ns, sizeof(ns), "%s",
511			    pmcstat_string_unintern(sym->ps_name));
512			break;
513		}
514	}
515
516	PMCSTAT_ATTRON(v_attrs);
517	PMCSTAT_PRINTW("%5.5s", vs);
518	PMCSTAT_ATTROFF(v_attrs);
519	PMCSTAT_PRINTW(" %-10.10s %-30.30s",
520	    pmcstat_string_unintern(cg->pcg_image->pi_name),
521	    ns);
522
523	nchildren = cg->pcg_nchildren;
524	if (nchildren == 0) {
525		PMCSTAT_PRINTW("\n");
526		return;
527	}
528
529	width = pmcstat_displaywidth - 40;
530
531	if ((sortbuffer = (struct pmcstat_cgnode **)
532		malloc(sizeof(struct pmcstat_cgnode *) *
533		    nchildren)) == NULL)
534		err(EX_OSERR, "ERROR: Cannot print callgraph");
535	cgn = sortbuffer;
536
537	LIST_FOREACH(pcg, &cg->pcg_children, pcg_sibling)
538	    *cgn++ = pcg;
539
540	assert(cgn - sortbuffer == (int)nchildren);
541
542	qsort(sortbuffer, nchildren, sizeof(struct pmcstat_cgnode *),
543	    pmcstat_cgnode_compare);
544
545	/* Count how many callers. */
546	for (cgn = sortbuffer, n = 0; n < nchildren; n++, cgn++) {
547		pcg = *cgn;
548
549		v = PMCPL_CG_COUNTP(pcg);
550		if (v < pmcstat_threshold)
551			break;
552	}
553	nchildren = n;
554
555	for (cgn = sortbuffer, n = 0; n < nchildren; n++, cgn++) {
556		pcg = *cgn;
557
558		/* Format value. */
559		if (nchildren > 1) {
560			v = PMCPL_CG_COUNTP(pcg);
561			vs_len = snprintf(vs, sizeof(vs), ":%.1f", v);
562			v_attrs = PMCSTAT_ATTRPERCENT(v);
563		} else
564			vs_len = 0;
565
566		/* Format name. */
567		sym = pmcstat_symbol_search(pcg->pcg_image, pcg->pcg_func);
568		if (sym != NULL) {
569			ns_len = snprintf(ns, sizeof(ns), "%s",
570			    pmcstat_string_unintern(sym->ps_name));
571		} else
572			ns_len = snprintf(ns, sizeof(ns), "%p",
573			    (void *)pcg->pcg_func);
574
575		len = ns_len + vs_len + 1;
576		if (width - len < 0) {
577			PMCSTAT_PRINTW(" ...");
578			break;
579		}
580		width -= len;
581
582		PMCSTAT_PRINTW(" %s", ns);
583		if (nchildren > 1) {
584			PMCSTAT_ATTRON(v_attrs);
585			PMCSTAT_PRINTW("%s", vs);
586			PMCSTAT_ATTROFF(v_attrs);
587		}
588	}
589	PMCSTAT_PRINTW("\n");
590	free(sortbuffer);
591}
592
593/*
594 * Top mode display.
595 */
596
597void
598pmcpl_cg_topdisplay(void)
599{
600	int n, nentries;
601	uint32_t nsamples;
602	struct pmcstat_cgnode **sortbuffer, **cgn;
603	struct pmcstat_cgnode_hash *pch;
604	struct pmcstat_pmcrecord *pmcr;
605
606	pmcr = pmcstat_pmcindex_to_pmcr(pmcstat_pmcinfilter);
607	if (!pmcr)
608		err(EX_SOFTWARE, "ERROR: invalid pmcindex");
609
610	/*
611	 * We pull out all callgraph nodes in the top-level hash table
612	 * with a matching PMC index.  We then sort these based on the
613	 * frequency of occurrence.  Each callgraph node is then
614	 * printed.
615	 */
616
617	nsamples = 0;
618
619	if ((sortbuffer = (struct pmcstat_cgnode **)
620	    malloc(sizeof(struct pmcstat_cgnode *) *
621	    pmcstat_cgnode_hash_count)) == NULL)
622		err(EX_OSERR, "ERROR: Cannot sort callgraph");
623	cgn = sortbuffer;
624
625	for (n = 0; n < PMCSTAT_NHASH; n++)
626		LIST_FOREACH(pch, &pmcstat_cgnode_hash[n], pch_next)
627		    if (pmcr == NULL || pch->pch_pmcid == pmcr->pr_pmcid) {
628			    nsamples += pch->pch_cgnode->pcg_count;
629			    *cgn++ = pch->pch_cgnode;
630		    }
631
632	nentries = cgn - sortbuffer;
633	assert(nentries <= pmcstat_cgnode_hash_count);
634
635	if (nentries == 0) {
636		free(sortbuffer);
637		return;
638	}
639
640	qsort(sortbuffer, nentries, sizeof(struct pmcstat_cgnode *),
641	    pmcstat_cgnode_compare);
642
643	PMCSTAT_PRINTW("%5.5s %-10.10s %-30.30s %s\n",
644	    "%SAMP", "IMAGE", "FUNCTION", "CALLERS");
645
646	nentries = min(pmcstat_displayheight - 2, nentries);
647
648	for (cgn = sortbuffer, n = 0; n < nentries; n++, cgn++) {
649		if (PMCPL_CG_COUNTP(*cgn) < pmcstat_threshold)
650			break;
651		pmcstat_cgnode_topprint(*cgn, 0, nsamples);
652	}
653
654	free(sortbuffer);
655}
656
657/*
658 * Handle top mode keypress.
659 */
660
661int
662pmcpl_cg_topkeypress(int c, void *arg)
663{
664	WINDOW *w;
665
666	w = (WINDOW *)arg;
667
668	(void) c; (void) w;
669
670	return 0;
671}
672
673int
674pmcpl_cg_init(void)
675{
676	int i;
677
678	pmcstat_cgnode_hash_count = 0;
679	pmcstat_previous_filename_printed = NULL;
680
681	for (i = 0; i < PMCSTAT_NHASH; i++) {
682		LIST_INIT(&pmcstat_cgnode_hash[i]);
683	}
684
685	return (0);
686}
687
688void
689pmcpl_cg_shutdown(FILE *mf)
690{
691	int i;
692	struct pmcstat_cgnode_hash *pch, *pchtmp;
693
694	(void) mf;
695
696	if (args.pa_flags & FLAG_DO_CALLGRAPHS)
697		pmcstat_callgraph_print();
698
699	/*
700	 * Free memory.
701	 */
702	for (i = 0; i < PMCSTAT_NHASH; i++) {
703		LIST_FOREACH_SAFE(pch, &pmcstat_cgnode_hash[i], pch_next,
704		    pchtmp) {
705			pmcstat_cgnode_free(pch->pch_cgnode);
706			LIST_REMOVE(pch, pch_next);
707			free(pch);
708		}
709	}
710}
711
712