2022-09-26 23:50:00 +00:00
|
|
|
import io.kotest.assertions.throwables.shouldThrowWithMessage
|
2022-09-21 00:59:36 +00:00
|
|
|
import io.kotest.core.spec.style.FunSpec
|
|
|
|
import io.kotest.matchers.collections.shouldBeEmpty
|
|
|
|
import io.kotest.matchers.shouldBe
|
2022-09-26 17:03:54 +00:00
|
|
|
import prog8.code.core.*
|
|
|
|
import prog8.code.target.VMTarget
|
|
|
|
import prog8.intermediate.*
|
2022-09-21 00:59:36 +00:00
|
|
|
import prog8.vm.VirtualMachine
|
|
|
|
import prog8.vm.VmRunner
|
|
|
|
|
|
|
|
class TestVm: FunSpec( {
|
2022-09-26 17:03:54 +00:00
|
|
|
|
|
|
|
fun getTestOptions(): CompilationOptions {
|
|
|
|
val target = VMTarget()
|
|
|
|
return CompilationOptions(
|
|
|
|
OutputType.RAW,
|
|
|
|
CbmPrgLauncherType.NONE,
|
|
|
|
ZeropageType.DONTUSE,
|
|
|
|
zpReserved = emptyList(),
|
|
|
|
floats = true,
|
|
|
|
noSysInit = false,
|
|
|
|
compTarget = target,
|
|
|
|
loadAddress = target.machine.PROGRAM_LOAD_ADDRESS
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-09-21 00:59:36 +00:00
|
|
|
test("vm execution: empty program") {
|
2022-09-26 17:03:54 +00:00
|
|
|
val program = IRProgram("test", IRSymbolTable(null), getTestOptions(), VMTarget())
|
|
|
|
val vm = VirtualMachine(program)
|
2022-09-21 00:59:36 +00:00
|
|
|
vm.callStack.shouldBeEmpty()
|
|
|
|
vm.valueStack.shouldBeEmpty()
|
2022-10-16 16:30:14 +00:00
|
|
|
vm.pcIndex shouldBe 0
|
2022-09-21 00:59:36 +00:00
|
|
|
vm.stepCount shouldBe 0
|
2022-09-24 14:00:25 +00:00
|
|
|
vm.run()
|
2022-09-21 00:59:36 +00:00
|
|
|
vm.callStack.shouldBeEmpty()
|
|
|
|
vm.valueStack.shouldBeEmpty()
|
2022-10-16 16:30:14 +00:00
|
|
|
vm.pcIndex shouldBe 0
|
|
|
|
vm.stepCount shouldBe 0
|
2022-09-21 00:59:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
test("vm execution: modify memory") {
|
2022-09-26 17:03:54 +00:00
|
|
|
val program = IRProgram("test", IRSymbolTable(null), getTestOptions(), VMTarget())
|
2022-09-26 23:50:00 +00:00
|
|
|
val block = IRBlock("testmain", null, IRBlock.BlockAlignment.NONE, Position.DUMMY)
|
|
|
|
val startSub = IRSubroutine("testmain.testsub", emptyList(), null, Position.DUMMY)
|
2022-10-25 21:51:22 +00:00
|
|
|
val code = IRCodeChunk(startSub.name, Position.DUMMY, null)
|
2022-09-26 23:50:00 +00:00
|
|
|
code += IRInstruction(Opcode.NOP)
|
2022-09-30 13:27:03 +00:00
|
|
|
code += IRInstruction(Opcode.LOAD, IRDataType.WORD, reg1=1, value=12345)
|
|
|
|
code += IRInstruction(Opcode.STOREM, IRDataType.WORD, reg1=1, value=1000)
|
2022-09-26 17:46:44 +00:00
|
|
|
code += IRInstruction(Opcode.RETURN)
|
2022-09-26 17:03:54 +00:00
|
|
|
startSub += code
|
|
|
|
block += startSub
|
|
|
|
program.addBlock(block)
|
|
|
|
val vm = VirtualMachine(program)
|
2022-09-21 00:59:36 +00:00
|
|
|
|
2022-09-26 17:03:54 +00:00
|
|
|
vm.memory.getUW(1000) shouldBe 0u
|
2022-09-21 00:59:36 +00:00
|
|
|
vm.callStack.shouldBeEmpty()
|
|
|
|
vm.valueStack.shouldBeEmpty()
|
2022-10-16 16:30:14 +00:00
|
|
|
vm.pcIndex shouldBe 0
|
2022-09-21 00:59:36 +00:00
|
|
|
vm.stepCount shouldBe 0
|
2022-09-24 14:00:25 +00:00
|
|
|
vm.run()
|
2022-10-16 16:30:14 +00:00
|
|
|
vm.stepCount shouldBe 4
|
2022-09-26 17:03:54 +00:00
|
|
|
vm.memory.getUW(1000) shouldBe 12345u
|
2022-09-21 00:59:36 +00:00
|
|
|
vm.callStack.shouldBeEmpty()
|
|
|
|
vm.valueStack.shouldBeEmpty()
|
2022-10-16 16:30:14 +00:00
|
|
|
vm.pcIndex shouldBe code.instructions.size-1
|
2022-10-04 20:54:14 +00:00
|
|
|
vm.stepCount shouldBe code.instructions.size
|
2022-09-26 23:50:00 +00:00
|
|
|
}
|
|
|
|
|
2022-10-16 16:30:14 +00:00
|
|
|
test("vm asmbinary not supported") {
|
2022-09-27 15:45:26 +00:00
|
|
|
val program = IRProgram("test", IRSymbolTable(null), getTestOptions(), VMTarget())
|
|
|
|
val block = IRBlock("testmain", null, IRBlock.BlockAlignment.NONE, Position.DUMMY)
|
|
|
|
val startSub = IRSubroutine("testmain.testsub", emptyList(), null, Position.DUMMY)
|
2022-10-25 21:51:22 +00:00
|
|
|
val code = IRCodeChunk(startSub.name, Position.DUMMY, null)
|
2022-09-27 15:45:26 +00:00
|
|
|
code += IRInstruction(Opcode.BINARYDATA, binaryData = listOf(1u,2u,3u))
|
|
|
|
code += IRInstruction(Opcode.RETURN)
|
|
|
|
startSub += code
|
|
|
|
block += startSub
|
|
|
|
program.addBlock(block)
|
|
|
|
val vm = VirtualMachine(program)
|
|
|
|
shouldThrowWithMessage<NotImplementedError>("An operation is not implemented: BINARYDATA not yet supported in VM") {
|
|
|
|
vm.run()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-16 16:30:14 +00:00
|
|
|
test("vm asmsub not supported") {
|
2022-09-26 23:50:00 +00:00
|
|
|
val program = IRProgram("test", IRSymbolTable(null), getTestOptions(), VMTarget())
|
|
|
|
val block = IRBlock("main", null, IRBlock.BlockAlignment.NONE, Position.DUMMY)
|
2022-09-30 12:05:11 +00:00
|
|
|
val startSub = IRAsmSubroutine(
|
|
|
|
"main.asmstart",
|
|
|
|
Position.DUMMY,
|
|
|
|
0x2000u,
|
|
|
|
emptySet(),
|
|
|
|
emptyList(),
|
|
|
|
emptyList(),
|
|
|
|
true,
|
|
|
|
"inlined asm here"
|
|
|
|
)
|
2022-09-26 23:50:00 +00:00
|
|
|
block += startSub
|
|
|
|
program.addBlock(block)
|
2022-09-27 01:32:39 +00:00
|
|
|
shouldThrowWithMessage<IRParseException>("vm currently does not support asmsubs: main.asmstart") {
|
2022-09-26 23:50:00 +00:00
|
|
|
VirtualMachine(program)
|
|
|
|
}
|
2022-09-21 00:59:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
test("vmrunner") {
|
|
|
|
val runner = VmRunner()
|
2022-09-26 17:03:54 +00:00
|
|
|
val irSource="""<PROGRAM NAME=test>
|
|
|
|
<OPTIONS>
|
|
|
|
</OPTIONS>
|
|
|
|
|
2022-09-30 13:27:03 +00:00
|
|
|
<ASMSYMBOLS>
|
|
|
|
</ASMSYMBOLS>
|
|
|
|
|
2022-09-26 17:03:54 +00:00
|
|
|
<VARIABLES>
|
|
|
|
</VARIABLES>
|
|
|
|
|
|
|
|
<MEMORYMAPPEDVARIABLES>
|
|
|
|
</MEMORYMAPPEDVARIABLES>
|
|
|
|
|
|
|
|
<MEMORYSLABS>
|
|
|
|
</MEMORYSLABS>
|
|
|
|
|
|
|
|
<INITGLOBALS>
|
|
|
|
</INITGLOBALS>
|
|
|
|
|
|
|
|
<BLOCK NAME=main ADDRESS=null ALIGN=NONE POS=[unittest: line 42 col 1-9]>
|
|
|
|
</BLOCK>
|
|
|
|
</PROGRAM>
|
|
|
|
"""
|
|
|
|
runner.runProgram(irSource)
|
2022-09-21 00:59:36 +00:00
|
|
|
}
|
|
|
|
})
|