planckforth/others/planck.py

283 lines
6 KiB
Python
Raw Normal View History

2021-01-04 14:16:12 +01:00
#!/usr/bin/env python3
2021-01-03 15:59:45 +01:00
# planck -
# Copyright (C) 2021 nineties
#
2021-01-05 18:12:47 +01:00
import os
2021-01-03 15:59:45 +01:00
import sys
2021-01-05 07:55:29 +01:00
import operator
2021-01-05 13:06:03 +01:00
2021-01-09 16:00:18 +01:00
IMPLEMENTATION = "Python 3.x"
2021-01-05 21:04:19 +01:00
MEMORY_SIZE = 0x20000
2021-01-03 15:59:45 +01:00
STACK_SIZE = 0x400
RSTACK_SIZE = 0x400
HERE_CELL = 0
2021-01-05 13:06:03 +01:00
LATEST_CELL = 4
memory = [0]*(MEMORY_SIZE>>2)
2021-01-03 15:59:45 +01:00
2021-01-05 13:06:03 +01:00
sp = MEMORY_SIZE>>2
rp = (MEMORY_SIZE - STACK_SIZE)>>2
2021-01-03 15:59:45 +01:00
ip = 0
2021-01-05 07:55:29 +01:00
np = 0
2021-01-03 15:59:45 +01:00
2021-01-05 06:22:10 +01:00
def aligned(n):
2021-01-05 13:06:03 +01:00
return (n + 4 - 1) & ~(4 - 1)
2021-01-05 06:22:10 +01:00
2021-01-05 07:55:29 +01:00
def align():
write(HERE_CELL, aligned(read(HERE_CELL)))
2021-01-03 15:59:45 +01:00
def read(addr):
2021-01-05 13:06:03 +01:00
return memory[addr>>2]
2021-01-03 15:59:45 +01:00
def write(addr, v):
2021-01-05 13:06:03 +01:00
memory[addr>>2] = v
2021-01-03 15:59:45 +01:00
2021-01-05 07:55:29 +01:00
def comma(v):
here = read(HERE_CELL)
write(here, v)
2021-01-05 13:06:03 +01:00
write(HERE_CELL, here + 4)
2021-01-05 07:55:29 +01:00
2021-01-03 15:59:45 +01:00
def read_byte(addr):
2021-01-05 13:06:03 +01:00
i = addr>>2
m = (addr&0x3)*8
v = memory[i]
return (v >> m) & 0xff
def write_byte(addr, c):
i = addr>>2
m = (addr&0x3)*8
v = memory[i]
memory[i] = (v & ~(0xff << m)) | (c&0xff) << m
2021-01-03 15:59:45 +01:00
2021-01-05 07:55:29 +01:00
def comma_byte(v):
2021-01-03 15:59:45 +01:00
here = read(HERE_CELL)
2021-01-05 13:06:03 +01:00
write_byte(here, v)
2021-01-05 07:55:29 +01:00
write(HERE_CELL, here + 1)
2021-01-03 15:59:45 +01:00
2021-01-05 07:55:29 +01:00
def comma_string(s):
2021-01-05 13:06:03 +01:00
for c in s:
comma_byte(ord(c))
comma_byte(0)
2021-01-03 15:59:45 +01:00
2021-01-05 19:22:01 +01:00
def write_string(addr, s):
for c in s:
write_byte(addr, c)
addr += 1
2021-01-05 18:12:47 +01:00
def read_string(addr):
s = ""
while True:
c = read_byte(addr)
if c == 0: break
s += chr(c)
addr += 1
return s
2021-01-05 19:22:01 +01:00
def read_bytes(addr, n):
data = []
for i in range(n):
data.append(read_byte(addr))
addr += 1
return bytes(data)
2021-01-03 15:59:45 +01:00
def find(c):
it = read(LATEST_CELL)
while it != 0:
2021-01-05 13:06:03 +01:00
n = read_byte(it + 4)
C = chr(read_byte(it + 4 + 1))
2021-01-03 15:59:45 +01:00
if (c == C and n == 1):
2021-01-05 13:06:03 +01:00
return it + 2*4
2021-01-03 15:59:45 +01:00
it = read(it)
raise Exception('Unknown word: {}'.format(c))
def push(v):
global sp
2021-01-05 13:06:03 +01:00
sp -= 4
2021-01-03 15:59:45 +01:00
write(sp, v)
def pop():
global sp
v = read(sp)
2021-01-05 13:06:03 +01:00
sp += 4
2021-01-03 15:59:45 +01:00
return v
def rpush(v):
global rp
2021-01-05 13:06:03 +01:00
rp -= 4
2021-01-03 15:59:45 +01:00
write(rp, v)
def rpop():
global rp
v = read(rp)
2021-01-05 13:06:03 +01:00
rp += 4
2021-01-03 15:59:45 +01:00
return v
2021-01-05 07:55:29 +01:00
operators = []
def add_operator(name, func):
funcid = len(operators)
here = read(HERE_CELL)
latest = read(LATEST_CELL)
write(LATEST_CELL, here)
comma(latest)
comma_byte(len(name))
comma_string(name)
align()
comma(funcid)
operators.append(func)
return funcid
def next(np):
2021-01-05 13:06:03 +01:00
return read(np), np + 4
2021-01-05 07:55:29 +01:00
def add_simple_operator(name, func):
def func_(ip, np):
func()
return next(np)
return add_operator(name, func_)
def add_binary_operator(name, op):
def func():
b = pop()
push(op(pop(), b))
return add_simple_operator(name, func)
2021-01-05 13:06:03 +01:00
write(HERE_CELL, 2*4)
2021-01-03 15:59:45 +01:00
write(LATEST_CELL, 0)
2021-01-05 08:05:07 +01:00
# Store command line arguments
argv_addrs = []
for arg in sys.argv:
argv_addrs.append(read(HERE_CELL))
comma_string(arg)
align()
ARGV_ADDR = read(HERE_CELL)
for addr in argv_addrs:
comma(addr)
2021-01-09 15:39:29 +01:00
# Version String
2021-01-09 16:00:18 +01:00
IMPLEMENTATION_ADDR = read(HERE_CELL)
comma_string(IMPLEMENTATION)
2021-01-09 15:39:29 +01:00
align()
2021-01-05 07:55:29 +01:00
def docol(ip, np):
rpush(np)
2021-01-05 13:06:03 +01:00
return next(ip + 4)
2021-01-05 07:55:29 +01:00
DOCOL_ID = add_operator('', docol)
add_simple_operator('Q', lambda: exit(0))
2021-01-05 13:06:03 +01:00
add_simple_operator('C', lambda: push(4))
2021-01-05 07:55:29 +01:00
add_simple_operator('h', lambda: push(HERE_CELL))
add_simple_operator('l', lambda: push(LATEST_CELL))
2021-01-05 08:05:18 +01:00
def key():
c = sys.stdin.read(1)
if c:
push(ord(c))
else:
exit(0)
add_simple_operator('k', key)
2021-01-05 07:55:29 +01:00
add_simple_operator('t', lambda: sys.stdout.write(chr(pop())))
add_operator('j', lambda ip,np: next(np + read(np)))
2021-01-05 13:06:03 +01:00
add_operator('J', lambda ip,np: next(np + (4 if pop() else read(np))))
2021-01-05 07:55:29 +01:00
add_simple_operator('f', lambda: push(find(chr(pop()))))
add_operator('x', lambda ip,np: (pop(), np))
add_simple_operator('@', lambda: push(read(pop())))
# NB: Python evaluates expressions from left to right
# https://docs.python.org/3/reference/expressions.html#evaluation-order
add_simple_operator('!', lambda: write(pop(), pop()))
add_simple_operator('?', lambda: push(read_byte(pop())))
add_simple_operator('$', lambda: write_byte(pop(), pop()))
add_simple_operator('d', lambda: push(sp))
def set_sp():
global sp
sp = pop()
add_simple_operator('D', set_sp)
add_simple_operator('r', lambda: push(rp))
def set_rp():
global rp
rp = pop()
add_simple_operator('R', set_rp)
add_simple_operator('i', lambda: push(DOCOL_ID))
add_operator('e', lambda ip,np: next(rpop()))
def lit(ip, np):
push(read(np))
2021-01-05 13:06:03 +01:00
return next(np + 4)
2021-01-05 07:55:29 +01:00
add_operator('L', lit)
def litstring(ip, np):
2021-01-05 13:06:03 +01:00
push(np + 4)
2021-01-09 13:30:02 +01:00
return next(np + 4 + read(np))
2021-01-05 07:55:29 +01:00
add_operator('S', litstring)
def divmod():
b = pop()
a = pop()
push(a%b)
push(a//b)
add_simple_operator('/', divmod)
2021-01-05 07:55:29 +01:00
add_binary_operator('+', operator.add)
add_binary_operator('-', operator.sub)
add_binary_operator('*', operator.mul)
add_binary_operator('%', operator.mod)
add_binary_operator('&', operator.and_)
add_binary_operator('|', operator.or_)
add_binary_operator('^', operator.xor)
add_binary_operator('<', operator.lt)
add_binary_operator('=', operator.eq)
2021-01-05 08:05:07 +01:00
def argv():
push(ARGV_ADDR)
push(len(sys.argv))
add_simple_operator('v', argv)
2021-01-09 16:00:18 +01:00
add_simple_operator('V', lambda: push(IMPLEMENTATION_ADDR))
2021-01-05 19:22:01 +01:00
SUCCESS = 0
2021-01-05 22:07:08 +01:00
ALLOCATE_ERROR = -59
2021-01-05 19:22:01 +01:00
CLOSE_FILE_ERROR = -62
OPEN_FILE_ERROR = -69
READ_FILE_ERROR = -70
WRITE_FILE_ERROR = -75
2021-01-05 18:12:47 +01:00
def openfile():
flag = pop()
name = read_string(pop())
fd = os.open(name, flag)
push(fd)
def closefile():
fd = pop()
os.close(fd)
2021-01-09 09:21:58 +01:00
push(0)
2021-01-05 19:22:01 +01:00
def readfile():
fd = pop()
size = pop()
addr = pop()
s = os.read(fd, size)
write_string(addr, s)
push(len(s))
def writefile():
fd = pop()
size = pop()
addr = pop()
n = os.write(fd, read_bytes(addr, size))
2021-01-09 09:21:58 +01:00
push(n)
add_simple_operator('(open)', openfile)
add_simple_operator('(close)', closefile)
add_simple_operator('(write)', writefile)
add_simple_operator('(read)', readfile)
2021-01-05 22:07:08 +01:00
def allocate():
size = pop()
n = (size + 4 - 1) // 4
addr = len(memory)*4
2021-01-09 09:21:58 +01:00
memory.extend([0]*n)
2021-01-05 22:07:08 +01:00
push(addr)
2021-01-09 09:21:58 +01:00
add_simple_operator('(allocate)', allocate)
2021-01-05 07:55:29 +01:00
start = read(HERE_CELL)
comma(find('k'))
comma(find('f'))
comma(find('x'))
comma(find('j'))
2021-01-05 13:06:03 +01:00
comma(-4*4)
2021-01-05 07:55:29 +01:00
ip, np = next(start)
2021-01-03 15:59:45 +01:00
while True:
2021-01-05 07:55:29 +01:00
ip, np = operators[read(ip)](ip, np)