ref: babf901b4a508c3ec5d1f89655f10377bbdf9637
dir: /os/port/devkprof.c/
#include "u.h"
#include "../port/lib.h"
#include "mem.h"
#include "dat.h"
#include "fns.h"
#include "../port/error.h"
#ifndef LRES
#define LRES 3 /* log of PC resolution */
#endif
#define SZ 4 /* sizeof of count cell; well known as 4 */
enum {
SpecialTotalTicks,
SpecialOutsideTicks,
SpecialMicroSecondsPerTick,
SpecialSamples,
SpecialSampleSize,
SpecialSampleLogBucketSize,
SpecialMax
};
struct
{
int minpc;
int maxpc;
int nbuf;
int time;
ulong *buf;
}kprof;
enum{
Qdir,
Qdata,
Qctl,
Kprofmaxqid,
};
Dirtab kproftab[]={
".", {Qdir, 0, QTDIR}, 0, 0500,
"kpdata", {Qdata}, 0, 0600,
"kpctl", {Qctl}, 0, 0600,
};
void kproftimer(ulong);
void (*kproftick)(ulong);
static void
kprofinit(void)
{
if(SZ != sizeof kprof.buf[0])
panic("kprof size");
}
static void
kprofbufinit(void)
{
kprof.buf[SpecialMicroSecondsPerTick] = archkprofmicrosecondspertick();
kprof.buf[SpecialSamples] = kprof.nbuf;
kprof.buf[SpecialSampleSize] = SZ;
kprof.buf[SpecialSampleLogBucketSize] = LRES;
}
static Chan *
kprofattach(char *spec)
{
ulong n;
/* allocate when first used */
kproftick = kproftimer;
kprof.minpc = KTZERO;
kprof.maxpc = (ulong)etext;
kprof.nbuf = (kprof.maxpc-kprof.minpc) >> LRES;
n = kprof.nbuf*SZ;
if(kprof.buf == 0) {
kprof.buf = xalloc(n);
if(kprof.buf == 0)
error(Enomem);
}
kproftab[0].length = n;
kprofbufinit();
return devattach('K', spec);
}
static Walkqid*
kprofwalk(Chan *c, Chan *nc, char **name, int nname)
{
return devwalk(c, nc, name, nname, kproftab, nelem(kproftab), devgen);
}
static int
kprofstat(Chan *c, uchar *db, int n)
{
return devstat(c, db, n, kproftab, nelem(kproftab), devgen);
}
static Chan *
kprofopen(Chan *c, int omode)
{
if(c->qid.type & QTDIR){
if(omode != OREAD)
error(Eperm);
}
c->mode = openmode(omode);
c->flag |= COPEN;
c->offset = 0;
return c;
}
void
kprofclose(Chan*)
{
}
static long
kprofread(Chan *c, void *va, long n, vlong offset)
{
ulong tabend;
ulong w, *bp;
uchar *a, *ea;
switch((ulong)c->qid.path){
case Qdir:
return devdirread(c, va, n, kproftab, nelem(kproftab), devgen);
case Qdata:
tabend = kprof.nbuf*SZ;
if(offset & (SZ-1))
error(Ebadarg);
if(offset >= tabend){
n = 0;
break;
}
if(offset+n > tabend)
n = tabend-offset;
n &= ~(SZ-1);
a = va;
ea = a + n;
bp = kprof.buf + offset/SZ;
while(a < ea){
w = *bp++;
*a++ = w>>24;
*a++ = w>>16;
*a++ = w>>8;
*a++ = w>>0;
}
break;
default:
n = 0;
break;
}
return n;
}
static long
kprofwrite(Chan *c, void *vp, long n, vlong offset)
{
char *a;
USED(offset);
a = vp;
switch((ulong)c->qid.path){
case Qctl:
if(strncmp(a, "startclr", 8) == 0){
memset((char *)kprof.buf, 0, kprof.nbuf*SZ);
kprofbufinit();
archkprofenable(1);
kprof.time = 1;
}else if(strncmp(a, "start", 5) == 0) {
archkprofenable(1);
kprof.time = 1;
}
else if(strncmp(a, "stop", 4) == 0) {
archkprofenable(0);
kprof.time = 0;
}
else
error(Ebadctl);
break;
default:
error(Ebadusefd);
}
return n;
}
void
kproftimer(ulong pc)
{
extern void spldone(void);
if(kprof.time == 0)
return;
/*
* if the pc is coming out of spllo or splx,
* use the pc saved when we went splhi.
*/
// if(pc>=(ulong)splx && pc<=(ulong)spldone)
// pc = m->splpc;
kprof.buf[SpecialTotalTicks]++;
if(kprof.minpc + (SpecialMax << LRES) <= pc && pc < kprof.maxpc){
pc -= kprof.minpc;
pc >>= LRES;
kprof.buf[pc]++;
}else
kprof.buf[SpecialOutsideTicks]++;
}
Dev kprofdevtab = {
'K',
"kprof",
devreset,
kprofinit,
devshutdown,
kprofattach,
kprofwalk,
kprofstat,
kprofopen,
devcreate,
kprofclose,
kprofread,
devbread,
kprofwrite,
devbwrite,
devremove,
devwstat,
};