xref: /plan9-contrib/sys/src/cmd/kprof.c (revision 3f6badabfb9f09c44fc31ca998a77c56f0fe9ddb)
13e12c5d1SDavid du Colombier #include <u.h>
23e12c5d1SDavid du Colombier #include <libc.h>
33e12c5d1SDavid du Colombier #include <bio.h>
4bd389b36SDavid du Colombier #include <mach.h>
53e12c5d1SDavid du Colombier 
63e12c5d1SDavid du Colombier #define	PCRES	8
73e12c5d1SDavid du Colombier 
83e12c5d1SDavid du Colombier struct COUNTER
93e12c5d1SDavid du Colombier {
103e12c5d1SDavid du Colombier 	char 	*name;		/* function name */
113e12c5d1SDavid du Colombier 	long	time;		/* ticks spent there */
123e12c5d1SDavid du Colombier };
133e12c5d1SDavid du Colombier 
143e12c5d1SDavid du Colombier void
error(int perr,char * s)153e12c5d1SDavid du Colombier error(int perr, char *s)
163e12c5d1SDavid du Colombier {
173e12c5d1SDavid du Colombier 	fprint(2, "kprof: %s", s);
183e12c5d1SDavid du Colombier 	if(perr){
193e12c5d1SDavid du Colombier 		fprint(2, ": ");
203e12c5d1SDavid du Colombier 		perror(0);
213e12c5d1SDavid du Colombier 	}else
223e12c5d1SDavid du Colombier 		fprint(2, "\n");
233e12c5d1SDavid du Colombier 	exits(s);
243e12c5d1SDavid du Colombier }
253e12c5d1SDavid du Colombier 
263e12c5d1SDavid du Colombier int
compar(void * va,void * vb)277dd7cddfSDavid du Colombier compar(void *va, void *vb)
283e12c5d1SDavid du Colombier {
297dd7cddfSDavid du Colombier 	struct COUNTER *a, *b;
307dd7cddfSDavid du Colombier 
317dd7cddfSDavid du Colombier 	a = va;
327dd7cddfSDavid du Colombier 	b = vb;
333e12c5d1SDavid du Colombier 	if(a->time < b->time)
343e12c5d1SDavid du Colombier 		return -1;
353e12c5d1SDavid du Colombier 	if(a->time == b->time)
363e12c5d1SDavid du Colombier 		return 0;
373e12c5d1SDavid du Colombier 	return 1;
383e12c5d1SDavid du Colombier }
393e12c5d1SDavid du Colombier void
main(int argc,char * argv[])403e12c5d1SDavid du Colombier main(int argc, char *argv[])
413e12c5d1SDavid du Colombier {
423e12c5d1SDavid du Colombier 	int fd;
43*3f6badabSDavid du Colombier 	long delta, i, j, k, n;
44*3f6badabSDavid du Colombier 	ulong sum;
453e12c5d1SDavid du Colombier 	ulong *data;
463c917a9eSDavid du Colombier 	vlong tbase;
47*3f6badabSDavid du Colombier 	char *name;
483e12c5d1SDavid du Colombier 	Biobuf outbuf;
49*3f6badabSDavid du Colombier 	struct COUNTER *cp;
503e12c5d1SDavid du Colombier 	Fhdr f;
519a747e4fSDavid du Colombier 	Dir *d;
52*3f6badabSDavid du Colombier 	Symbol s;
533e12c5d1SDavid du Colombier 
543e12c5d1SDavid du Colombier 	if(argc != 3)
553e12c5d1SDavid du Colombier 		error(0, "usage: kprof text data");
563e12c5d1SDavid du Colombier 	/*
573e12c5d1SDavid du Colombier 	 * Read symbol table
583e12c5d1SDavid du Colombier 	 */
593e12c5d1SDavid du Colombier 	fd = open(argv[1], OREAD);
603e12c5d1SDavid du Colombier 	if(fd < 0)
613e12c5d1SDavid du Colombier 		error(1, argv[1]);
623e12c5d1SDavid du Colombier 	if (!crackhdr(fd, &f))
633e12c5d1SDavid du Colombier 		error(1, "read text header");
643e12c5d1SDavid du Colombier 	if (f.type == FNONE)
653e12c5d1SDavid du Colombier 		error(0, "text file not an a.out");
663e12c5d1SDavid du Colombier 	if (syminit(fd, &f) < 0)
67219b2ee8SDavid du Colombier 		error(1, "syminit");
683e12c5d1SDavid du Colombier 	close(fd);
693e12c5d1SDavid du Colombier 	/*
703e12c5d1SDavid du Colombier 	 * Read timing data
713e12c5d1SDavid du Colombier 	 */
723e12c5d1SDavid du Colombier 	fd = open(argv[2], OREAD);
733e12c5d1SDavid du Colombier 	if(fd < 0)
743e12c5d1SDavid du Colombier 		error(1, argv[2]);
759a747e4fSDavid du Colombier 	d = dirfstat(fd);
769a747e4fSDavid du Colombier 	if(d == nil)
773e12c5d1SDavid du Colombier 		error(1, "stat");
789a747e4fSDavid du Colombier 	n = d->length/sizeof(data[0]);
793e12c5d1SDavid du Colombier 	if(n < 2)
803e12c5d1SDavid du Colombier 		error(0, "data file too short");
819a747e4fSDavid du Colombier 	data = malloc(d->length);
823e12c5d1SDavid du Colombier 	if(data == 0)
833e12c5d1SDavid du Colombier 		error(1, "malloc");
849a747e4fSDavid du Colombier 	if(read(fd, data, d->length) < 0)
853e12c5d1SDavid du Colombier 		error(1, "text read");
863e12c5d1SDavid du Colombier 	close(fd);
873e12c5d1SDavid du Colombier 	for(i=0; i<n; i++)
883e12c5d1SDavid du Colombier 		data[i] = beswal(data[i]);
89*3f6badabSDavid du Colombier 
903e12c5d1SDavid du Colombier 	delta = data[0]-data[1];
913e12c5d1SDavid du Colombier 	print("total: %ld	in kernel text: %ld	outside kernel text: %ld\n",
923e12c5d1SDavid du Colombier 		data[0], delta, data[1]);
933e12c5d1SDavid du Colombier 	if(data[0] == 0)
943e12c5d1SDavid du Colombier 		exits(0);
95*3f6badabSDavid du Colombier 	/* get text symbol with lowest address */
963e12c5d1SDavid du Colombier 	if (!textsym(&s, 0))
973e12c5d1SDavid du Colombier 		error(0, "no text symbols");
9842860f21SDavid du Colombier 
9942860f21SDavid du Colombier 	tbase = mach->kbase;
100*3f6badabSDavid du Colombier  	if(tbase != (s.value & ~0xFFF)) {
10142860f21SDavid du Colombier 		print("warning: kbase %.8llux != tbase %.8llux\n",
10242860f21SDavid du Colombier 			tbase, s.value&~0xFFF);
103*3f6badabSDavid du Colombier 		tbase = s.value;
104*3f6badabSDavid du Colombier 	}
10542860f21SDavid du Colombier 	print("KTZERO %.8llux PGSIZE %dKb\n", tbase, mach->pgsize/1024);
1063e12c5d1SDavid du Colombier 	/*
1073e12c5d1SDavid du Colombier 	 * Accumulate counts for each function
1083e12c5d1SDavid du Colombier 	 */
1093e12c5d1SDavid du Colombier 	cp = 0;
1103e12c5d1SDavid du Colombier 	k = 0;
11142860f21SDavid du Colombier 	for (i = 0, j = 2; j < n; i++) {
1123e12c5d1SDavid du Colombier 		name = s.name;		/* save name */
1133e12c5d1SDavid du Colombier 		if (!textsym(&s, i))	/* get next symbol */
1143e12c5d1SDavid du Colombier 			break;
11542860f21SDavid du Colombier 		s.value -= tbase;
11642860f21SDavid du Colombier 		s.value /= PCRES;
1173e12c5d1SDavid du Colombier 		sum = 0;
11842860f21SDavid du Colombier 		while (j < n && j < s.value)
1193e12c5d1SDavid du Colombier 			sum += data[j++];
1203e12c5d1SDavid du Colombier 		if (sum) {
1213e12c5d1SDavid du Colombier 			cp = realloc(cp, (k+1)*sizeof(struct COUNTER));
1223e12c5d1SDavid du Colombier 			if (cp == 0)
1233e12c5d1SDavid du Colombier 				error(1, "realloc");
1243e12c5d1SDavid du Colombier 			cp[k].name = name;
1253e12c5d1SDavid du Colombier 			cp[k].time = sum;
1263e12c5d1SDavid du Colombier 			k++;
1273e12c5d1SDavid du Colombier 		}
1283e12c5d1SDavid du Colombier 	}
1293e12c5d1SDavid du Colombier 	if (!k)
1303e12c5d1SDavid du Colombier 		error(0, "no counts");
1313e12c5d1SDavid du Colombier 	cp[k].time = 0;			/* "etext" can take no time */
1323e12c5d1SDavid du Colombier 	/*
1333e12c5d1SDavid du Colombier 	 * Sort by time and print
1343e12c5d1SDavid du Colombier 	 */
1353e12c5d1SDavid du Colombier 	qsort(cp, k, sizeof(struct COUNTER), compar);
1363e12c5d1SDavid du Colombier 	Binit(&outbuf, 1, OWRITE);
1373e12c5d1SDavid du Colombier 	Bprint(&outbuf, "ms	  %%	sym\n");
1383e12c5d1SDavid du Colombier 	while(--k>=0)
1394e47a2baSDavid du Colombier 		Bprint(&outbuf, "%ld\t%3lld.%lld\t%s\n",
1403e12c5d1SDavid du Colombier 				cp[k].time,
1414e47a2baSDavid du Colombier 				100LL*cp[k].time/delta,
1424e47a2baSDavid du Colombier 				(1000LL*cp[k].time/delta)%10,
1433e12c5d1SDavid du Colombier 				cp[k].name);
1443e12c5d1SDavid du Colombier 	exits(0);
1453e12c5d1SDavid du Colombier }
146