#!/usr/bin/env python3 # planck - # Copyright (C) 2021 nineties # import os import sys import operator import array import ctypes import platform RUNTIME_NAME = "Python {}".format(platform.python_version()) COPYRIGHT = "Copyright (c) 2021 Koichi Nakamura " VERSION = "{}:{}".format(RUNTIME_NAME, COPYRIGHT) MEMORY_SIZE = 0x10000 memory = array.array('i', [0]*MEMORY_SIZE) CELL = memory.itemsize STACK_SIZE = 0x100 RSTACK_SIZE = 0x100 HERE_CELL = 0 LATEST_CELL = CELL sp = MEMORY_SIZE * CELL rp = (MEMORY_SIZE - STACK_SIZE) * CELL ip = 0 np = 0 def aligned(n): return (n + CELL - 1) & ~(CELL - 1) def align(): write(HERE_CELL, aligned(read(HERE_CELL))) def read(addr): return memory[addr//CELL] def write(addr, v): memory[addr//CELL] = ctypes.c_int(v).value def comma(v): here = read(HERE_CELL) write(here, v) write(HERE_CELL, here + CELL) def read_byte(addr): i = addr // CELL m = (addr % CELL)*8 v = memory[i] return ctypes.c_int8((v >> m) & 0xff).value # sign extension def write_byte(addr, c): i = addr // CELL m = (addr % CELL)*8 v = memory[i] memory[i] = (v & ~(0xff << m)) | (c&0xff) << m def comma_byte(v): here = read(HERE_CELL) write_byte(here, v) write(HERE_CELL, here + 1) def comma_string(s): for c in s: comma_byte(ord(c)) comma_byte(0) def write_string(addr, s): for c in s: write_byte(addr, c) addr += 1 def read_string(addr): s = "" while True: c = read_byte(addr) if c == 0: break s += chr(c) addr += 1 return s def read_bytes(addr, n): data = [] for i in range(n): data.append(read_byte(addr)) addr += 1 return bytes(data) def find(c): it = read(LATEST_CELL) while it != 0: n = read_byte(it + CELL) C = chr(read_byte(it + CELL + 1)) if (c == C and n == 1): return it + 2*CELL it = read(it) raise Exception('Unknown word: {}'.format(c)) def push(v): global sp sp -= CELL write(sp, v) def pop(): global sp v = read(sp) sp += CELL return v def rpush(v): global rp rp -= CELL write(rp, v) def rpop(): global rp v = read(rp) rp += CELL return v 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): return read(np), np + CELL def add_simple_operator(name, func): def func_(ip, np): func() return next(np) return add_operator(name, func_) def add_uint_operator(name, op): def func(): b = ctypes.c_uint(pop()).value a = ctypes.c_uint(pop()).value push(op(a, b)) return add_simple_operator(name, func) def add_int_operator(name, op): def func(): b = pop() a = pop() push(op(a, b)) return add_simple_operator(name, func) write(HERE_CELL, 2*CELL) write(LATEST_CELL, 0) # 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) # Version String VERSION_ADDR = read(HERE_CELL) comma_string(VERSION) align() def docol(ip, np): rpush(np) return next(ip + CELL) DOCOL_ID = add_operator('', docol) add_simple_operator('Q', lambda: exit(0)) add_simple_operator('C', lambda: push(CELL)) add_simple_operator('h', lambda: push(HERE_CELL)) add_simple_operator('l', lambda: push(LATEST_CELL)) def key(): c = sys.stdin.read(1) if c: push(ord(c)) else: exit(0) add_simple_operator('k', key) add_simple_operator('t', lambda: sys.stdout.write(chr(pop()))) add_operator('j', lambda ip,np: next(np + read(np))) add_operator('J', lambda ip,np: next(np + (CELL if pop() else read(np)))) 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)) return next(np + CELL) add_operator('L', lit) def litstring(ip, np): push(np + CELL) return next(np + CELL + read(np)) add_operator('S', litstring) def divmod(): b = ctypes.c_int(pop()).value a = ctypes.c_int(pop()).value push(a%b) push(a//b) add_simple_operator('/', divmod) add_int_operator('+', operator.add) add_int_operator('-', operator.sub) add_int_operator('*', operator.mul) add_uint_operator('&', operator.and_) add_uint_operator('|', operator.or_) add_uint_operator('^', operator.xor) add_int_operator('<', operator.lt) add_uint_operator('u', operator.lt) add_uint_operator('=', operator.eq) def argv(): push(ARGV_ADDR) push(len(sys.argv)) add_simple_operator('v', argv) add_simple_operator('V', lambda: push(VERSION_ADDR)) SUCCESS = 0 ALLOCATE_ERROR = -59 CLOSE_FILE_ERROR = -62 OPEN_FILE_ERROR = -69 READ_FILE_ERROR = -70 WRITE_FILE_ERROR = -75 def openfile(): flag = pop() name = read_string(pop()) fd = os.open(name, flag) push(fd) def closefile(): fd = pop() os.close(fd) push(0) 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)) push(n) add_simple_operator('(open)', openfile) add_simple_operator('(close)', closefile) add_simple_operator('(write)', writefile) add_simple_operator('(read)', readfile) def allocate(): size = pop() n = (size + CELL - 1) // CELL addr = len(memory) * CELL memory.extend([0]*n) push(addr) add_simple_operator('(allocate)', allocate) start = read(HERE_CELL) comma(find('k')) comma(find('f')) comma(find('x')) comma(find('j')) comma(-4*CELL) ip, np = next(start) while True: ip, np = operators[read(ip)](ip, np)