git: 9front

ref: 7fded5d9435a01292e4187d949edbc45615cd770
dir: /sys/src/cmd/flambe.c/

View raw version
#include <u.h>
#include <libc.h>
#include <thread.h>
#include <draw.h>
#include <mouse.h>
#include <keyboard.h>
#include <bio.h>
#include <mach.h>
#include <plumb.h>

typedef struct Data	Data;

struct Data
{
	ushort	down;
	ushort	right;
	ulong	pc;
	ulong	count;
	uvlong	time;
	uint 	rec;
};
enum { Datasz = 2+2+4+4+8 };

Data*	data;
long	ndata;
uvlong	cyclefreq;

void
syms(char *cout)
{
	Fhdr f;
	int fd;

	if((fd = open(cout, 0)) < 0)
		sysfatal("%r");
	if(!crackhdr(fd, &f))
		sysfatal("can't read text file header: %r");
	if(f.type == FNONE)
		sysfatal("text file not an a.out");
	if(syminit(fd, &f) < 0)
		sysfatal("syminit: %r");
	close(fd);
}

#define GET2(p) (u16int)(p)[1] | (u16int)(p)[0]<<8
#define GET4(p) (u32int)(p)[3] | (u32int)(p)[2]<<8 | (u32int)(p)[1]<<16 | (u32int)(p)[0]<<24
#define GET8(p) (u64int)(p)[7] | (u64int)(p)[6]<<8 | (u64int)(p)[5]<<16 | (u64int)(p)[4]<<24 | \
		(u64int)(p)[3]<<32 | (u64int)(p)[2]<<40 | (u64int)(p)[1]<<48 | (u64int)(p)[0]<<56

void
datas(char *dout)
{
	int fd;
	Dir *d;
	int i;
	uchar hdr[3+1+8], *buf, *p;

	if((fd = open(dout, 0)) < 0){
		perror(dout);
		exits("open");
	}
	d = dirfstat(fd);
	if(d == nil){
		perror(dout);
		exits("stat");
	}
	d->length -= sizeof hdr;
	ndata = d->length/Datasz;
	data = malloc(ndata*sizeof(Data));
	buf = malloc(d->length);
	if(buf == 0 || data == 0)
		sysfatal("malloc");
	if(read(fd, hdr, sizeof hdr) != sizeof hdr)
		sysfatal("read data header: %r");
	if(memcmp(hdr, "pr\x0f", 3) != 0)
		sysfatal("bad magic");
	cyclefreq = GET8(hdr+4);
	if(readn(fd, buf, d->length) != d->length)
		sysfatal("data file read: %r");
	for(p = buf, i = 0; i < ndata; i++){
		data[i].down = GET2(p); p += 2;
		data[i].right = GET2(p); p += 2;
		data[i].pc = GET4(p); p += 4;
		data[i].count = GET4(p); p += 4;
		data[i].time = GET8(p); p += 8;
	}
	free(buf);
	free(d);
	close(fd);
}

char*
name(ulong pc)
{
	Symbol s;
	static char buf[16];

	if (findsym(pc, CTEXT, &s))
		return(s.name);
	snprint(buf, sizeof(buf), "#%lux", pc);
	return buf;
}

static void
plumbpc(ulong pc)
{
	Symbol s;
	char buf[256], wd[256];
	int fd;

	if(!findsym(pc, CTEXT, &s))
		return;
	
	fileline(buf, sizeof buf, pc);
	fd = plumbopen("send", OWRITE);
	if(fd < 0)
		return;
	getwd(wd, sizeof wd);
	plumbsendtext(fd, "flambe", "edit", wd, buf);
	close(fd);
}

int rowh;
Image **cols;
int ncols;
uvlong total;
Rectangle *clicks;

void
gencols(void)
{
	int i, h;
	ulong col, step;

	h = Dy(screen->r) / rowh + 1;
	if(ncols == h)
		return;
	for(i = 0; i < ncols; i++)
		freeimage(cols[i]);
	free(cols);

	ncols = h;
	cols = malloc(sizeof(Image*)*ncols);
	col = DRed;
	step = (0xFF/ncols)<<16;
	for(i = 0; i < ncols; i++){
		cols[i] = allocimagemix(display, DWhite, col);
		col += step;
	}
}

