summaryrefslogtreecommitdiff
path: root/sources
diff options
context:
space:
mode:
Diffstat (limited to 'sources')
-rw-r--r--sources/chip8.c7
-rw-r--r--sources/iui.c306
-rw-r--r--sources/main.c454
-rw-r--r--sources/mem.c139
-rw-r--r--sources/meta/exec.c26
-rw-r--r--sources/plat.c24
-rw-r--r--sources/x11.c8
7 files changed, 672 insertions, 292 deletions
diff --git a/sources/chip8.c b/sources/chip8.c
index 1c5f907..836b9ef 100644
--- a/sources/chip8.c
+++ b/sources/chip8.c
@@ -7,7 +7,7 @@
#define C8_RESET_VECTOR 0x200
-#define C8_CYCLES_PER_FRAME 1000
+#define C8_CYCLES_PER_FRAME 20
static const u8 fnt[] =
{
@@ -36,6 +36,7 @@ typedef struct
u8 sp;
u16 i;
+ u16 prevkeys;
u16 keys;
u64 cycles;
@@ -137,8 +138,8 @@ void c8_dump_state(chip8* c8, bool stack)
printf("V%x: %02X (%3d)\n", i+3, c8->V[i+3], c8->V[i+3]);
}
- printf(" I:%04X SP: %02X DT: %02X ST: %02X CYCLES: %llu\n",
- c8->i, c8->sp, c8->dt, c8->st, c8->cycles);
+ printf(" I:%04X SP: %02X DT: %02X ST: %02X CYCLES: %llu, KEYS: %X\n",
+ c8->i, c8->sp, c8->dt, c8->st, c8->cycles, c8->keys);
if(stack)
{
diff --git a/sources/iui.c b/sources/iui.c
new file mode 100644
index 0000000..4e656c6
--- /dev/null
+++ b/sources/iui.c
@@ -0,0 +1,306 @@
+/*
+ * Copyright (C) 2025 dwlr <dweller@cabin.digital>
+ *
+ * BSD 3-Clause License (BSD-3-Clause)
+ * See LICENSE for details
+ */
+
+
+#define IUI_TXT_CENTER bit(0)
+#define IUI_CLIP_CLEAR bit(0)
+
+typedef struct
+{
+ enum
+ {
+ IUI_ILL,
+ IUI_BOX,
+ IUI_FILL,
+ IUI_LINE,
+ IUI_TEXT,
+ IUI_CLIP,
+
+ } type;
+
+ u32 color;
+ u16 x, y, w, h;
+ u16 size;
+ u16 flags;
+ char* text;
+
+} iui_cmd;
+
+#define IUI_CMDBUF_LEN 512
+typedef struct
+{
+ arena* ar;
+ void* usr;
+
+ usize hot;
+ usize active;
+ usize kbdfocus;
+
+ /* mouse */
+ u16 x;
+ u16 y;
+
+ u8 pbtns;
+ u8 btns;
+
+ /* TODO: event pump */
+
+ usize ncmds;
+ iui_cmd cmds[IUI_CMDBUF_LEN]; /* TODO: arena? */
+
+} iui;
+
+
+void iui_init(iui* ui, arena* ar, void* usr)
+{
+ if(!ui)
+ {
+ report(WARN, "iui: iui_init() ui == NULL!\n");
+ return;
+ }
+
+ ui->ar = ar;
+ ui->usr = usr;
+}
+
+void iui_start(iui* ui)
+{
+ assert(ui);
+ ui->hot = 0;
+
+ ar_clear(ui->ar);
+}
+
+void iui_end(iui* ui)
+{
+ assert(ui);
+
+ if(!ui->btns) ui->active = 0;
+ else if(!ui->active) ui->kbdfocus = 0;
+
+ ui->pbtns = ui->btns;
+}
+
+bool iui_mouse_inside(iui* ui, u16 x, u16 y, u16 w, u16 h)
+{
+ return !(ui->x < x || ui->y < y || ui->x >= x + w || ui->y >= y + h);
+}
+
+int iui_button(iui* ui, u16 x, u16 y, int w, int h, char* label)
+{
+ xctx* x11 = (xctx*)ui->usr;
+ usize id = (usize)label;
+ bool imhot, imactive;
+
+ if(iui_mouse_inside(ui, x, y, w, h))
+ {
+ bool click = ui->btns & bit(1) && !(ui->pbtns & bit(1));
+ if(!ui->active && click) ui->active = id;
+ ui->hot = id;
+ }
+
+ imhot = ui->hot == id;
+ imactive = ui->active == id;
+
+ {
+ iui_cmd cmd = {0};
+ cmd.type = IUI_FILL;
+
+ cmd.x = x;
+ cmd.y = y;
+ cmd.w = w;
+ cmd.h = h;
+
+ if(imhot) cmd.size = 2;
+ else cmd.size = 1;
+
+ if(imactive) cmd.color = x11->cols.as.fg;
+ else cmd.color = x11->cols.as.bg;
+
+ ui->cmds[ui->ncmds++] = cmd;
+
+ cmd.type = IUI_BOX;
+ cmd.color = x11->cols.as.fg;
+ ui->cmds[ui->ncmds++] = cmd;
+
+ cmd.type = IUI_TEXT;
+ cmd.size = 14;
+ cmd.text = label;
+ cmd.flags = IUI_TXT_CENTER;
+
+ if(imactive) cmd.color = x11->cols.as.bg;
+ else cmd.color = x11->cols.as.fg;
+
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+
+ return imhot && imactive && !(ui->btns & bit(1));
+}
+
+void iui_label(iui* ui, u16 x, u16 y, int w, int h, char* fmt, ...)
+{
+ int len = 0;
+ char* buf = NULL;
+ va_list ap = {0};
+ xctx* x11 = (xctx*)ui->usr;
+
+
+ va_start(ap, fmt);
+ len = vsnprintf(NULL, 0, fmt, ap);
+ va_end(ap);
+
+ if(len) len++;
+ buf = ar_push(ui->ar, len);
+
+ va_start(ap, fmt);
+ len = vsnprintf(buf, len, fmt, ap);
+ va_end(ap);
+
+ {
+ iui_cmd cmd = {0};
+ cmd.type = IUI_FILL;
+
+ cmd.x = x;
+ cmd.y = y;
+ cmd.w = w;
+ cmd.h = h;
+
+ {
+ iui_cmd clip = cmd;
+ clip.type = IUI_CLIP;
+ clip.flags = 0;
+ clip.x++;
+ clip.y++;
+ clip.w--;
+ clip.h--;
+
+ ui->cmds[ui->ncmds++] = clip;
+ }
+
+ {
+ cmd.type = IUI_TEXT;
+ cmd.size = 14;
+ cmd.text = buf;
+ cmd.flags = 0;
+
+ cmd.x = x + 6;
+ cmd.y = y;
+ cmd.w = w;
+ cmd.h = h;
+
+ cmd.color = x11->cols.as.fg;
+
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+
+ cmd.type = IUI_CLIP;
+ cmd.flags = IUI_CLIP_CLEAR;
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+}
+
+int iui_editbox(iui* ui, u16 x, u16 y, int w, int h, char* buf, usize buflen)
+{
+ xctx* x11 = (xctx*)ui->usr;
+ usize id = (usize)buf;
+ bool imhot, imactive;
+ usize len;
+
+ len = strlen(buf);
+
+ if(iui_mouse_inside(ui, x, y, w, h))
+ {
+ bool click = ui->btns & bit(1) && !(ui->pbtns & bit(1));
+ if(!ui->active && click) ui->active = id;
+ ui->hot = id;
+ }
+
+ imhot = ui->hot == id;
+ imactive = ui->active == id;
+
+ {
+ iui_cmd cmd = {0};
+ cmd.type = IUI_FILL;
+
+ cmd.x = x;
+ cmd.y = y;
+ cmd.w = w;
+ cmd.h = h;
+
+ if(imhot) cmd.size = 2;
+ else cmd.size = 1;
+
+ cmd.color = x11->cols.as.bg;
+ ui->cmds[ui->ncmds++] = cmd;
+
+ cmd.type = IUI_BOX;
+ cmd.color = x11->cols.as.fg;
+ ui->cmds[ui->ncmds++] = cmd;
+
+ {
+ iui_cmd clip = cmd;
+ clip.type = IUI_CLIP;
+ clip.flags = 0;
+ clip.x++;
+ clip.y++;
+ clip.w--;
+ clip.h--;
+
+ ui->cmds[ui->ncmds++] = clip;
+ }
+
+ if(imactive) ui->kbdfocus = id;
+ if(ui->kbdfocus == id)
+ {
+ cmd.type = IUI_FILL;
+ cmd.x = x + 6 + 6 * len;
+ cmd.y = y + 6;
+ cmd.w = 10;
+ cmd.h = h - 12;
+ cmd.color = x11->cols.as.fg;
+
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+ else
+ {
+ cmd.type = IUI_BOX;
+ cmd.x = x + 6 + 6 * len;
+ cmd.y = y + 6;
+ cmd.w = 9;
+ cmd.h = h - 13;
+ cmd.color = x11->cols.as.fg;
+ cmd.size = 1;
+
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+
+ {
+ cmd.type = IUI_TEXT;
+ cmd.size = 14;
+ cmd.text = buf;
+ cmd.flags = 0;
+
+ cmd.x = x + 6;
+ cmd.y = y;
+ cmd.w = w;
+ cmd.h = h;
+
+ cmd.color = x11->cols.as.fg;
+
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+
+ cmd.type = IUI_CLIP;
+ cmd.flags = IUI_CLIP_CLEAR;
+ ui->cmds[ui->ncmds++] = cmd;
+ }
+
+ return imhot && imactive && !(ui->btns & bit(1));
+}
+
+
+void iui_draw(iui* ui);
diff --git a/sources/main.c b/sources/main.c
index 4049a30..7027451 100644
--- a/sources/main.c
+++ b/sources/main.c
@@ -21,6 +21,8 @@
#include <sys/stat.h>
#include <fcntl.h>
+#include <pthread.h>
+
#include <X11/Xlib.h>
#include <X11/Xatom.h>
#include <X11/Xutil.h>
@@ -28,209 +30,17 @@
#include "bits.c"
#include "log.c"
+#include "plat.c"
+#include "mem.c"
#include "tga.c"
#include "x11.c"
+#include "iui.c"
#include "chip8.c"
#include "meta/disasm.c"
#include "meta/exec.c"
-typedef struct
-{
- enum
- {
- IUI_ILL,
- IUI_BOX,
- IUI_FILL,
- IUI_LINE,
- IUI_TEXT,
-
- } type;
-
- u32 color;
- u16 x, y, w, h;
- u16 size;
- char* text;
-
-} iui_cmd;
-
-#define IUI_CMDBUF_LEN 512
-typedef struct
-{
- void* usr;
-
- usize hot;
- usize active;
- usize kbdfocus;
-
- /* mouse */
- u16 x;
- u16 y;
-
- u8 pbtns;
- u8 btns;
-
- usize ncmds;
- iui_cmd cmds[IUI_CMDBUF_LEN]; /* TODO: arena? */
-
-} iui;
-
-
-void iui_start(iui* ui, void* usr)
-{
- assert(ui);
-
- ui->usr = usr;
- ui->hot = 0;
-}
-
-void iui_end(iui* ui)
-{
- assert(ui);
-
- if(!ui->btns) ui->active = 0;
- else
- {
- if(!ui->active) ui->kbdfocus = 0;
- }
-
- ui->pbtns = ui->btns;
-}
-
-bool iui_mouse_inside(iui* ui, u16 x, u16 y, u16 w, u16 h)
-{
- return !(ui->x < x || ui->y < y || ui->x >= x + w || ui->y >= y + h);
-}
-
-int iui_button(iui* ui, u16 x, u16 y, int w, int h, char* label)
-{
- xctx* x11 = (xctx*)ui->usr;
- usize id = (usize)label;
- bool imhot, imactive;
-
- if(iui_mouse_inside(ui, x, y, w, h))
- {
- bool click = ui->btns & bit(1) && !(ui->pbtns & bit(1));
- if(!ui->active && click) ui->active = id;
- ui->hot = id;
- }
-
- imhot = ui->hot == id;
- imactive = ui->active == id;
-
- {
- iui_cmd cmd = {0};
- cmd.type = IUI_FILL;
-
- cmd.x = x;
- cmd.y = y;
- cmd.w = w;
- cmd.h = h;
-
- if(imhot) cmd.size = 2;
- else cmd.size = 1;
-
- if(imactive) cmd.color = x11->cols.as.fg;
- else cmd.color = x11->cols.as.bg;
-
- ui->cmds[ui->ncmds++] = cmd;
-
- cmd.type = IUI_BOX;
- cmd.color = x11->cols.as.fg;
- ui->cmds[ui->ncmds++] = cmd;
-
- cmd.type = IUI_TEXT;
- cmd.size = 14;
- cmd.text = label;
-
- if(imactive) cmd.color = x11->cols.as.bg;
- else cmd.color = x11->cols.as.fg;
-
- ui->cmds[ui->ncmds++] = cmd;
- }
-
- return imhot && imactive && !(ui->btns & bit(1));
-}
-
-int iui_textbox(iui* ui, u16 x, u16 y, int w, int h, char* buf, usize buflen)
-{
- xctx* x11 = (xctx*)ui->usr;
- usize id = (usize)buf;
- bool imhot, imactive;
- usize len;
-
- len = strlen(buf);
-
- if(iui_mouse_inside(ui, x, y, w, h))
- {
- bool click = ui->btns & bit(1) && !(ui->pbtns & bit(1));
- if(!ui->active && click) ui->active = id;
- ui->hot = id;
- }
-
- imhot = ui->hot == id;
- imactive = ui->active == id;
-
- {
- XGCValues gcvals = {0};
-
- if(imhot) gcvals.line_width = 2;
- else gcvals.line_width = 1;
-
- gcvals.foreground = x11->cols.as.bg;
-
- XChangeGC(x11->disp, x11->gc, GCForeground | GCLineWidth, &gcvals);
- XFillRectangle(x11->disp, x11->wnd, x11->gc, x, y, w, h);
-
- XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
- XDrawLine(x11->disp, x11->wnd, x11->gc, x, y, x, y + h);
- XDrawLine(x11->disp, x11->wnd, x11->gc, x + w, y, x, y);
- XDrawLine(x11->disp, x11->wnd, x11->gc, x, y + h, x + w, y + h);
- XDrawLine(x11->disp, x11->wnd, x11->gc, x + w, y + h, x + w, y);
- }
-
- if(imactive) ui->kbdfocus = id;
- if(ui->kbdfocus == id)
- {
-
- XGCValues gcvals = {0};
- gcvals.foreground = x11->cols.as.fg;
-
- XChangeGC(x11->disp, x11->gc, GCForeground, &gcvals);
- XFillRectangle(x11->disp, x11->wnd, x11->gc, x + 6 + (6 * len), y + 6, 10, h - 12);
- }
- else
- {
- XGCValues gcvals = {0};
- gcvals.foreground = x11->cols.as.fg;
- gcvals.line_width = 1;
-
- XChangeGC(x11->disp, x11->gc, GCForeground | GCLineWidth, &gcvals);
- XDrawRectangle(x11->disp, x11->wnd, x11->gc, x + 6 + (6 * len), y + 6, 9, h - 13);
- }
-
- {
- int xx, yy;
- int di, fa, fd;
- int width, height;
- XCharStruct chs = {0};
-
- XTextExtents(x11->xfs, buf, buflen, &di, &fa, &fd, &chs);
- height = chs.ascent + chs.descent;
-
- xx = x + 6;
- yy = y + h / 2 + height / 2;
-
-
- if(imactive) XSetForeground(x11->disp, x11->gc, x11->cols.as.bg);
- else XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
- XDrawString(x11->disp, x11->wnd, x11->gc, xx, yy, buf, len);
- }
-
- return imhot && imactive && !(ui->btns & bit(1));
-}
-
void iui_draw(iui* ui)
{
usize i;
@@ -283,7 +93,10 @@ void iui_draw(iui* ui)
width = chs.rbearing - chs.lbearing;
height = chs.ascent + chs.descent;
- xx = cmd->x + cmd->w / 2 - width / 2;
+ if(cmd->flags & IUI_TXT_CENTER)
+ xx = cmd->x + cmd->w / 2 - width / 2;
+ else
+ xx = cmd->x;
yy = cmd->y + cmd->h / 2 + height / 2;
XSetForeground(x11->disp, x11->gc, cmd->color);
@@ -291,6 +104,22 @@ void iui_draw(iui* ui)
}
break;
+ case IUI_CLIP:
+ {
+ if(cmd->flags & IUI_CLIP_CLEAR) XSetClipMask(x11->disp, x11->gc, None);
+ else
+ {
+ XRectangle clip_region = {0};
+ clip_region.x = cmd->x;
+ clip_region.y = cmd->y;
+ clip_region.width = cmd->w;
+ clip_region.height = cmd->h;
+
+ XSetClipRectangles(x11->disp, x11->gc, 0, 0, &clip_region, 1, Unsorted);
+ }
+ }
+ break;
+
default:
report(WARN, "IUI: Unknown draw command %u\n", cmd->type);
break;
@@ -300,15 +129,106 @@ void iui_draw(iui* ui)
ui->ncmds = 0;
}
+void c8_dump_state_iui(chip8* c8, bool stack, iui* ui, u16 x, u16 y)
+{
+ int i = 0;
+
+ u16 dx = x, dy = y;
+ u16 w = 80, h = 10, p = 5;
+
+ for(i = 0; i < (int)sizeof(c8->V); i += 4)
+ {
+ iui_label(ui, dx, dy, w, h, "V%x: %02X (%3d)", i+0, c8->V[i+0], c8->V[i+0]);
+ dx += w + p;
+
+ iui_label(ui, dx, dy, w, h, "V%x: %02X (%3d)", i+1, c8->V[i+1], c8->V[i+1]);
+ dx += w + p;
+
+ iui_label(ui, dx, dy, w, h, "V%x: %02X (%3d)", i+2, c8->V[i+2], c8->V[i+2]);
+ dx += w + p;
+
+ iui_label(ui, dx, dy, w, h, "V%x: %02X (%3d)", i+3, c8->V[i+3], c8->V[i+3]);
+
+ dx = x;
+ dy += h + p;
+ }
+
+ iui_label(ui, dx, dy, 1000, h, "I:%04X SP: %02X DT: %02X ST: %02X CYCLES: %llu KEYS: %X",
+ c8->i, c8->sp, c8->dt, c8->st, c8->cycles, c8->keys);
+#if 0
+ if(stack)
+ {
+ iui_label(ui, "stack:\n");
+ for(i = 0; i < (int)sizeof(c8->STACK); i += 2)
+ {
+ iui_label(ui, " %-2d: %04X ", i+0, c8->STACK[i+0]);
+ iui_label(ui, "%-2d: %04X\n", i+1, c8->STACK[i+1]);
+ }
+ }
+#endif
+}
+
int main(int argc, char** argv)
{
+ chip8 c8 = {0};
xctx x11 = {0};
+ arena ar = {0};
iui ui = {0};
bool run = true;
- char txtbuf[256] = {0};
+
+ u64 sz = 0;
+ FILE* f = NULL;
+ char buf[512] = {0};
+
+ srand(time(NULL));
+
+ if(argc > 1)
+ {
+ u64 got = 0;
+
+ f = fopen(argv[1], "r");
+ if(!f) return 1;
+
+ fseek(f, 0, SEEK_END);
+ sz = (u64)ftell(f);
+ rewind(f);
+
+ if(sz >= (sizeof(c8.RAM) - 512)) return 2;
+
+ got = fread(c8.RAM + C8_RESET_VECTOR, 1, sz, f);
+ if(got != sz) return 3;
+
+ fclose(f);
+ }
+ else
+ {
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x60;
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x04;
+
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x61;
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x03;
+
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x72;
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x01;
+
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x80;
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x14;
+
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x30;
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x10;
+
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x12;
+ c8.RAM[C8_RESET_VECTOR + iota] = 0x04;
+
+ sz = iota;
+ }
x11_init(800, 600, "xip-8", argc, argv, &x11);
+ ar_create(4*MB, true, &ar);
+ iui_init(&ui, &ar, &x11);
+ c8_reset(&c8);
+ c8.running = false;
while(run)
{
@@ -337,6 +257,35 @@ int main(int argc, char** argv)
len = XLookupString(&ev.xkey, name, sizeof(name) - 1, &ks, NULL);
if(keyup && ks == XK_Escape) run = false;
+
+ if(!ui.kbdfocus)
+ {
+ u16 bits = 0;
+ switch(name[0])
+ {
+ case 'x': bits = bit(0); break;
+ case '1': bits = bit(1); break;
+ case '2': bits = bit(2); break;
+ case '3': bits = bit(3); break;
+ case 'q': bits = bit(4); break;
+ case 'w': bits = bit(5); break;
+ case 'e': bits = bit(6); break;
+ case 'd': bits = bit(7); break;
+ case 's': bits = bit(8); break;
+ case 'a': bits = bit(9); break;
+ case 'z': bits = bit(10); break;
+ case 'c': bits = bit(11); break;
+ case '4': bits = bit(12); break;
+ case 'r': bits = bit(13); break;
+ case 'f': bits = bit(14); break;
+ case 'v': bits = bit(15); break;
+ default: break;
+ }
+
+ if(keyup) c8.keys &= ~bits;
+ else c8.keys |= bits;
+ }
+
if(!keyup && len == 1)
{
if(ui.kbdfocus && (isgraph(name[0]) || name[0] == ' '))
@@ -395,6 +344,12 @@ int main(int argc, char** argv)
}
}
+ if(c8.running)
+ {
+ c8_disasm(c8.RAM, c8.pc, buf);
+ c8_exec(c8.RAM, c8.pc, &c8);
+ }
+
{
int len = 0;
char buf[512] = {0};
@@ -414,8 +369,6 @@ int main(int argc, char** argv)
len = snprintf(buf, lengthof(buf), "cur: %dx%d", ui.x, ui.y);
XDrawString(x11.disp, x11.wnd, x11.gc, 650, 65, buf, len);
- iui_draw(&ui);
-
len = snprintf(buf, lengthof(buf), "btn: %d%d%d",
ui.btns & bit(1) ? 1 : 0,
ui.btns & bit(2) ? 2 : 0,
@@ -430,98 +383,35 @@ int main(int argc, char** argv)
len = snprintf(buf, lengthof(buf), "kbd: %p", (void*)ui.kbdfocus);
XDrawString(x11.disp, x11.wnd, x11.gc, 650, 125, buf, len);
- }
-
- iui_start(&ui, &x11);
- if(iui_button(&ui, 25, 25, 100, 35, "Click ME!")) report(DBG, "CLICK!\n");
- if(iui_button(&ui, 150, 25, 100, 35, "NOT ME!!!")) report(DBG, "WHYYYYYY!\n");
-
- iui_textbox(&ui, 25, 75, 225, 35, txtbuf, lengthof(txtbuf));
- if(iui_button(&ui, 275, 75, 100, 35, "OK")) report(DBG, "textbox: '%s'!\n", txtbuf);
- iui_end(&ui);
- usleep(10000);
- }
+ len = snprintf(buf, lengthof(buf), "ar: %lu bytes", ar.cursor - ar.base);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 145, buf, len);
- x11_cleanup(&x11);
- return 0;
-}
-
-int main2(int argc, char** argv)
-{
- u64 sz = 0;
- FILE* f = NULL;
- chip8 c8 = {0};
-
- srand(time(NULL));
-
- c8_reset(&c8);
-
- if(argc > 1)
- {
- u64 got = 0;
-
- f = fopen(argv[1], "r");
- if(!f) return 1;
-
- fseek(f, 0, SEEK_END);
- sz = (u64)ftell(f);
- rewind(f);
-
- if(sz >= (sizeof(c8.RAM) - 512)) return 2;
-
- got = fread(c8.RAM + C8_RESET_VECTOR, 1, sz, f);
- if(got != sz) return 3;
-
- fclose(f);
- }
- else
- {
- c8.RAM[C8_RESET_VECTOR + iota] = 0x60;
- c8.RAM[C8_RESET_VECTOR + iota] = 0x04;
-
- c8.RAM[C8_RESET_VECTOR + iota] = 0x61;
- c8.RAM[C8_RESET_VECTOR + iota] = 0x03;
-
- c8.RAM[C8_RESET_VECTOR + iota] = 0x72;
- c8.RAM[C8_RESET_VECTOR + iota] = 0x01;
-
- c8.RAM[C8_RESET_VECTOR + iota] = 0x80;
- c8.RAM[C8_RESET_VECTOR + iota] = 0x14;
-
- c8.RAM[C8_RESET_VECTOR + iota] = 0x30;
- c8.RAM[C8_RESET_VECTOR + iota] = 0x10;
-
- c8.RAM[C8_RESET_VECTOR + iota] = 0x12;
- c8.RAM[C8_RESET_VECTOR + iota] = 0x04;
-
- sz = iota;
- }
+ iui_draw(&ui);
+ }
- printf("Disasm:\n");
- {
- u64 i;
- char buf[512] = {0};
+ iui_start(&ui);
+ c8_dump_state_iui(&c8, false, &ui, 25, 25);
+ iui_editbox(&ui, 25, 600 - 45, 225, 35, buf, lengthof(buf));
+ if(!c8.running && iui_button(&ui, 270, 600 - 45, 100, 35, "Go")) c8.running = true;
+ else if(c8.running && iui_button(&ui, 270, 600 - 45, 100, 35, "Stop")) c8.running = false;
+ iui_end(&ui);
- for(i = 0; i < sz; i += 2)
{
- u16 pc = C8_RESET_VECTOR + i;
- c8_disasm(c8.RAM, pc, buf);
- printf("%04X: %s\n", pc, buf);
- }
- }
+ usize x, y, s = 12;
- printf("\nExec:\n");
- while(c8.running)
- {
- char buf[512] = {0};
+ XSetForeground(x11.disp, x11.gc, x11.cols.as.fg);
+ XSetBackground(x11.disp, x11.gc, x11.cols.as.bg);
- c8_disasm(c8.RAM, c8.pc, buf);
- printf("%04X: %s\n", c8.pc, buf);
+ for(y = 0; y < 32; y++)
+ for(x = 0; x < 64; x++)
+ if(c8.DISPLAY[y] & bit(x))
+ XFillRectangle(x11.disp, x11.wnd, x11.gc, 25 + x*s, 128 + y*s, s, s);
+ }
- c8_exec(c8.RAM, c8.pc, &c8);
- c8_dump_state(&c8, false);
+ usleep(100);
}
+ x11_cleanup(&x11);
return 0;
}
diff --git a/sources/mem.c b/sources/mem.c
new file mode 100644
index 0000000..3b111f7
--- /dev/null
+++ b/sources/mem.c
@@ -0,0 +1,139 @@
+/*
+ * Copyright (C) 2025 dwlr <dweller@cabin.digital>
+ *
+ * BSD 3-Clause License (BSD-3-Clause)
+ * See LICENSE for details
+ */
+
+
+typedef struct
+{
+ u8* base;
+ u8* cursor;
+ usize size;
+
+} arena;
+
+
+typedef struct
+{
+ u8* base;
+ ssize pages;
+
+} do_pagefault_args;
+void* do_pagefault(void* arg)
+{
+ ssize i = 0;
+ ssize p = mm_pagesize();
+ do_pagefault_args* args = (do_pagefault_args*)arg;
+ if(!args)
+ {
+ report(ERR, "do_pagefault() with NULL args!\n");
+ return NULL;
+ }
+
+ for(i = 0; i < args->pages; i++) args->base[p * i] = 0;
+
+ return NULL;
+}
+
+bool ar_create(usize size, bool prefault, arena* ar)
+{
+ ssize p = mm_pagesize();
+
+ if(!ar) return false;
+ if(size <= 0) return false;
+
+ size += (size % p) ? (p - (size % p)) : 0;
+
+ ar->base = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANON, -1, 0);
+ if(ar->base == MAP_FAILED) return false;
+ ar->size = size;
+ ar->cursor = ar->base;
+
+ if(prefault)
+ {
+ ssize i = 0;
+ ssize cpus = cpu_count();
+ pthread_t* t = NULL;
+ do_pagefault_args* args = NULL;
+
+ t = calloc(cpus, sizeof(t[0]));
+ if(!t) report(FATAL, "ar_create()/prefault/calloc(t) failed!\n");
+ else
+ {
+ args = calloc(cpus, sizeof(args[0]));
+ if(!args)
+ {
+ free(t);
+ report(FATAL, "ar_create()/prefault/calloc(args) failed!\n");
+ }
+ }
+
+ for(i = 1; i < cpus; i++)
+ {
+ args[i].base = ar->base + ((size / cpus) * i);
+ args[i].pages = ((size / p) / cpus);
+ pthread_create(&t[i], NULL, do_pagefault, &args[i]);
+ }
+
+ args[0].base = ar->base;
+ args[0].pages = ((size / p) / cpus);
+ do_pagefault(&args[0]);
+
+ for(i = 1; i < cpus; i++) pthread_join(t[i], NULL);
+
+ free(args);
+ free(t);
+ }
+
+ return true;
+}
+
+#define ar_push(ar, sz) ar_pusha((ar), (sz), sizeof(int), true)
+void* ar_pusha(arena* ar, usize size, usize alignment, bool zero)
+{
+ usize cur, ovfl;
+
+ if(!ar) return NULL;
+ else
+ {
+ cur = align((usize)ar->cursor, alignment);
+ ovfl = (cur + size);
+ if(ovfl < cur) return NULL;
+
+ if(ovfl <= (usize)(ar->base + ar->size))
+ {
+ ar->cursor = (u8*)ovfl;
+
+ if(zero) memset((u8*)cur, 0, size);
+ return (u8*)cur;
+ }
+ else
+ {
+ report(ERR, "arena(%P): overflow @ %P x %llu\n", ar, cur, size);
+ return NULL;
+ }
+ }
+}
+
+void ar_pop(arena* ar, usize size)
+{
+ if(!ar) return;
+ else ar->cursor = min(ar->base, ar->cursor - size);
+}
+
+void ar_clear(arena* ar)
+{
+ if(ar) ar->cursor = ar->base;
+}
+
+void ar_destroy(arena* ar)
+{
+ if(!ar) return;
+
+ munmap(ar->base, ar->size);
+ ar->base = ar->cursor = NULL;
+ ar->size = 0;
+}
+
diff --git a/sources/meta/exec.c b/sources/meta/exec.c
index b5aaba4..663a2ae 100644
--- a/sources/meta/exec.c
+++ b/sources/meta/exec.c
@@ -34,6 +34,7 @@ BEGIN \
if(c8->st) c8->st--; \
} \
c8->pc += 2; \
+ c8->prevkeys = c8->keys; \
END
#define X_C8_ILL \
@@ -104,12 +105,12 @@ END
#define X_C8_SKP \
BEGIN \
- if(c8->keys & c8->V[x]) c8->pc += 2; \
+ if(c8->keys & bit(c8->V[x])) c8->pc += 2; \
END
#define X_C8_SKNP \
BEGIN \
- if(!(c8->keys & c8->V[x])) c8->pc += 2; \
+ if(!(c8->keys & bit(c8->V[x]))) c8->pc += 2; \
END
#define X_C8_MVDT \
@@ -117,10 +118,19 @@ BEGIN \
c8->V[x] = c8->dt; \
END
-/* TODO: implement */
#define X_C8_LDK \
BEGIN \
- if(!c8->keys) c8->pc -= 2; \
+ int i; \
+ if(c8->keys == c8->prevkeys) c8->pc -= 2; \
+ else \
+ { \
+ for(i = 7; i >= 0; i--) \
+ if(c8->keys & bit(i)) \
+ { \
+ c8->V[x] = i; \
+ break; \
+ } \
+ } \
END
#define X_C8_LDDT \
@@ -227,8 +237,6 @@ BEGIN \
if(c8->V[x] != c8->V[y]) c8->pc += 2; \
END
-
-/* TODO: actually implement */
#define X_C8_DRW \
BEGIN \
u8 l; \
@@ -236,7 +244,11 @@ BEGIN \
u8 cy = c8->V[y]; \
\
for(l = 0; l < n; l++) \
- c8->DISPLAY[cy + l * cx] ^= c8->RAM[c8->i + l]; \
+ { \
+ u64 pix = (u64)c8->RAM[c8->i + l]; \
+ pix = (pix << (64 - cx)) | (pix >> cx); \
+ c8->DISPLAY[(cy + l) % 32] ^= pix; \
+ } \
END
c8_decode_generate(exec)
diff --git a/sources/plat.c b/sources/plat.c
new file mode 100644
index 0000000..afa0f21
--- /dev/null
+++ b/sources/plat.c
@@ -0,0 +1,24 @@
+/*
+ * Copyright (C) 2025 dwlr <dweller@cabin.digital>
+ *
+ * BSD 3-Clause License (BSD-3-Clause)
+ * See LICENSE for details
+ */
+
+
+ssize cpu_count(void)
+{
+ static ssize cpus = 0;
+ if(!cpus) cpus = sysconf(_SC_NPROCESSORS_ONLN);
+
+ return cpus;
+}
+
+ssize mm_pagesize(void)
+{
+ static ssize page_size = 0;
+ if(!page_size) page_size = sysconf(_SC_PAGE_SIZE);
+
+ return page_size;
+}
+
diff --git a/sources/x11.c b/sources/x11.c
index b9f27b3..7d353cb 100644
--- a/sources/x11.c
+++ b/sources/x11.c
@@ -1,3 +1,11 @@
+/*
+ * Copyright (C) 2025 dwlr <dweller@cabin.digital>
+ *
+ * BSD 3-Clause License (BSD-3-Clause)
+ * See LICENSE for details
+ */
+
+
#if true != True
#error "true != Xlib's True"
#endif