ref: 9b2d3ed37a9d58d8ea4692858b47a9f1b03d7e65
dir: /sys/src/games/snes/snes.c/
#include <u.h> #include <libc.h> #include <thread.h> #include <draw.h> #include <keyboard.h> #include <mouse.h> #include <ctype.h> #include "dat.h" #include "fns.h" uchar *prg, *sram; int nprg, nsram, hirom, battery; int ppuclock, spcclock, dspclock, stimerclock, saveclock, msgclock, paused, perfclock, cpupause; Mousectl *mc; Channel *flushc, *msgc; QLock pauselock; u32int keys; int savefd, scale, profile, mouse, loadreq, savereq; Rectangle picr; Image *tmp, *bg; void flushram(void) { if(savefd >= 0) pwrite(savefd, sram, nsram, 0); saveclock = 0; } void loadrom(char *file) { int fd; vlong size; fd = open(file, OREAD); if(fd < 0) sysfatal("open: %r"); size = seek(fd, 0, 2); if(size < 0) sysfatal("seek: %r"); if(size == 0) sysfatal("empty file"); if((size & 1023) == 512){ size -= 512; seek(fd, 512, 0); }else if((size & 1023) == 0) seek(fd, 0, 0); else sysfatal("invalid rom size"); if(size >= 16*1048576) sysfatal("rom too big"); nprg = (size + 32767) / 32768; prg = malloc(nprg * 32768); if(prg == nil) sysfatal("malloc: %r"); if(readn(fd, prg, size) < size) sysfatal("read: %r"); close(fd); if(hirom < 0){ hirom = 0; if((memread(0xffd5) & ~0x10) != 0x20) if((memread(0x1ffd5) & ~0x10) == 0x21) hirom = 1; else sysfatal("invalid rom (ffd5 = %.2x, 1ffd5 = %.2x)", memread(0xffd5), memread(0x1ffd5)); } if(hirom) nprg >>= 1; switch(memread(0xffd6)){ case 0: break; case 2: battery++; case 1: nsram = memread(0xffd8); if(nsram == 0) break; if(nsram >= 0x0c) sysfatal("invalid rom (too much ram specified)"); nsram = 1<<(nsram + 10); sram = malloc(nsram); if(sram == nil) sysfatal("malloc: %r"); break; default: print("unknown rom type %d\n", memread(0xffd5)); } } void loadbat(char *file) { static char buf[512]; char *s; if(battery && nsram != 0){ buf[sizeof buf - 1] = 0; strncpy(buf, file, sizeof buf - 5); s = buf + strlen(buf) - 4; if(s < buf || strcmp(s, ".smc") != 0) s += 4; strcpy(s, ".sav"); savefd = create(buf, ORDWR | OEXCL, 0666); if(savefd < 0) savefd = open(buf, ORDWR); if(savefd < 0) message("open: %r"); else readn(savefd, sram, nsram); atexit(flushram); } } void keyproc(void *) { int fd, n, k; static char buf[256]; char *s; Rune r; fd = open("/dev/kbd", OREAD); if(fd < 0) sysfatal("open: %r"); for(;;){ if(buf[0] != 0){ n = strlen(buf)+1; memmove(buf, buf+n, sizeof(buf)-n); } if(buf[0] == 0){ n = read(fd, buf, sizeof(buf)-1); if(n <= 0) sysfatal("read /dev/kbd: %r"); buf[n-1] = 0; buf[n] = 0; } if(buf[0] == 'c'){ if(utfrune(buf, KF|5)) savereq = 1; if(utfrune(buf, KF|6)) loadreq = 1; if(utfrune(buf, Kdel)){ close(fd); threadexitsall(nil); } if(utfrune(buf, 't')) trace = !trace; } if(buf[0] != 'k' && buf[0] != 'K') continue; s = buf + 1; k = 0xffff; while(*s != 0){ s += chartorune(&r, s); switch(r){ case Kdel: close(fd); threadexitsall(nil); case 'z': k |= 1<<31; break; case 'x': k |= 1<<23; break; case 'a': k |= 1<<30; break; case 's': k |= 1<<22; break; case 'q': k |= 1<<21; break; case 'w': k |= 1<<20; break; case Kshift: k |= 1<<29; break; case 10: k |= 1<<28; break; case Kup: k |= 1<<27; break; case Kdown: k |= 1<<26; break; case Kleft: k |= 1<<25; break; case Kright: k |= 1<<24; break; case Kesc: if(paused) qunlock(&pauselock); else qlock(&pauselock); paused = !paused; break; } } if(!mouse) keys = k; } } void screeninit(void) { Point p; p = divpt(addpt(screen->r.min, screen->r.max), 2); picr = (Rectangle){subpt(p, Pt(scale * 128, scale * 112)), addpt(p, Pt(scale * 128, scale * 127))}; if(tmp != nil) freeimage(tmp); tmp = allocimage(display, Rect(0, 0, scale * 256, scale > 1 ? 1 : scale * 239), RGB15, scale > 1, 0); if(bg != nil) freeimage(bg); bg = allocimage(display, Rect(0, 0, 1, 1), screen->chan, 1, 0xCCCCCCFF); draw(screen, screen->r, bg, nil, ZP); } void screenproc(void *) { extern uchar pic[256*239*2*3]; char *s; Mouse m; Point p; enum { AMOUSE, ARESIZE, AFLUSH, AMSG, AEND }; Alt a[AEND+1] = { { mc->c, &m, CHANRCV }, { mc->resizec, nil, CHANRCV }, { flushc, nil, CHANRCV }, { msgc, &s, CHANRCV }, { nil, nil, CHANEND } }; for(;;){ switch(alt(a)){ case AMOUSE: if(mouse && ptinrect(m.xy, picr)){ p = subpt(m.xy, picr.min); p.x /= scale; p.y /= scale; keys = keys & 0xff3f0000 | p.x | p.y << 8; if((m.buttons & 1) != 0) keys |= 1<<22; if((m.buttons & 4) != 0) keys |= 1<<23; if((m.buttons & 2) != 0) lastkeys = keys; } break; case ARESIZE: if(getwindow(display, Refnone) < 0) sysfatal("resize failed: %r"); screeninit(); /* wet floor */ case AFLUSH: if(scale == 1){ loadimage(tmp, tmp->r, pic, 256*239*2); draw(screen, picr, tmp, nil, ZP); } else { Rectangle r; uchar *s; int w; s = pic; r = picr; w = 256*2*scale; while(r.min.y < picr.max.y){ loadimage(tmp, tmp->r, s, w); s += w; r.max.y = r.min.y+scale; draw(screen, r, tmp, nil, ZP); r.min.y = r.max.y; } } flushimage(display, 1); break; case AMSG: draw(screen, rectaddpt(Rect(10, 10, 200, 30), screen->r.min), bg, nil, ZP); if(s != nil){ string(screen, addpt(screen->r.min, Pt(10, 10)), display->black, ZP, display->defaultfont, s); free(s); } break; } } } void timing(void) { static vlong old; vlong new; new = nsec(); if(new != old) message("%6.2f%%", 1e11 / (new - old)); old = nsec(); } void threadmain(int argc, char **argv) { int t; extern u16int pc; scale = 1; hirom = -1; ARGBEGIN { case '2': scale = 2; break; case '3': scale = 3; break; case 'a': audioinit(); break; case 's': battery++; break; case 'm': mouse++; keys = 1<<16; break; case 'h': hirom++; break; case 'T': profile++; break; default: goto usage; } ARGEND; if(argc != 1){ usage: fprint(2, "usage: %s [-23ahmsT] rom\n", argv0); threadexitsall("usage"); } loadrom(argv[0]); if(initdraw(nil, nil, argv0) < 0) sysfatal("initdraw: %r"); flushc = chancreate(sizeof(ulong), 1); msgc = chancreate(sizeof(char*), 0); mc = initmouse(nil, screen); if(mc == nil) sysfatal("initmouse: %r"); screeninit(); proccreate(keyproc, 0, 8192); proccreate(screenproc, 0, 8192); loadbat(argv[0]); cpureset(); memreset(); spcreset(); dspreset(); for(;;){ if(savereq){ savestate("snes.save"); savereq = 0; } if(loadreq){ loadstate("snes.save"); loadreq = 0; } if(paused){ qlock(&pauselock); qunlock(&pauselock); } if(cpupause){ t = 40; cpupause = 0; }else t = cpustep(); spcclock -= t; stimerclock += t; ppuclock += t; dspclock += t; perfclock -= t; while(ppuclock >= 4){ ppustep(); ppuclock -= 4; } while(spcclock < 0) spcclock += spcstep() * SPCDIV; while(stimerclock >= SPCDIV*16){ spctimerstep(); stimerclock -= SPCDIV*16; } while(dspclock >= SPCDIV){ dspstep(); dspclock -= SPCDIV; } if(saveclock > 0){ saveclock -= t; if(saveclock <= 0) flushram(); } if(msgclock > 0){ msgclock -= t; if(msgclock <= 0){ sendp(msgc, nil); /* clear message */ msgclock = 0; } } if(profile && perfclock <= 0){ perfclock = FREQ; timing(); } } } void flush(void) { sendul(flushc, 1); /* flush screen */ audioout(); } void message(char *fmt, ...) { va_list va; va_start(va, fmt); sendp(msgc, vsmprint(fmt, va)); msgclock = FREQ; va_end(va); }