Image*
colfor(int h)
{
	if(h % 2 == 0)
		h += (ncols/3);
	return cols[h % ncols];
}

void
onhover(int i)
{
	Rectangle r;
	char buf[128];

	r = screen->r;
	r.max.y = r.min.y + (font->height+2)*2;
	draw(screen, r, display->white, nil, ZP);
	string(screen, r.min, display->black, ZP, font, name(data[i].pc));

	r.min.y += font->height + 1;
	snprint(buf, sizeof buf, "Time: %.8f(s) %.2f%%, Calls: %lud", (double)data[i].time/cyclefreq, (double)data[i].time/total * 100, data[i].count);
	string(screen, r.min, display->black, ZP, font, buf);
	flushimage(display, 1);
}

void
graph(int i, int x, int h)
{
	Rectangle r, r2;

	if(i >= ndata)
		sysfatal("corrupted profile data");
	r.min = (Point){x, screen->r.max.y - rowh*h};
	r.max = (Point){
		x + ((double)data[i].time * Dx(screen->r)) / total,
		r.min.y + rowh
	};
	clicks[i] = r;
	if(Dx(r) > 6){
		draw(screen, r, colfor(h), nil, ZP);
		r2 = r;
		r2.min.x  = r2.max.x - 2;
		draw(screen, r2, display->black, nil, ZP);
		screen->clipr = r;
		r2.min = r.min;
		r2.min.x += 4;
		string(screen, r2.min, display->black, ZP, font, name(data[i].pc));
		screen->clipr = screen->r;
	}
	if(data[i].right != 0xFFFF)
		graph(data[i].right, r.max.x, h);
	if(data[i].down != 0xFFFF)
		graph(data[i].down, x, h + 1);
}

void
redraw(int i)
{
	total = data[i].time;
	memset(clicks, 0, ndata * sizeof(Rectangle));
	gencols();
	draw(screen, screen->r, display->white, nil, ZP);
	graph(i, screen->r.min.x, 1);
	flushimage(display, 1);
}

enum
{
	Ckey,
	Cmouse,
	Cresize,
	Numchan,
};

void
usage(void)
{
	fprint(2, "%s: $O.out prof\n", argv0);
	exits("usage");
}

/* syminit has a Biobuf on the stack */
mainstacksize = 64*1024;

void
threadmain(int argc, char **argv)
{
	Mousectl *mctl;
	Keyboardctl *kctl;
	Mouse m;
	Rune r;
	vlong t;
	int i;
	Alt a[Numchan+1] = {
		[Ckey] = {nil, &r, CHANRCV},
		[Cmouse] = {nil, &m, CHANRCV },
		[Cresize] = {nil, nil, CHANRCV},
		{nil, nil, CHANEND},
	};

	ARGBEGIN{
	default:
		usage();
		break;
	}ARGEND;
	if(argc != 2)
		usage();
	syms(argv[0]);
	datas(argv[1]);
	for(int i = 1; i < ndata; i++){
		t = data[i].time;
		if(t < 0)
			data[i].time =  t + data[0].time;
	}

	if(initdraw(nil, nil, "flambe") < 0)
		sysfatal("initdraw: %r");
	if((kctl = initkeyboard(nil)) == nil)
		sysfatal("initkeyboard: %r");
	a[Ckey].c = kctl->c;
	if((mctl = initmouse(nil, screen)) == nil)
		sysfatal("initmouse: %r");
	a[Cmouse].c = mctl->c;
	a[Cresize].c = mctl->resizec;

	rowh = font->height + 2;
	clicks = malloc(sizeof(Rectangle)*ndata);

	redraw(1);
	for(;;)
	switch(alt(a)){
		case Ckey:
			switch(r){
				case Kesc:
					redraw(1);
					break;
				case Kdel:
				case 'q':
					threadexitsall(nil);
				default:
					break;
			}
			break;
		case Cmouse:
			if(m.buttons == 16){
				redraw(1);
				break;
			}
			for(i = 0; i < ndata; i++){
				if(!ptinrect(m.xy, clicks[i]))
					continue;
				switch(m.buttons){
				case 0:
					onhover(i);
					break;
				case 1: case 8:
					redraw(i);
					break;
				case 4:
					plumbpc(data[i].pc);
					break;
				}
				break;
			}
			break;
		case Cresize:
			getwindow(display, Refnone);
			redraw(1);
			break;
	}
}