17dd7cddfSDavid du Colombier #include <u.h>
27dd7cddfSDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <bio.h>
47dd7cddfSDavid du Colombier #include <mach.h>
57dd7cddfSDavid du Colombier
67dd7cddfSDavid du Colombier #define PCRES 8
77dd7cddfSDavid du Colombier
87dd7cddfSDavid du Colombier struct COUNTER
97dd7cddfSDavid du Colombier {
107dd7cddfSDavid du Colombier char *name; /* function name */
117dd7cddfSDavid du Colombier long time; /* ticks spent there */
127dd7cddfSDavid du Colombier };
137dd7cddfSDavid du Colombier
147dd7cddfSDavid du Colombier void
error(int perr,char * s)157dd7cddfSDavid du Colombier error(int perr, char *s)
167dd7cddfSDavid du Colombier {
177dd7cddfSDavid du Colombier fprint(2, "tprof: %s", s);
187dd7cddfSDavid du Colombier if(perr){
197dd7cddfSDavid du Colombier fprint(2, ": ");
207dd7cddfSDavid du Colombier perror(0);
217dd7cddfSDavid du Colombier }else
227dd7cddfSDavid du Colombier fprint(2, "\n");
237dd7cddfSDavid du Colombier exits(s);
247dd7cddfSDavid du Colombier }
257dd7cddfSDavid du Colombier
267dd7cddfSDavid du Colombier int
compar(void * va,void * vb)277dd7cddfSDavid du Colombier compar(void *va, void *vb)
287dd7cddfSDavid du Colombier {
297dd7cddfSDavid du Colombier struct COUNTER *a, *b;
307dd7cddfSDavid du Colombier
317dd7cddfSDavid du Colombier a = va;
327dd7cddfSDavid du Colombier b = vb;
337dd7cddfSDavid du Colombier if(a->time < b->time)
347dd7cddfSDavid du Colombier return -1;
357dd7cddfSDavid du Colombier if(a->time == b->time)
367dd7cddfSDavid du Colombier return 0;
377dd7cddfSDavid du Colombier return 1;
387dd7cddfSDavid du Colombier }
397dd7cddfSDavid du Colombier void
main(int argc,char * argv[])407dd7cddfSDavid du Colombier main(int argc, char *argv[])
417dd7cddfSDavid du Colombier {
427dd7cddfSDavid du Colombier int fd;
437dd7cddfSDavid du Colombier long i, j, k, n;
449a747e4fSDavid du Colombier Dir *d;
457dd7cddfSDavid du Colombier char *name;
467dd7cddfSDavid du Colombier ulong *data;
477dd7cddfSDavid du Colombier ulong tbase, sum;
487dd7cddfSDavid du Colombier long delta;
497dd7cddfSDavid du Colombier Symbol s;
507dd7cddfSDavid du Colombier Biobuf outbuf;
517dd7cddfSDavid du Colombier Fhdr f;
527dd7cddfSDavid du Colombier struct COUNTER *cp;
5367031067SDavid du Colombier char filebuf[128], *file;
547dd7cddfSDavid du Colombier
5567031067SDavid du Colombier if(argc != 2 && argc != 3)
5667031067SDavid du Colombier error(0, "usage: tprof pid [binary]");
577dd7cddfSDavid du Colombier /*
587dd7cddfSDavid du Colombier * Read symbol table
597dd7cddfSDavid du Colombier */
6067031067SDavid du Colombier if(argc == 2){
6167031067SDavid du Colombier file = filebuf;
6267031067SDavid du Colombier snprint(filebuf, sizeof filebuf, "/proc/%s/text", argv[1]);
6367031067SDavid du Colombier }else
6467031067SDavid du Colombier file = argv[2];
6567031067SDavid du Colombier
667dd7cddfSDavid du Colombier fd = open(file, OREAD);
677dd7cddfSDavid du Colombier if(fd < 0)
687dd7cddfSDavid du Colombier error(1, file);
697dd7cddfSDavid du Colombier
707dd7cddfSDavid du Colombier if (!crackhdr(fd, &f))
717dd7cddfSDavid du Colombier error(1, "read text header");
727dd7cddfSDavid du Colombier if (f.type == FNONE)
737dd7cddfSDavid du Colombier error(0, "text file not an a.out");
747dd7cddfSDavid du Colombier machbytype(f.type);
757dd7cddfSDavid du Colombier if (syminit(fd, &f) < 0)
767dd7cddfSDavid du Colombier error(1, "syminit");
777dd7cddfSDavid du Colombier close(fd);
787dd7cddfSDavid du Colombier /*
797dd7cddfSDavid du Colombier * Read timing data
807dd7cddfSDavid du Colombier */
8116941224SDavid du Colombier file = smprint("/proc/%s/profile", argv[1]);
827dd7cddfSDavid du Colombier fd = open(file, OREAD);
837dd7cddfSDavid du Colombier if(fd < 0)
847dd7cddfSDavid du Colombier error(1, file);
8516941224SDavid du Colombier free(file);
869a747e4fSDavid du Colombier d = dirfstat(fd);
879a747e4fSDavid du Colombier if(d == nil)
887dd7cddfSDavid du Colombier error(1, "stat");
899a747e4fSDavid du Colombier n = d->length/sizeof(data[0]);
907dd7cddfSDavid du Colombier if(n < 2)
917dd7cddfSDavid du Colombier error(0, "data file too short");
929a747e4fSDavid du Colombier data = malloc(d->length);
937dd7cddfSDavid du Colombier if(data == 0)
947dd7cddfSDavid du Colombier error(1, "malloc");
959a747e4fSDavid du Colombier if(read(fd, data, d->length) < 0)
967dd7cddfSDavid du Colombier error(1, "text read");
977dd7cddfSDavid du Colombier close(fd);
987dd7cddfSDavid du Colombier
997dd7cddfSDavid du Colombier for(i=0; i<n; i++)
1007dd7cddfSDavid du Colombier data[i] = machdata->swal(data[i]);
1017dd7cddfSDavid du Colombier
1027dd7cddfSDavid du Colombier delta = data[0]-data[1];
1037dd7cddfSDavid du Colombier print("total: %ld\n", data[0]);
1047dd7cddfSDavid du Colombier if(data[0] == 0)
1057dd7cddfSDavid du Colombier exits(0);
1067dd7cddfSDavid du Colombier if (!textsym(&s, 0))
1077dd7cddfSDavid du Colombier error(0, "no text symbols");
1087dd7cddfSDavid du Colombier tbase = s.value & ~(mach->pgsize-1); /* align down to page */
1097dd7cddfSDavid du Colombier print("TEXT %.8lux\n", tbase);
1107dd7cddfSDavid du Colombier /*
1117dd7cddfSDavid du Colombier * Accumulate counts for each function
1127dd7cddfSDavid du Colombier */
1137dd7cddfSDavid du Colombier cp = 0;
1147dd7cddfSDavid du Colombier k = 0;
1157dd7cddfSDavid du Colombier for (i = 0, j = (s.value-tbase)/PCRES+2; j < n; i++) {
1167dd7cddfSDavid du Colombier name = s.name; /* save name */
1177dd7cddfSDavid du Colombier if (!textsym(&s, i)) /* get next symbol */
1187dd7cddfSDavid du Colombier break;
1197dd7cddfSDavid du Colombier sum = 0;
1207dd7cddfSDavid du Colombier while (j < n && j*PCRES < s.value-tbase)
1217dd7cddfSDavid du Colombier sum += data[j++];
1227dd7cddfSDavid du Colombier if (sum) {
1237dd7cddfSDavid du Colombier cp = realloc(cp, (k+1)*sizeof(struct COUNTER));
1247dd7cddfSDavid du Colombier if (cp == 0)
1257dd7cddfSDavid du Colombier error(1, "realloc");
1267dd7cddfSDavid du Colombier cp[k].name = name;
1277dd7cddfSDavid du Colombier cp[k].time = sum;
1287dd7cddfSDavid du Colombier k++;
1297dd7cddfSDavid du Colombier }
1307dd7cddfSDavid du Colombier }
1317dd7cddfSDavid du Colombier if (!k)
1327dd7cddfSDavid du Colombier error(0, "no counts");
1337dd7cddfSDavid du Colombier cp[k].time = 0; /* "etext" can take no time */
1347dd7cddfSDavid du Colombier /*
1357dd7cddfSDavid du Colombier * Sort by time and print
1367dd7cddfSDavid du Colombier */
1377dd7cddfSDavid du Colombier qsort(cp, k, sizeof(struct COUNTER), compar);
1387dd7cddfSDavid du Colombier Binit(&outbuf, 1, OWRITE);
1397dd7cddfSDavid du Colombier Bprint(&outbuf, " ms %% sym\n");
1407dd7cddfSDavid du Colombier while(--k>=0)
141*4e47a2baSDavid du Colombier Bprint(&outbuf, "%6ld\t%3lld.%lld\t%s\n",
1427dd7cddfSDavid du Colombier cp[k].time,
143*4e47a2baSDavid du Colombier 100LL*cp[k].time/delta,
144*4e47a2baSDavid du Colombier (1000LL*cp[k].time/delta)%10,
1457dd7cddfSDavid du Colombier cp[k].name);
1467dd7cddfSDavid du Colombier exits(0);
1477dd7cddfSDavid du Colombier }
148