code: 9ferno

ref: 83246e296ea433b65b9d295b5e08fedd39ff1ab7
dir: /os/port/devkprof.c/

View raw version
#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((u64)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((u64)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,
};