ref: 3891487cd9fb5088d99344d33cf4e37e0abf7f9b
dir: /wl3d.c/
#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); }