mirror of
https://github.com/irmen/prog8.git
synced 2025-01-11 13:29:45 +00:00
tweaks
This commit is contained in:
parent
e39ae3c346
commit
6d7d5a4d13
@ -1043,21 +1043,18 @@ class Petscii {
|
||||
)
|
||||
|
||||
// encoding: from unicode to Petscii/Screencodes (0-255)
|
||||
private val encodingPetsciiLowercase = decodingPetsciiLowercase.mapIndexed { index, unicode -> Pair(unicode, index) }.toMap()
|
||||
private val encodingPetsciiUppercase = decodingPetsciiUppercase.mapIndexed { index, unicode -> Pair(unicode, index) }.toMap()
|
||||
private val encodingScreencodeLowercase = decodingScreencodeLowercase.mapIndexed { index, unicode -> Pair(unicode, index) }.toMap()
|
||||
private val encodingScreencodeUppercase = decodingScreencodeUppercase.mapIndexed { index, unicode -> Pair(unicode, index) }.toMap()
|
||||
private val encodingPetsciiLowercase = decodingPetsciiLowercase.withIndex().associate{it.value to it.index}
|
||||
private val encodingPetsciiUppercase = decodingPetsciiUppercase.withIndex().associate{it.value to it.index}
|
||||
private val encodingScreencodeLowercase = decodingScreencodeLowercase.withIndex().associate{it.value to it.index}
|
||||
private val encodingScreencodeUppercase = decodingScreencodeUppercase.withIndex().associate{it.value to it.index}
|
||||
|
||||
|
||||
fun encodePetscii(text: String, lowercase: Boolean = false): ShortArray {
|
||||
val result = ShortArray(text.length)
|
||||
fun encodePetscii(text: String, lowercase: Boolean = false): List<Short> {
|
||||
val lookup = if(lowercase) encodingPetsciiLowercase else encodingPetsciiUppercase
|
||||
text.forEachIndexed { index, c ->
|
||||
val petscii = lookup[c] ?: throw CompilerException("no Petscii character for '$c'")
|
||||
result[index] = petscii.toShort()
|
||||
return text.map {
|
||||
val petscii = lookup[it] ?: throw CompilerException("no Petscii character for '$it'")
|
||||
petscii.toShort()
|
||||
}
|
||||
|
||||
return result
|
||||
}
|
||||
|
||||
fun decodePetscii(petscii: Iterable<Short>, lowercase: Boolean = false): String {
|
||||
@ -1065,14 +1062,12 @@ class Petscii {
|
||||
return petscii.map { decodeTable[it.toInt()] }.joinToString("")
|
||||
}
|
||||
|
||||
fun encodeScreencode(text: String, lowercase: Boolean = false): ShortArray {
|
||||
val result = ShortArray(text.length)
|
||||
fun encodeScreencode(text: String, lowercase: Boolean = false): List<Short> {
|
||||
val lookup = if(lowercase) encodingScreencodeLowercase else encodingScreencodeUppercase
|
||||
text.forEachIndexed { index, c ->
|
||||
val screencode = lookup[c] ?: throw CompilerException("no Screencode character for '$c'")
|
||||
result[index] = screencode.toShort()
|
||||
return text.map{
|
||||
val screencode = lookup[it] ?: throw CompilerException("no Screencode character for '$it'")
|
||||
screencode.toShort()
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
fun decodeScreencode(screencode: Iterable<Short>, lowercase: Boolean = false): String {
|
||||
|
@ -3,7 +3,7 @@ package il65.stackvm
|
||||
import il65.ast.DataType
|
||||
import il65.compiler.Mflpt5
|
||||
import il65.compiler.Petscii
|
||||
import java.io.FileOutputStream
|
||||
import java.io.File
|
||||
import java.io.PrintWriter
|
||||
import java.util.*
|
||||
import kotlin.math.max
|
||||
@ -13,12 +13,13 @@ enum class Opcode {
|
||||
|
||||
// pushing values on the (evaluation) stack
|
||||
PUSH, // push constant byte value
|
||||
PUSH_LOCAL, // push block-local variable
|
||||
PUSH_VAR, // push a variable
|
||||
DUP, // push topmost value once again
|
||||
|
||||
// popping values off the (evaluation) stack, possibly storing them in another location
|
||||
DISCARD, // discard X bytes from the top of the stack
|
||||
POP_MEM, // pop value into destination memory address
|
||||
POP_LOCAL, // pop value into block-local variable
|
||||
POP_VAR, // pop value into variable
|
||||
|
||||
// numeric and bitwise arithmetic
|
||||
ADD,
|
||||
@ -30,27 +31,27 @@ enum class Opcode {
|
||||
SHL,
|
||||
SHL_MEM,
|
||||
SHL_MEM_W,
|
||||
SHL_LOCAL,
|
||||
SHL_VAR,
|
||||
SHR,
|
||||
SHR_MEM,
|
||||
SHR_MEM_W,
|
||||
SHR_LOCAL,
|
||||
SHR_VAR,
|
||||
ROL,
|
||||
ROL_MEM,
|
||||
ROL_MEM_W,
|
||||
ROL_LOCAL,
|
||||
ROL_VAR,
|
||||
ROR,
|
||||
ROR_MEM,
|
||||
ROR_MEM_W,
|
||||
ROR_LOCAL,
|
||||
ROR_VAR,
|
||||
ROL2,
|
||||
ROL2_MEM,
|
||||
ROL2_MEM_W,
|
||||
ROL2_LOCAL,
|
||||
ROL2_VAR,
|
||||
ROR2,
|
||||
ROR2_MEM,
|
||||
ROR2_MEM_W,
|
||||
ROR2_LOCAL,
|
||||
ROR2_VAR,
|
||||
AND,
|
||||
OR,
|
||||
XOR,
|
||||
@ -62,16 +63,24 @@ enum class Opcode {
|
||||
INC,
|
||||
INC_MEM,
|
||||
INC_MEM_W,
|
||||
INC_LOCAL,
|
||||
INC_VAR,
|
||||
DEC,
|
||||
DEC_MEM,
|
||||
DEC_MEM_W,
|
||||
DEC_LOCAL,
|
||||
DEC_VAR,
|
||||
|
||||
// comparisons (?)
|
||||
|
||||
// branching
|
||||
JUMP,
|
||||
BCS,
|
||||
BCC,
|
||||
//BEQ, // @todo not implemented status flag Z
|
||||
//BNE, // @todo not implemented status flag Z
|
||||
//BVS, // @todo not implemented status flag V
|
||||
//BVC, // @todo not implemented status flag V
|
||||
//BMI, // @todo not implemented status flag N
|
||||
//BPL, // @todo not implemented status flag N
|
||||
|
||||
// subroutine calling
|
||||
CALL,
|
||||
@ -79,16 +88,18 @@ enum class Opcode {
|
||||
SYSCALL,
|
||||
|
||||
// misc
|
||||
SWAP,
|
||||
SEC,
|
||||
CLC,
|
||||
TERMINATE
|
||||
}
|
||||
|
||||
enum class Syscall(val callNr: Short) {
|
||||
WRITECHR(10),
|
||||
WRITESTR(11),
|
||||
WRITE_NUM(12),
|
||||
WRITE_CHAR(13)
|
||||
WRITE_MEMCHR(10), // print a single char from the memory
|
||||
WRITE_MEMSTR(11), // print a 0-terminated petscii string from the memory
|
||||
WRITE_NUM(12), // pop from the evaluation stack and print it as a number
|
||||
WRITE_CHAR(13), // pop from the evaluation stack and print it as a single petscii character
|
||||
WRITE_VAR(14), // print the number or string from the given variable
|
||||
}
|
||||
|
||||
class Memory {
|
||||
@ -148,27 +159,31 @@ class Memory {
|
||||
}
|
||||
|
||||
|
||||
data class Value(val type: DataType, private val value: Number)
|
||||
{
|
||||
data class Value(val type: DataType, private val numericvalue: Number?, val stringvalue: String?=null) {
|
||||
private var byteval: Short? = null
|
||||
private var wordval: Int? = null
|
||||
private var floatval: Double? = null
|
||||
|
||||
init {
|
||||
when(type) {
|
||||
DataType.BYTE -> byteval = (value.toInt() and 255).toShort()
|
||||
DataType.WORD -> wordval = value.toInt() and 65535
|
||||
DataType.FLOAT -> floatval = value.toDouble()
|
||||
else -> throw VmExecutionException("invalid value datatype $type")
|
||||
DataType.BYTE -> byteval = (numericvalue!!.toInt() and 255).toShort()
|
||||
DataType.WORD -> wordval = numericvalue!!.toInt() and 65535
|
||||
DataType.FLOAT -> floatval = numericvalue!!.toDouble()
|
||||
DataType.STR -> if(stringvalue==null) throw VmExecutionException("expect stringvalue for STR type")
|
||||
else -> throw VmExecutionException("invalid datatype $type")
|
||||
}
|
||||
}
|
||||
|
||||
override fun toString(): String {
|
||||
return "$type: $numericvalue $stringvalue"
|
||||
}
|
||||
|
||||
fun numericValue(): Number {
|
||||
return when(type) {
|
||||
DataType.BYTE -> byteval!!
|
||||
DataType.WORD-> wordval!!
|
||||
DataType.FLOAT -> floatval!!
|
||||
else -> throw VmExecutionException("invalid datatype")
|
||||
else -> throw VmExecutionException("invalid datatype for numeric value: $type")
|
||||
}
|
||||
}
|
||||
|
||||
@ -177,7 +192,7 @@ data class Value(val type: DataType, private val value: Number)
|
||||
DataType.BYTE -> byteval!!.toInt()
|
||||
DataType.WORD-> wordval!!
|
||||
DataType.FLOAT -> floatval!!.toInt()
|
||||
else -> throw VmExecutionException("invalid datatype")
|
||||
else -> throw VmExecutionException("invalid datatype for integer value: $type")
|
||||
}
|
||||
}
|
||||
|
||||
@ -360,9 +375,25 @@ data class Value(val type: DataType, private val value: Number)
|
||||
}
|
||||
|
||||
|
||||
data class Instruction(val opcode: Opcode, val args: List<Value>) {
|
||||
data class Instruction(val opcode: Opcode,
|
||||
val args: List<Value> = emptyList(),
|
||||
val subArgAllocations: List<String> = emptyList(),
|
||||
val callLabel: String? = null) {
|
||||
lateinit var next: Instruction
|
||||
var nextAlt: Instruction? = null
|
||||
|
||||
init {
|
||||
if(callLabel!=null) {
|
||||
if(args.size!=subArgAllocations.size)
|
||||
throw VmExecutionException("for $opcode the subArgAllocations size is not the same as the arg list size")
|
||||
}
|
||||
}
|
||||
override fun toString(): String {
|
||||
return if(callLabel==null)
|
||||
"$opcode $args"
|
||||
else
|
||||
"$opcode $callLabel $args $subArgAllocations"
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -377,72 +408,92 @@ private class MyStack<T> : Stack<T>() {
|
||||
|
||||
fun pop2() : Pair<T, T> = Pair(pop(), pop())
|
||||
|
||||
fun printTop(amount: Int) {
|
||||
peek(amount).reversed().forEach { println(" $it") }
|
||||
fun printTop(amount: Int, output: PrintWriter) {
|
||||
peek(amount).reversed().forEach { output.println(" $it") }
|
||||
}
|
||||
}
|
||||
|
||||
class Program {
|
||||
var prog: List<Instruction> = listOf()
|
||||
class Program (prog: List<Instruction>, labels: Map<String, Instruction>, val variables: Map<String, Value>) {
|
||||
|
||||
fun load(prog: List<Instruction>) {
|
||||
this.prog = prog.plus(listOf(
|
||||
Instruction(Opcode.TERMINATE, listOf()),
|
||||
Instruction(Opcode.TERMINATE, listOf())
|
||||
))
|
||||
connectPointers()
|
||||
val program: List<Instruction> = prog.plus(listOf(
|
||||
Instruction(Opcode.TERMINATE, listOf()),
|
||||
Instruction(Opcode.TERMINATE, listOf())
|
||||
))
|
||||
|
||||
init {
|
||||
connect(labels)
|
||||
}
|
||||
|
||||
fun connectPointers() {
|
||||
val it1 = prog.iterator()
|
||||
val it2 = prog.iterator()
|
||||
private fun connect(labels: Map<String, Instruction>) {
|
||||
val it1 = program.iterator()
|
||||
val it2 = program.iterator()
|
||||
it2.next()
|
||||
|
||||
while(it1.hasNext() && it2.hasNext()) {
|
||||
val instr = it1.next()
|
||||
val nextInstr = it2.next()
|
||||
when(instr.opcode) {
|
||||
Opcode.TERMINATE -> instr.next = instr
|
||||
Opcode.CALL -> TODO()
|
||||
Opcode.JUMP -> TODO()
|
||||
Opcode.RETURN -> TODO()
|
||||
Opcode.TERMINATE -> instr.next = instr // won't ever execute a next instruction
|
||||
Opcode.RETURN -> instr.next = instr // kinda a special one, in actuality the return instruction is dynamic
|
||||
Opcode.JUMP -> labels[instr.callLabel]?.let {instr.next=it}
|
||||
Opcode.BCC -> labels[instr.callLabel]?.let {instr.next=it}
|
||||
Opcode.BCS -> labels[instr.callLabel]?.let {instr.next=it}
|
||||
Opcode.CALL -> {
|
||||
val jumpInstr = labels[instr.callLabel] ?: throw VmExecutionException("undefined label: ${instr.callLabel}")
|
||||
instr.next=jumpInstr
|
||||
instr.nextAlt = nextInstr // instruction to return to
|
||||
}
|
||||
else -> instr.next = nextInstr
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class StackVm(val vmOutputFile: String) {
|
||||
|
||||
class StackVm(val traceOutputFile: String?) {
|
||||
val mem = Memory()
|
||||
private val es = MyStack<Value>()
|
||||
private val evalstack = MyStack<Value>() // evaluation stack
|
||||
private val callstack = MyStack<Instruction>() // subroutine call stack (@todo maybe use evalstack as well for this?)
|
||||
private var variables = mutableMapOf<String, Value>() // all variables (set of all vars used by all blocks/subroutines) key = their fully scoped name
|
||||
private var carry: Boolean = false
|
||||
private var program = listOf<Instruction>()
|
||||
private var output = PrintWriter(FileOutputStream(vmOutputFile, true))
|
||||
private var traceOutput = if(traceOutputFile!=null) File(traceOutputFile).printWriter() else null
|
||||
|
||||
fun run(program: Program) {
|
||||
this.program = program.prog
|
||||
this.program = program.program
|
||||
this.variables = program.variables.toMutableMap()
|
||||
var ins = this.program[0]
|
||||
|
||||
output.println("---- new session at ${Date()} ----")
|
||||
try {
|
||||
while (true) {
|
||||
ins = dispatch(ins)
|
||||
|
||||
if(evalstack.size > 128)
|
||||
throw VmExecutionException("too many values on evaluation stack")
|
||||
if(callstack.size > 128)
|
||||
throw VmExecutionException("too many nested/recursive calls")
|
||||
}
|
||||
} catch (x: VmTerminationException) {
|
||||
output.println("---- session ended ----")
|
||||
output.flush()
|
||||
output.close()
|
||||
println("Execution terminated.")
|
||||
println("\n\nExecution terminated.")
|
||||
} finally {
|
||||
traceOutput?.close()
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
fun dispatch(ins: Instruction) : Instruction {
|
||||
println("dispatching: $ins")
|
||||
traceOutput?.println("\n$ins")
|
||||
when (ins.opcode) {
|
||||
Opcode.PUSH -> es.push(ins.args[0])
|
||||
Opcode.DISCARD -> es.pop()
|
||||
Opcode.PUSH -> evalstack.push(ins.args[0])
|
||||
Opcode.DUP -> evalstack.push(evalstack.peek())
|
||||
Opcode.DISCARD -> evalstack.pop()
|
||||
Opcode.SWAP -> {
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(top)
|
||||
evalstack.push(second)
|
||||
}
|
||||
Opcode.POP_MEM -> {
|
||||
val value = es.pop()
|
||||
val value = evalstack.pop()
|
||||
val address = ins.args[0].integerValue()
|
||||
when (value.type) {
|
||||
DataType.BYTE -> mem.setByte(address, value.integerValue().toShort())
|
||||
@ -452,89 +503,98 @@ class StackVm(val vmOutputFile: String) {
|
||||
}
|
||||
}
|
||||
Opcode.ADD -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.add(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.add(top))
|
||||
}
|
||||
Opcode.SUB -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.sub(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.sub(top))
|
||||
}
|
||||
Opcode.MUL -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.mul(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.mul(top))
|
||||
}
|
||||
Opcode.DIV -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.div(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.div(top))
|
||||
}
|
||||
Opcode.POW -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.pow(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.pow(top))
|
||||
}
|
||||
Opcode.NEG -> {
|
||||
val v = es.pop()
|
||||
es.push(v.neg())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.neg())
|
||||
}
|
||||
Opcode.SHL -> {
|
||||
val v = es.pop()
|
||||
es.push(v.shl())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.shl())
|
||||
}
|
||||
Opcode.SHR -> {
|
||||
val v = es.pop()
|
||||
es.push(v.shr())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.shr())
|
||||
}
|
||||
Opcode.ROL -> {
|
||||
val v = es.pop()
|
||||
val v = evalstack.pop()
|
||||
val (result, newCarry) = v.rol(carry)
|
||||
this.carry = newCarry
|
||||
es.push(result)
|
||||
evalstack.push(result)
|
||||
}
|
||||
Opcode.ROL2 -> {
|
||||
val v = es.pop()
|
||||
es.push(v.rol2())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.rol2())
|
||||
}
|
||||
Opcode.ROR -> {
|
||||
val v = es.pop()
|
||||
val v = evalstack.pop()
|
||||
val (result, newCarry) = v.ror(carry)
|
||||
this.carry = newCarry
|
||||
es.push(result)
|
||||
evalstack.push(result)
|
||||
}
|
||||
Opcode.ROR2 -> {
|
||||
val v = es.pop()
|
||||
es.push(v.ror2())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.ror2())
|
||||
}
|
||||
Opcode.AND -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.and(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.and(top))
|
||||
}
|
||||
Opcode.OR -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.or(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.or(top))
|
||||
}
|
||||
Opcode.XOR -> {
|
||||
val (top, second) = es.pop2()
|
||||
es.push(second.xor(top))
|
||||
val (top, second) = evalstack.pop2()
|
||||
evalstack.push(second.xor(top))
|
||||
}
|
||||
Opcode.INV -> {
|
||||
val v = es.pop()
|
||||
es.push(v.inv())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.inv())
|
||||
}
|
||||
Opcode.INC -> {
|
||||
val v = es.pop()
|
||||
es.push(v.inc())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.inc())
|
||||
}
|
||||
Opcode.DEC -> {
|
||||
val v = es.pop()
|
||||
es.push(v.dec())
|
||||
val v = evalstack.pop()
|
||||
evalstack.push(v.dec())
|
||||
}
|
||||
Opcode.SYSCALL -> {
|
||||
val callId = ins.args[0].integerValue().toShort()
|
||||
val syscall = Syscall.values().first { it.callNr == callId }
|
||||
when (syscall) {
|
||||
Syscall.WRITECHR -> output.print(Petscii.decodePetscii(listOf(mem.getByte(ins.args[1].integerValue())), true))
|
||||
Syscall.WRITESTR -> output.print(mem.getString(ins.args[1].integerValue()))
|
||||
Syscall.WRITE_NUM -> output.print(es.pop().numericValue())
|
||||
Syscall.WRITE_CHAR -> output.print(Petscii.decodePetscii(listOf(es.pop().integerValue().toShort()), true))
|
||||
Syscall.WRITE_MEMCHR -> print(Petscii.decodePetscii(listOf(mem.getByte(ins.args[1].integerValue())), true))
|
||||
Syscall.WRITE_MEMSTR -> print(mem.getString(ins.args[1].integerValue()))
|
||||
Syscall.WRITE_NUM -> print(evalstack.pop().numericValue())
|
||||
Syscall.WRITE_CHAR -> print(Petscii.decodePetscii(listOf(evalstack.pop().integerValue().toShort()), true))
|
||||
Syscall.WRITE_VAR -> {
|
||||
val varname = ins.args[1].stringvalue ?: throw VmExecutionException("$syscall expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
when(variable.type) {
|
||||
DataType.BYTE, DataType.WORD, DataType.FLOAT -> print(variable.numericValue())
|
||||
DataType.STR -> print(variable.stringvalue)
|
||||
else -> throw VmExecutionException("invalid datatype")
|
||||
}
|
||||
}
|
||||
else -> throw VmExecutionException("unimplemented syscall $syscall")
|
||||
}
|
||||
}
|
||||
@ -640,22 +700,73 @@ class StackVm(val vmOutputFile: String) {
|
||||
mem.setWord(addr, newValue.integerValue())
|
||||
}
|
||||
|
||||
Opcode.JUMP -> {} // do nothing; the next instruction is wired up already
|
||||
Opcode.CALL -> TODO()
|
||||
Opcode.RETURN -> TODO()
|
||||
Opcode.PUSH_LOCAL -> TODO()
|
||||
Opcode.POP_LOCAL -> TODO()
|
||||
Opcode.INC_LOCAL -> TODO()
|
||||
Opcode.DEC_LOCAL -> TODO()
|
||||
Opcode.SHL_LOCAL -> TODO()
|
||||
Opcode.SHR_LOCAL -> TODO()
|
||||
Opcode.ROL_LOCAL -> TODO()
|
||||
Opcode.ROR_LOCAL -> TODO()
|
||||
Opcode.ROL2_LOCAL -> TODO()
|
||||
Opcode.ROR2_LOCAL -> TODO()
|
||||
Opcode.JUMP -> {} // do nothing; the next instruction is wired up already to the jump target
|
||||
Opcode.BCS -> return if(carry) ins.next else ins.nextAlt!!
|
||||
Opcode.BCC -> return if(carry) ins.nextAlt!! else ins.next
|
||||
Opcode.CALL -> callstack.push(ins.nextAlt)
|
||||
Opcode.RETURN -> return callstack.pop()
|
||||
Opcode.PUSH_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
evalstack.push(variable)
|
||||
}
|
||||
Opcode.POP_VAR -> {
|
||||
val value = evalstack.pop()
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
if(variable.type!=value.type) throw VmExecutionException("value datatype ${value.type} is not the same as variable datatype ${variable.type}")
|
||||
variables[varname] = value
|
||||
}
|
||||
Opcode.SHL_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
variables[varname] = variable.shl()
|
||||
}
|
||||
Opcode.SHR_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
variables[varname] = variable.shr()
|
||||
}
|
||||
Opcode.ROL_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
val (newValue, newCarry) = variable.rol(carry)
|
||||
variables[varname] = newValue
|
||||
carry = newCarry
|
||||
}
|
||||
Opcode.ROR_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
val (newValue, newCarry) = variable.ror(carry)
|
||||
variables[varname] = newValue
|
||||
carry = newCarry
|
||||
}
|
||||
Opcode.ROL2_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
variables[varname] = variable.rol2()
|
||||
}
|
||||
Opcode.ROR2_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
variables[varname] = variable.ror2()
|
||||
}
|
||||
Opcode.INC_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
variables[varname] = variable.inc()
|
||||
}
|
||||
Opcode.DEC_VAR -> {
|
||||
val varname = ins.args[0].stringvalue ?: throw VmExecutionException("${ins.opcode} expects string argument (the variable name)")
|
||||
val variable = variables[varname] ?: throw VmExecutionException("unknown variable: $varname")
|
||||
variables[varname] = variable.dec()
|
||||
}
|
||||
}
|
||||
|
||||
if(traceOutput!=null) {
|
||||
traceOutput?.println(" evalstack (size=${evalstack.size}):")
|
||||
evalstack.printTop(4, traceOutput!!)
|
||||
}
|
||||
println(" es:")
|
||||
es.printTop(5)
|
||||
|
||||
return ins.next
|
||||
}
|
||||
@ -663,47 +774,29 @@ class StackVm(val vmOutputFile: String) {
|
||||
|
||||
|
||||
fun main(args: Array<String>) {
|
||||
val vm = StackVm(vmOutputFile = "vmout.txt")
|
||||
val program = Program()
|
||||
program.load(listOf(
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.BYTE, 100))),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.BYTE, 133))),
|
||||
Instruction(Opcode.ADD, emptyList()),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.BYTE, 3))),
|
||||
Instruction(Opcode.DIV, emptyList()),
|
||||
Instruction(Opcode.INC, emptyList()),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.BYTE, 2))),
|
||||
Instruction(Opcode.SHR, emptyList()),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITESTR.callNr), Value(DataType.WORD, 0x1000))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_NUM.callNr))),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.BYTE, Petscii.encodePetscii("@", true)[0]))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_CHAR.callNr))),
|
||||
|
||||
Instruction(Opcode.CLC, emptyList()),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.WORD, 2))),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.ROR2, emptyList()),
|
||||
Instruction(Opcode.CLC, emptyList()),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.WORD, 16384))),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList()),
|
||||
Instruction(Opcode.ROL2, emptyList())
|
||||
))
|
||||
val vm = StackVm(traceOutputFile = "vmtrace.txt")
|
||||
vm.mem.setString(0x1000, "Hallo!\n")
|
||||
|
||||
val instructions = listOf(
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_VAR.callNr), Value(DataType.STR, null, stringvalue = "main.var1"))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_MEMSTR.callNr), Value(DataType.WORD, 0x1000))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_VAR.callNr), Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.PUSH, listOf(Value(DataType.WORD, 12345))),
|
||||
Instruction(Opcode.POP_VAR, listOf(Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_VAR.callNr), Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.DEC_VAR, listOf(Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_VAR.callNr), Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.SHR_VAR, listOf(Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.SYSCALL, listOf(Value(DataType.BYTE, Syscall.WRITE_VAR.callNr), Value(DataType.STR, null, "main.var2"))),
|
||||
Instruction(Opcode.TERMINATE)
|
||||
)
|
||||
|
||||
val labels = listOf(Pair("derp", instructions[0])).toMap()
|
||||
val variables = mapOf(
|
||||
"main.var1" to Value(DataType.STR, null, "Dit is variabele main.var1!\n"),
|
||||
"main.var2" to Value(DataType.WORD, 9999)
|
||||
)
|
||||
|
||||
val program = Program(instructions, labels, variables)
|
||||
vm.run(program)
|
||||
}
|
@ -210,9 +210,9 @@ class TestPetscii {
|
||||
@Test
|
||||
fun testLowercase() {
|
||||
assertThat(Petscii.encodePetscii("hello WORLD 123 @!£", true), equalTo(
|
||||
shortArrayOf(72, 69, 76, 76, 79, 32, 0xd7, 0xcf, 0xd2, 0xcc, 0xc4, 32, 49, 50, 51, 32, 64, 33, 0x5c)))
|
||||
assertThat(Petscii.encodePetscii("\uf11a", true), equalTo(shortArrayOf(0x12))) // reverse vid
|
||||
assertThat(Petscii.encodePetscii("✓", true), equalTo(shortArrayOf(0xfa)))
|
||||
listOf<Short>(72, 69, 76, 76, 79, 32, 0xd7, 0xcf, 0xd2, 0xcc, 0xc4, 32, 49, 50, 51, 32, 64, 33, 0x5c)))
|
||||
assertThat(Petscii.encodePetscii("\uf11a", true), equalTo(listOf<Short>(0x12))) // reverse vid
|
||||
assertThat(Petscii.encodePetscii("✓", true), equalTo(listOf<Short>(0xfa)))
|
||||
assertFailsWith<CompilerException> { Petscii.encodePetscii("π", true) }
|
||||
assertFailsWith<CompilerException> { Petscii.encodePetscii("♥", true) }
|
||||
|
||||
@ -224,10 +224,10 @@ class TestPetscii {
|
||||
@Test
|
||||
fun testUppercase() {
|
||||
assertThat(Petscii.encodePetscii("HELLO 123 @!£"), equalTo(
|
||||
shortArrayOf(72, 69, 76, 76, 79, 32, 49, 50, 51, 32, 64, 33, 0x5c)))
|
||||
assertThat(Petscii.encodePetscii("\uf11a"), equalTo(shortArrayOf(0x12))) // reverse vid
|
||||
assertThat(Petscii.encodePetscii("♥"), equalTo(shortArrayOf(0xd3)))
|
||||
assertThat(Petscii.encodePetscii("π"), equalTo(shortArrayOf(0xff)))
|
||||
listOf<Short>(72, 69, 76, 76, 79, 32, 49, 50, 51, 32, 64, 33, 0x5c)))
|
||||
assertThat(Petscii.encodePetscii("\uf11a"), equalTo(listOf<Short>(0x12))) // reverse vid
|
||||
assertThat(Petscii.encodePetscii("♥"), equalTo(listOf<Short>(0xd3)))
|
||||
assertThat(Petscii.encodePetscii("π"), equalTo(listOf<Short>(0xff)))
|
||||
assertFailsWith<CompilerException> { Petscii.encodePetscii("✓") }
|
||||
|
||||
assertThat(Petscii.decodePetscii(listOf(72, 0x5c, 0xd3, 0xff)), equalTo("H£♥π"))
|
||||
@ -238,9 +238,9 @@ class TestPetscii {
|
||||
@Test
|
||||
fun testScreencodeLowercase() {
|
||||
assertThat(Petscii.encodeScreencode("hello WORLD 123 @!£", true), equalTo(
|
||||
shortArrayOf(0x08, 0x05, 0x0c, 0x0c, 0x0f, 0x20, 0x57, 0x4f, 0x52, 0x4c, 0x44, 0x20, 0x31, 0x32, 0x33, 0x20, 0x00, 0x21, 0x1c)
|
||||
listOf<Short>(0x08, 0x05, 0x0c, 0x0c, 0x0f, 0x20, 0x57, 0x4f, 0x52, 0x4c, 0x44, 0x20, 0x31, 0x32, 0x33, 0x20, 0x00, 0x21, 0x1c)
|
||||
))
|
||||
assertThat(Petscii.encodeScreencode("✓", true), equalTo(shortArrayOf(0x7a)))
|
||||
assertThat(Petscii.encodeScreencode("✓", true), equalTo(listOf<Short>(0x7a)))
|
||||
assertFailsWith<CompilerException> { Petscii.encodeScreencode("♥", true) }
|
||||
assertFailsWith<CompilerException> { Petscii.encodeScreencode("π", true) }
|
||||
|
||||
@ -252,9 +252,9 @@ class TestPetscii {
|
||||
@Test
|
||||
fun testScreencodeUppercase() {
|
||||
assertThat(Petscii.encodeScreencode("WORLD 123 @!£"), equalTo(
|
||||
shortArrayOf(0x17, 0x0f, 0x12, 0x0c, 0x04, 0x20, 0x31, 0x32, 0x33, 0x20, 0x00, 0x21, 0x1c)))
|
||||
assertThat(Petscii.encodeScreencode("♥"), equalTo(shortArrayOf(0x53)))
|
||||
assertThat(Petscii.encodeScreencode("π"), equalTo(shortArrayOf(0x5e)))
|
||||
listOf<Short>(0x17, 0x0f, 0x12, 0x0c, 0x04, 0x20, 0x31, 0x32, 0x33, 0x20, 0x00, 0x21, 0x1c)))
|
||||
assertThat(Petscii.encodeScreencode("♥"), equalTo(listOf<Short>(0x53)))
|
||||
assertThat(Petscii.encodeScreencode("π"), equalTo(listOf<Short>(0x5e)))
|
||||
assertFailsWith<CompilerException> { Petscii.encodeScreencode("✓") }
|
||||
assertFailsWith<CompilerException> { Petscii.encodeScreencode("hello") }
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user