uxn11/src/uxn11.c

247 lines
7.1 KiB
C
Raw Normal View History

2022-03-26 21:32:46 -04:00
#include <stdio.h>
#include <stdlib.h>
2022-03-27 12:11:14 -04:00
#include <X11/Xlib.h>
2022-03-27 17:29:46 -04:00
#include <X11/Xutil.h>
2022-03-27 22:10:32 -04:00
#include <sys/timerfd.h>
#include <unistd.h>
#include <poll.h>
2022-03-26 21:32:46 -04:00
#include "uxn.h"
#include "devices/system.h"
#include "devices/screen.h"
2022-03-26 22:58:48 -04:00
#include "devices/controller.h"
2022-03-27 00:03:03 -04:00
#include "devices/mouse.h"
2022-03-27 13:01:06 -04:00
#include "devices/file.h"
2022-03-27 00:03:03 -04:00
#include "devices/datetime.h"
2022-03-26 22:58:48 -04:00
2022-03-27 01:38:03 -04:00
static XImage *ximage;
2022-03-27 12:11:14 -04:00
static Display *display;
static Visual *visual;
static Window window;
static Device *devscreen, *devctrl, *devmouse;
#define WIDTH 64 * 8
#define HEIGHT 40 * 8
2022-03-26 21:32:46 -04:00
static int
error(char *msg, const char *err)
{
fprintf(stderr, "Error %s: %s\n", msg, err);
return 0;
}
void
system_deo_special(Device *d, Uint8 port)
{
if(port > 0x7 && port < 0xe)
screen_palette(&uxn_screen, &d->dat[0x8]);
}
2022-03-27 13:01:06 -04:00
static int
console_input(Uxn *u, char c)
{
Device *d = &u->dev[1];
d->dat[0x2] = c;
return uxn_eval(u, GETVECTOR(d));
}
2022-03-26 21:32:46 -04:00
static void
console_deo(Device *d, Uint8 port)
{
FILE *fd = port == 0x8 ? stdout : port == 0x9 ? stderr
: 0;
if(fd) {
fputc(d->dat[port], fd);
fflush(fd);
}
}
static Uint8
nil_dei(Device *d, Uint8 port)
{
return d->dat[port];
}
static void
nil_deo(Device *d, Uint8 port)
{
(void)d;
(void)port;
}
static int
load(Uxn *u, char *filepath)
{
FILE *f;
int r;
if(!(f = fopen(filepath, "rb"))) return 0;
r = fread(u->ram + PAGE_PROGRAM, 1, 0x10000 - PAGE_PROGRAM, f);
fclose(f);
if(r < 1) return 0;
fprintf(stderr, "Loaded %s\n", filepath);
return 1;
}
2022-03-27 12:11:14 -04:00
static void
redraw(void)
2022-03-26 23:20:29 -04:00
{
screen_redraw(&uxn_screen, uxn_screen.pixels);
XPutImage(display, window, DefaultGC(display, 0), ximage, 0, 0, 0, 0, uxn_screen.width, uxn_screen.height);
}
2022-03-26 22:58:48 -04:00
/* /usr/include/X11/keysymdef.h */
2022-03-27 00:03:03 -04:00
#define XK_Escape 0xff1b
2022-03-26 22:58:48 -04:00
#define XK_Left 0xff51
#define XK_Up 0xff52
#define XK_Right 0xff53
#define XK_Down 0xff54
#define XK_Home 0xff50
#define XK_Shift 0xffe1
#define XK_Control 0xffe3
#define XK_Alt 0xffe9
2022-03-27 12:11:14 -04:00
static void
processEvent(void)
2022-03-26 21:32:46 -04:00
{
XEvent ev;
XNextEvent(display, &ev);
switch(ev.type) {
case Expose:
2022-03-27 12:11:14 -04:00
redraw();
2022-03-26 21:32:46 -04:00
break;
2022-03-27 12:11:14 -04:00
case ClientMessage: {
XDestroyImage(ximage);
XDestroyWindow(display, window);
XCloseDisplay(display);
exit(0);
} break;
2022-03-26 22:58:48 -04:00
case KeyPress: {
XKeyPressedEvent *e = (XKeyPressedEvent *)&ev;
2022-03-27 17:29:46 -04:00
char buf[7];
2022-03-27 00:03:03 -04:00
if(e->keycode == XKeysymToKeycode(display, XK_Escape)) exit(0);
2022-03-26 22:58:48 -04:00
if(e->keycode == XKeysymToKeycode(display, XK_Up)) controller_down(devctrl, 0x10);
if(e->keycode == XKeysymToKeycode(display, XK_Down)) controller_down(devctrl, 0x20);
if(e->keycode == XKeysymToKeycode(display, XK_Left)) controller_down(devctrl, 0x40);
if(e->keycode == XKeysymToKeycode(display, XK_Right)) controller_down(devctrl, 0x80);
if(e->keycode == XKeysymToKeycode(display, XK_Control)) controller_down(devctrl, 0x01);
if(e->keycode == XKeysymToKeycode(display, XK_Alt)) controller_down(devctrl, 0x02);
if(e->keycode == XKeysymToKeycode(display, XK_Shift)) controller_down(devctrl, 0x04);
if(e->keycode == XKeysymToKeycode(display, XK_Home)) controller_down(devctrl, 0x08);
2022-03-27 17:29:46 -04:00
XLookupString(e, buf, 7, NULL, NULL);
controller_key(devctrl, buf[0]);
2022-03-26 22:58:48 -04:00
} break;
case KeyRelease: {
XKeyPressedEvent *e = (XKeyPressedEvent *)&ev;
if(e->keycode == XKeysymToKeycode(display, XK_Up)) controller_up(devctrl, 0x10);
if(e->keycode == XKeysymToKeycode(display, XK_Down)) controller_up(devctrl, 0x20);
if(e->keycode == XKeysymToKeycode(display, XK_Left)) controller_up(devctrl, 0x40);
if(e->keycode == XKeysymToKeycode(display, XK_Right)) controller_up(devctrl, 0x80);
if(e->keycode == XKeysymToKeycode(display, XK_Control)) controller_up(devctrl, 0x01);
if(e->keycode == XKeysymToKeycode(display, XK_Alt)) controller_up(devctrl, 0x02);
if(e->keycode == XKeysymToKeycode(display, XK_Shift)) controller_up(devctrl, 0x04);
if(e->keycode == XKeysymToKeycode(display, XK_Home)) controller_up(devctrl, 0x08);
} break;
2022-03-27 00:03:03 -04:00
case ButtonPress: {
XButtonPressedEvent *e = (XButtonPressedEvent *)&ev;
2022-03-27 13:01:06 -04:00
mouse_down(devmouse, 0x1 << e->button - 1);
2022-03-27 00:03:03 -04:00
} break;
case ButtonRelease: {
XButtonPressedEvent *e = (XButtonPressedEvent *)&ev;
2022-03-27 13:01:06 -04:00
mouse_up(devmouse, 0x1 << e->button - 1);
2022-03-27 00:03:03 -04:00
} break;
case MotionNotify: {
XMotionEvent *e = (XMotionEvent *)&ev;
mouse_pos(devmouse, e->x, e->y);
} break;
2022-03-26 23:20:29 -04:00
}
2022-03-26 21:32:46 -04:00
}
static int
2022-03-27 12:11:14 -04:00
start(Uxn *u, char *rom)
2022-03-26 21:32:46 -04:00
{
if(!uxn_boot(u, (Uint8 *)calloc(0x10000, sizeof(Uint8))))
return error("Boot", "Failed");
2022-03-27 12:11:14 -04:00
if(!load(u, rom))
return error("Load", "Failed");
2022-03-26 21:46:17 -04:00
/* system */ uxn_port(u, 0x0, system_dei, system_deo);
/* console */ uxn_port(u, 0x1, nil_dei, console_deo);
2022-03-27 00:03:03 -04:00
/* screen */ devscreen = uxn_port(u, 0x2, screen_dei, screen_deo);
2022-03-26 21:32:46 -04:00
/* empty */ uxn_port(u, 0x3, nil_dei, nil_deo);
/* empty */ uxn_port(u, 0x4, nil_dei, nil_deo);
/* empty */ uxn_port(u, 0x5, nil_dei, nil_deo);
/* empty */ uxn_port(u, 0x6, nil_dei, nil_deo);
/* empty */ uxn_port(u, 0x7, nil_dei, nil_deo);
2022-03-26 22:58:48 -04:00
/* control */ devctrl = uxn_port(u, 0x8, nil_dei, nil_deo);
2022-03-27 00:03:03 -04:00
/* mouse */ devmouse = uxn_port(u, 0x9, nil_dei, nil_deo);
2022-03-26 21:32:46 -04:00
/* file */ uxn_port(u, 0xa, nil_dei, nil_deo);
/* datetime */ uxn_port(u, 0xb, nil_dei, nil_deo);
2022-03-27 00:03:03 -04:00
/* empty */ uxn_port(u, 0xc, datetime_dei, nil_deo);
2022-03-26 21:32:46 -04:00
/* empty */ uxn_port(u, 0xd, nil_dei, nil_deo);
/* empty */ uxn_port(u, 0xe, nil_dei, nil_deo);
/* empty */ uxn_port(u, 0xf, nil_dei, nil_deo);
2022-03-27 12:11:14 -04:00
screen_resize(&uxn_screen, WIDTH, HEIGHT);
if(!uxn_eval(u, PAGE_PROGRAM))
return error("Boot", "Failed to start rom.");
return 1;
}
static int
init(void)
{
Atom wmDelete;
display = XOpenDisplay(NULL);
visual = DefaultVisual(display, 0);
window = XCreateSimpleWindow(display, RootWindow(display, 0), 0, 0, uxn_screen.width, uxn_screen.height, 1, 0, 0);
if(visual->class != TrueColor)
return error("Init", "True-color visual failed");
XSelectInput(display, window, ButtonPressMask | ButtonReleaseMask | PointerMotionMask | ExposureMask | KeyPressMask | KeyReleaseMask);
wmDelete = XInternAtom(display, "WM_DELETE_WINDOW", True);
XSetWMProtocols(display, window, &wmDelete, 1);
XMapWindow(display, window);
ximage = XCreateImage(display, visual, DefaultDepth(display, DefaultScreen(display)), ZPixmap, 0, (char *)uxn_screen.pixels, uxn_screen.width, uxn_screen.height, 32, 0);
2022-03-26 21:32:46 -04:00
return 1;
}
int
main(int argc, char **argv)
{
Uxn u;
2022-03-27 13:01:06 -04:00
int i;
2022-03-27 22:10:32 -04:00
char expirations[8];
struct pollfd fds[2];
static const struct itimerspec screen_tspec = {{0, 16666666}, {0, 16666666}};
2022-03-26 21:32:46 -04:00
if(argc < 2)
return error("Usage", "uxncli game.rom args");
2022-03-27 12:11:14 -04:00
if(!start(&u, argv[1]))
2022-03-26 21:32:46 -04:00
return error("Start", "Failed");
2022-03-27 12:11:14 -04:00
if(!init())
return error("Init", "Failed");
2022-03-27 13:01:06 -04:00
/* console vector */
for(i = 2; i < argc; i++) {
char *p = argv[i];
while(*p) console_input(&u, *p++);
console_input(&u, '\n');
}
2022-03-27 22:10:32 -04:00
fds[0].fd = XConnectionNumber(display);
fds[1].fd = timerfd_create(CLOCK_MONOTONIC, 0);
timerfd_settime(fds[1].fd, 0, &screen_tspec, NULL);
fds[0].events = fds[1].events = POLLIN;
2022-03-27 13:01:06 -04:00
/* main loop */
2022-03-26 21:32:46 -04:00
while(1) {
2022-03-27 22:10:32 -04:00
if(poll(fds, 2, 1000) <= 0)
continue;
while(XPending(display))
processEvent();
if(poll(&fds[1], 1, 0)) {
read(fds[1].fd, expirations, 8); /* Indicate we handled the timer */
uxn_eval(&u, GETVECTOR(devscreen)); /* Call the vector once, even if the timer fired multiple times */
}
2022-03-27 12:11:14 -04:00
if(uxn_screen.fg.changed || uxn_screen.bg.changed)
redraw();
2022-03-26 21:32:46 -04:00
}
2022-03-27 01:38:03 -04:00
XDestroyImage(ximage);
2022-03-26 21:32:46 -04:00
return 0;
2022-03-27 01:38:03 -04:00
}