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;
433e12c5d1SDavid du Colombier long i, j, k, n;
443e12c5d1SDavid du Colombier char *name;
453e12c5d1SDavid du Colombier ulong *data;
463c917a9eSDavid du Colombier vlong tbase;
473c917a9eSDavid du Colombier ulong sum;
483e12c5d1SDavid du Colombier long delta;
493e12c5d1SDavid du Colombier Symbol s;
503e12c5d1SDavid du Colombier Biobuf outbuf;
513e12c5d1SDavid du Colombier Fhdr f;
529a747e4fSDavid du Colombier Dir *d;
533e12c5d1SDavid du Colombier struct COUNTER *cp;
543e12c5d1SDavid du Colombier
553e12c5d1SDavid du Colombier if(argc != 3)
563e12c5d1SDavid du Colombier error(0, "usage: kprof text data");
573e12c5d1SDavid du Colombier /*
583e12c5d1SDavid du Colombier * Read symbol table
593e12c5d1SDavid du Colombier */
603e12c5d1SDavid du Colombier fd = open(argv[1], OREAD);
613e12c5d1SDavid du Colombier if(fd < 0)
623e12c5d1SDavid du Colombier error(1, argv[1]);
633e12c5d1SDavid du Colombier if (!crackhdr(fd, &f))
643e12c5d1SDavid du Colombier error(1, "read text header");
653e12c5d1SDavid du Colombier if (f.type == FNONE)
663e12c5d1SDavid du Colombier error(0, "text file not an a.out");
673e12c5d1SDavid du Colombier if (syminit(fd, &f) < 0)
68219b2ee8SDavid du Colombier error(1, "syminit");
693e12c5d1SDavid du Colombier close(fd);
703e12c5d1SDavid du Colombier /*
713e12c5d1SDavid du Colombier * Read timing data
723e12c5d1SDavid du Colombier */
733e12c5d1SDavid du Colombier fd = open(argv[2], OREAD);
743e12c5d1SDavid du Colombier if(fd < 0)
753e12c5d1SDavid du Colombier error(1, argv[2]);
769a747e4fSDavid du Colombier d = dirfstat(fd);
779a747e4fSDavid du Colombier if(d == nil)
783e12c5d1SDavid du Colombier error(1, "stat");
799a747e4fSDavid du Colombier n = d->length/sizeof(data[0]);
803e12c5d1SDavid du Colombier if(n < 2)
813e12c5d1SDavid du Colombier error(0, "data file too short");
829a747e4fSDavid du Colombier data = malloc(d->length);
833e12c5d1SDavid du Colombier if(data == 0)
843e12c5d1SDavid du Colombier error(1, "malloc");
859a747e4fSDavid du Colombier if(read(fd, data, d->length) < 0)
863e12c5d1SDavid du Colombier error(1, "text read");
873e12c5d1SDavid du Colombier close(fd);
883e12c5d1SDavid du Colombier for(i=0; i<n; i++)
893e12c5d1SDavid du Colombier data[i] = beswal(data[i]);
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);
953e12c5d1SDavid du Colombier if (!textsym(&s, 0))
963e12c5d1SDavid du Colombier error(0, "no text symbols");
97*42860f21SDavid du Colombier
98*42860f21SDavid du Colombier tbase = mach->kbase;
99*42860f21SDavid du Colombier if(tbase != s.value & ~0xFFF)
100*42860f21SDavid du Colombier print("warning: kbase %.8llux != tbase %.8llux\n",
101*42860f21SDavid du Colombier tbase, s.value&~0xFFF);
102*42860f21SDavid du Colombier print("KTZERO %.8llux PGSIZE %dKb\n", tbase, mach->pgsize/1024);
1033e12c5d1SDavid du Colombier /*
1043e12c5d1SDavid du Colombier * Accumulate counts for each function
1053e12c5d1SDavid du Colombier */
1063e12c5d1SDavid du Colombier cp = 0;
1073e12c5d1SDavid du Colombier k = 0;
108*42860f21SDavid du Colombier for (i = 0, j = 2; j < n; i++) {
1093e12c5d1SDavid du Colombier name = s.name; /* save name */
1103e12c5d1SDavid du Colombier if (!textsym(&s, i)) /* get next symbol */
1113e12c5d1SDavid du Colombier break;
112*42860f21SDavid du Colombier s.value -= tbase;
113*42860f21SDavid du Colombier s.value /= PCRES;
1143e12c5d1SDavid du Colombier sum = 0;
115*42860f21SDavid du Colombier while (j < n && j < s.value)
1163e12c5d1SDavid du Colombier sum += data[j++];
1173e12c5d1SDavid du Colombier if (sum) {
1183e12c5d1SDavid du Colombier cp = realloc(cp, (k+1)*sizeof(struct COUNTER));
1193e12c5d1SDavid du Colombier if (cp == 0)
1203e12c5d1SDavid du Colombier error(1, "realloc");
1213e12c5d1SDavid du Colombier cp[k].name = name;
1223e12c5d1SDavid du Colombier cp[k].time = sum;
1233e12c5d1SDavid du Colombier k++;
1243e12c5d1SDavid du Colombier }
1253e12c5d1SDavid du Colombier }
1263e12c5d1SDavid du Colombier if (!k)
1273e12c5d1SDavid du Colombier error(0, "no counts");
1283e12c5d1SDavid du Colombier cp[k].time = 0; /* "etext" can take no time */
1293e12c5d1SDavid du Colombier /*
1303e12c5d1SDavid du Colombier * Sort by time and print
1313e12c5d1SDavid du Colombier */
1323e12c5d1SDavid du Colombier qsort(cp, k, sizeof(struct COUNTER), compar);
1333e12c5d1SDavid du Colombier Binit(&outbuf, 1, OWRITE);
1343e12c5d1SDavid du Colombier Bprint(&outbuf, "ms %% sym\n");
1353e12c5d1SDavid du Colombier while(--k>=0)
1364e47a2baSDavid du Colombier Bprint(&outbuf, "%ld\t%3lld.%lld\t%s\n",
1373e12c5d1SDavid du Colombier cp[k].time,
1384e47a2baSDavid du Colombier 100LL*cp[k].time/delta,
1394e47a2baSDavid du Colombier (1000LL*cp[k].time/delta)%10,
1403e12c5d1SDavid du Colombier cp[k].name);
1413e12c5d1SDavid du Colombier exits(0);
1423e12c5d1SDavid du Colombier }
143