shithub: wl3d

ref: 3891487cd9fb5088d99344d33cf4e37e0abf7f9b
dir: /wl3d.c/

View raw version
#include <u.h>
#include <libc.h>
#include <thread.h>
#include <draw.h>
#include <mouse.h>
#include <keyboard.h>
#include "dat.h"
#include "fns.h"

mainstacksize = 16*1024;
char *ext = "wl6";
int grabon;
int cson, kbon, mson;
int (*step)(void);
Channel *csc, *kbc, *msc;

enum{
	BILLION = 1000000000,
	MILLION = 1000000
};
static Point p0;
static Rectangle fbr, grabr;
static Image *fb;
static Channel *reszc;

static void
mproc(void *)
{
	int n, fd, nerr;
	char buf[1+5*12], *px, *py, *pb;
	Point o, p;
	Mouse m;

	fd = open("/dev/mouse", ORDWR);
	if(fd < 0)
		sysfatal("mproc: %r");
	nerr = 0;
	px = buf+1;
	py = px + 12;
	pb = py + 12;
	o = p0;
	for(;;){
		n = read(fd, buf, sizeof buf);
		if(n != 1+4*12){
			if(n < 0 || ++nerr > 10)
				break;
			fprint(2, "mproc: bad count %d not 49: %r\n", n);
			continue;
		}
		nerr = 0;
		switch(*buf){
		case 'r':
			send(reszc, nil);
			/* wet floor */
		case 'm':
			if(!mson)
				break;
			p.x = strtol(px, nil, 10);
			p.y = strtol(py, nil, 10);
			m.xy.x = p.x - o.x;
			m.xy.y = o.y - p.y;
			m.buttons = *pb;
			nbsend(msc, &m);
			if(!ptinrect(p, grabr)){
				fprint(fd, "m%d %d", p0.x, p0.y);
				p = p0;
			}
			o = p;
		}
	}
}

static void
kproc(void *)
{
	int n, k, fd;
	char c, buf[256], *s;
	Rune r, *a;

	fd = open("/dev/kbd", OREAD);
	if(fd < 0)
		sysfatal("kproc: %r");
	memset(buf, 0, sizeof buf);
	for(;;){
		n = read(fd, buf, sizeof(buf)-1);
		if(n <= 0)
			break;
		c = *buf;
		if(c == 'c' && cson){
			chartorune(&r, buf+1);
			send(csc, &r);
		}
		if(c != 'k' || c != 'K' || !kbon)
			continue;
		s = buf+1;
		k = 0;
		while(*s != 0){
			s += chartorune(&r, s);
			for(a=keys; a<keys+Ke; a++)
				if(r == *a){
					k |= 1<<a-keys;
					break;
				}
		}
		send(kbc, &k);
	}
}

static void
resetfb(void)
{
	Point p, d;

	scale = Dx(screen->r) / Vw;
	if(scale <= 0)
		scale = 1;
	else if(scale > 10)
		scale = 10;
	p = divpt(addpt(screen->r.min, screen->r.max), 2);
	d = Pt(Vw/2 * scale, Vh/2 * scale);
	fbr = Rpt(subpt(p, d), addpt(p, d));
	d = Pt(Vh/4, Vh/4);
	grabr = Rpt(subpt(p, d), addpt(p, d));
	p0 = p;

	freeimage(fb);
	free(px);
	npx = Vt * scale;
	px = emalloc(npx);
	fb = allocimage(display, Rect(0,0,Vw*scale,scale==1 ? Vh : 1), RGB24, 1, 0);
	if(fb == nil)
		sysfatal("resetfb: %r");

	draw(screen, screen->r, display->black, nil, ZP);
	out();
}

static void
usage(void)
{
	fprint(2, "usage: %s [-23dios] [-m dir] [-w map] [-x difficulty]\n", argv0);
	threadexits("usage");
}

void *
emalloc(ulong n)
{
	void *p;

	p = mallocz(n, 1);
	if(p == nil)
		sysfatal("emalloc: %r");
	return p;
}

/* use only for shortening buffers, no zeroing done */
void *
erealloc(void *p, ulong n)
{
	p = realloc(p, n);
	if(p == nil)
		sysfatal("erealloc: %r");
	return p;
}

void
grab(int on)
{
	static char nocurs[2*4+2*2*16];
	static int fd = -1;

	if(mson == on)
		return;
	if(mson = on && grabon){
		fd = open("/dev/cursor", ORDWR|OCEXEC);
		if(fd < 0){
			fprint(2, "grab: %r\n");
			return;
		}
		write(fd, nocurs, sizeof nocurs);
	}else if(fd >= 0){
		close(fd);
		fd = -1;
	}
}

void
toss(void)
{
	while(nbrecv(csc, nil) != 0);
	while(nbrecv(msc, nil) != 0);
	while(nbrecv(kbc, nil) != 0);
}

void
flush(void)
{
	Rectangle r;
	uchar *p;

	if(scale == 1){
		loadimage(fb, fb->r, px, npx);
		draw(screen, fbr, fb, nil, ZP);
	}else{
		p = px;
		r = fbr;
		while(r.min.y < fbr.max.y){
			r.max.y = r.min.y + scale;
			p += loadimage(fb, fb->r, p, npx/Vh);
			draw(screen, r, fb, nil, ZP);
			r.min.y = r.max.y;
		}
	}
	flushimage(display, 1);
}

void
threadmain(int argc, char **argv)
{
	int n;
	vlong t0, t, dt, Δ;
	char *datdir = "/sys/games/lib/wl3d/";

	n = 0;
	step = mstep;
	ARGBEGIN{
	case '2': ext = "sd2"; break;
	case '3': ext = "sd3"; break;
	case 'd': ext = "wl1"; break;
	case 'i': n++; break;
	case 'm': datdir = EARGF(usage()); break;
	case 'o': ext = "sdm"; break;
	case 's': ext = "sod"; break;
	case 'w': /* TODO: warp to ep, level */ break;
	case 'x': /* TODO: set difficulty for warp */ break;
	default:
		usage();
	}ARGEND;
	dat(datdir);

	if(initdraw(nil, nil, "wl3d") < 0)
		sysfatal("initdraw: %r");
	resetfb();
	kbc = chancreate(sizeof(int), 20);
	csc = chancreate(sizeof(Rune), 20);
	reszc = chancreate(sizeof(int), 2);
	msc = chancreate(sizeof(Mouse), 0);
	if(kbc == nil || csc == nil | reszc == nil || msc == nil)
		sysfatal("chancreate: %r");
	if(proccreate(kproc, nil, 8192) < 0 || proccreate(mproc, nil, 8192) < 0)
		sysfatal("proccreate: %r");

	init(n);
	t0 = Δ = 0;
	for(;;){
		if(nbrecv(reszc, nil) != 0){
			if(getwindow(display, Refnone) < 0)
				sysfatal("resize failed: %r");
			resetfb();
		}
		if(step() < 0)
			break;
		t = nsec();
		dt = 0;
		if(t0 != 0){
			dt = BILLION/Tb - (t - t0) - Δ;
			if(dt >= MILLION)
				sleep(dt/MILLION);
		}
		t0 = nsec();
		if(dt != 0){
			dt = (t0 - t) - (dt / MILLION) * MILLION;
			Δ += (dt - Δ) / 100;
		}
	}
	threadexitsall(nil);
}