2021-01-03 15:59:45 +01:00
|
|
|
/* planck -
|
|
|
|
* Copyright (C) 2021 nineties
|
|
|
|
*/
|
|
|
|
#ifndef INCLUDE_DEFS
|
|
|
|
#define INCLUDE_DEFS
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2021-01-05 18:12:47 +01:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
2021-01-03 15:59:45 +01:00
|
|
|
|
|
|
|
typedef uintptr_t cell;
|
|
|
|
typedef void (**cfa)();
|
|
|
|
|
|
|
|
#define CELL sizeof(cell)
|
|
|
|
#define STACK_SIZE 1024
|
|
|
|
#define RSTACK_SIZE 1024
|
2021-01-05 21:04:19 +01:00
|
|
|
#define MEMORY_SIZE 0x20000
|
2021-01-03 15:59:45 +01:00
|
|
|
|
|
|
|
typedef struct builtin {
|
|
|
|
struct builtin *prev;
|
|
|
|
char len;
|
|
|
|
char name;
|
|
|
|
char padding[CELL-2];
|
|
|
|
void (*fun)();
|
|
|
|
} builtin;
|
|
|
|
|
|
|
|
static cell stack[STACK_SIZE];
|
|
|
|
static cell rstack[RSTACK_SIZE];
|
|
|
|
static cell *dsp = stack + STACK_SIZE;
|
|
|
|
static cell *rsp = rstack + RSTACK_SIZE;
|
|
|
|
|
|
|
|
static cell memory[MEMORY_SIZE];
|
|
|
|
static builtin *latest = 0;
|
|
|
|
static cell *here = memory;
|
|
|
|
static cell *pc = NULL;
|
|
|
|
static cfa ip = NULL;
|
|
|
|
|
|
|
|
#define next() (*(ip = (cfa)(*pc++)))()
|
|
|
|
|
|
|
|
static void push(cell v) { *(--dsp) = v; }
|
|
|
|
static cell pop(void) { return *dsp++; }
|
|
|
|
static void rpush(cell v) { *(--rsp) = v; }
|
|
|
|
static cell rpop(void) { return *rsp++; }
|
|
|
|
|
|
|
|
static void docol(void) {
|
|
|
|
rpush((cell) pc);
|
|
|
|
pc = (cell*)ip + 1;
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
|
|
|
|
static cfa find(char c) {
|
|
|
|
for (builtin *it = latest; it; it = it->prev)
|
|
|
|
if (it->len == 1 && it->name == c)
|
|
|
|
return &it->fun;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-01-05 06:19:53 +01:00
|
|
|
static int saved_argc = 0;
|
|
|
|
static char **saved_argv = 0;
|
|
|
|
|
2021-01-03 15:59:45 +01:00
|
|
|
#define defcode(name, label) \
|
|
|
|
static void label()
|
|
|
|
#include "planck.c"
|
|
|
|
#undef defcode
|
|
|
|
|
2021-01-05 17:51:01 +01:00
|
|
|
static void align() {
|
|
|
|
here = (cell*)((((cell)here) + CELL - 1) & ~(CELL - 1));
|
|
|
|
}
|
|
|
|
static void comma(cell v) { *here++ = v; }
|
|
|
|
static void comma_byte(char c) {
|
|
|
|
*(char*)here = c;
|
|
|
|
here = (cell*)(((char*)here) + 1);
|
|
|
|
}
|
|
|
|
static void comma_string(char *s) {
|
|
|
|
while (*s) comma_byte(*s++);
|
|
|
|
comma_byte(0);
|
|
|
|
}
|
|
|
|
|
2021-01-05 06:19:53 +01:00
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
saved_argc = argc;
|
|
|
|
saved_argv = argv;
|
|
|
|
|
2021-01-05 17:51:01 +01:00
|
|
|
#define defcode(name, label) \
|
|
|
|
comma((cell) latest); \
|
|
|
|
latest = (void*)here - CELL; \
|
|
|
|
comma_byte(strlen(name)); \
|
|
|
|
comma_string(name); \
|
|
|
|
align(); \
|
|
|
|
comma((cell) label); \
|
2021-01-03 15:59:45 +01:00
|
|
|
if (0) // skip function body
|
2021-01-05 17:51:01 +01:00
|
|
|
|
2021-01-03 15:59:45 +01:00
|
|
|
#include "planck.c"
|
|
|
|
|
|
|
|
cfa start = (cfa) here;
|
|
|
|
*here++ = (cell) find('k');
|
|
|
|
*here++ = (cell) find('f');
|
|
|
|
*here++ = (cell) find('x');
|
|
|
|
*here++ = (cell) find('j');
|
|
|
|
*here++ = (cell) -4 * CELL;
|
|
|
|
pc = (cell*) start;
|
|
|
|
next();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else
|
2021-01-05 17:51:01 +01:00
|
|
|
defcode("Q", quit) { exit(0); }
|
|
|
|
defcode("C", cell_) { push(CELL); next(); }
|
|
|
|
defcode("h", here_) { push((cell)&here); next(); }
|
|
|
|
defcode("l", latest_) { push((cell)&latest); next(); }
|
|
|
|
defcode("i", docol_) { push((cell)docol); next(); }
|
|
|
|
defcode("e", exit_) { pc = (cell*)rpop(); next(); }
|
|
|
|
defcode("@", fetch) { cell *p = (cell*)pop(); push(*p); next(); }
|
|
|
|
defcode("!", store) { cell *p = (cell*)pop(); *p = pop(); next(); }
|
|
|
|
defcode("?", cfetch) { char *p = (char*)pop(); push(*p); next(); }
|
|
|
|
defcode("$", cstore) { char *p = (char*)pop(); *p = pop(); next(); }
|
|
|
|
defcode("d", dfetch) { push((cell)dsp); next(); }
|
|
|
|
defcode("D", dstore) { dsp = (cell*) pop(); next(); }
|
|
|
|
defcode("r", rfetch) { push((cell)rsp); next(); }
|
|
|
|
defcode("R", rstore) { rsp = (cell*) pop(); next(); }
|
|
|
|
defcode("j", jump) { pc += (int)*pc/CELL; next(); }
|
|
|
|
defcode("J", jump0) { pc += (int)(pop()?1:*pc/CELL); next(); }
|
|
|
|
defcode("L", lit) { push(*pc++); next(); }
|
|
|
|
defcode("S", litstring) {
|
|
|
|
int n = *pc++;
|
|
|
|
push((cell)pc);
|
|
|
|
pc += (n + CELL - 1)/CELL;
|
|
|
|
next();
|
2021-01-03 15:59:45 +01:00
|
|
|
}
|
2021-01-05 17:51:01 +01:00
|
|
|
defcode("k", key) {
|
2021-01-03 15:59:45 +01:00
|
|
|
int c = getchar();
|
|
|
|
if (c <= 0)
|
|
|
|
exit(0);
|
|
|
|
push(c);
|
|
|
|
next();
|
|
|
|
}
|
2021-01-05 17:51:01 +01:00
|
|
|
defcode("t", type) { putchar(pop()); next(); }
|
|
|
|
defcode("x", exec) { (*(ip = (cfa) pop()))(); }
|
|
|
|
defcode("f", find_) { push((cell) find(pop())); next(); }
|
|
|
|
defcode("v", argv_) { push((cell) saved_argv); push(saved_argc); next(); }
|
|
|
|
#define defbinary(name, label, op, ty) \
|
|
|
|
defcode(name, label) { \
|
2021-01-03 15:59:45 +01:00
|
|
|
ty b = (ty) pop(); \
|
|
|
|
*dsp = (cell)((ty) *dsp op b); \
|
|
|
|
next(); \
|
|
|
|
}
|
2021-01-05 17:51:01 +01:00
|
|
|
defbinary("+", add, +, intptr_t)
|
|
|
|
defbinary("-", sub, -, intptr_t)
|
|
|
|
defbinary("*", mul, *, intptr_t)
|
|
|
|
defbinary("/", div_, /, uintptr_t)
|
|
|
|
defbinary("%", mod, %, uintptr_t)
|
|
|
|
defbinary("&", and, &, uintptr_t)
|
|
|
|
defbinary("|", or, |, uintptr_t)
|
|
|
|
defbinary("^", xor, ^, uintptr_t)
|
|
|
|
defbinary("<", lt, <, intptr_t)
|
|
|
|
defbinary("=", eq, ==, intptr_t)
|
2021-01-05 19:22:01 +01:00
|
|
|
|
|
|
|
/* File IO */
|
|
|
|
#define SUCCESS 0
|
|
|
|
#define CLOSE_FILE_ERROR -62
|
|
|
|
#define OPEN_FILE_ERROR -69
|
|
|
|
#define READ_FILE_ERROR -70
|
|
|
|
#define WRITE_FILE_ERROR -75
|
2021-01-05 18:12:47 +01:00
|
|
|
defcode("(open-file)", openfile) {
|
|
|
|
int flags = pop();
|
|
|
|
char *name = (char*) pop();
|
|
|
|
int fd = open(name, flags);
|
|
|
|
push(fd);
|
2021-01-05 19:22:01 +01:00
|
|
|
push((fd >= 0) ? SUCCESS : OPEN_FILE_ERROR);
|
2021-01-05 18:12:47 +01:00
|
|
|
next();
|
|
|
|
}
|
|
|
|
defcode("(close-file)", closefile) {
|
|
|
|
int fd = pop();
|
|
|
|
int r = close(fd);
|
2021-01-05 19:22:01 +01:00
|
|
|
push((r >= 0) ? SUCCESS : CLOSE_FILE_ERROR);
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
defcode("(read-file)", readfile) {
|
|
|
|
int fd = pop();
|
|
|
|
int size = pop();
|
|
|
|
char *buf = (char*) pop();
|
|
|
|
int r = read(fd, buf, size);
|
|
|
|
push(r);
|
|
|
|
push((r >= 0) ? SUCCESS : READ_FILE_ERROR);
|
|
|
|
next();
|
|
|
|
}
|
|
|
|
defcode("(write-file)", writefile) {
|
|
|
|
int fd = pop();
|
|
|
|
int size = pop();
|
|
|
|
char *buf = (char*) pop();
|
|
|
|
int r = write(fd, buf, size);
|
|
|
|
push((r == size) ? SUCCESS : WRITE_FILE_ERROR);
|
2021-01-05 18:12:47 +01:00
|
|
|
next();
|
|
|
|
}
|
2021-01-03 15:59:45 +01:00
|
|
|
|
|
|
|
#endif
|