Add microcompiler

This commit is contained in:
Gabriel Tofvesson 2019-04-06 23:14:45 +02:00
parent e2025ec55d
commit 2c9e19b6fe

363
microcompiler.kt Normal file
View File

@ -0,0 +1,363 @@
enum class ALU(val value: Int) {
NOP(0b0000), // No operation
MOV(0b0001), // Move from bus to AR
MVN(0b0010), // Move inverse of bus to AR
MVZ(0b0011), // Set AR to zero
ADD(0b0100), // Add bus to AR
SUB(0b0101), // Subtract bus from AR
AND(0b0110), // Bitwise AND with bus and AR
ORR(0b0111), // Bitwise OR with bus and AR
ADN(0b1000), // Add bus to AR without setting flags
LSL(0b1001), // Logical shift left AR
BSL(0b1010), // Shift contents of (AR and HR) left (32-bit shift)
ASR(0b1011), // Arithmetic shift right
BSR(0b1100), // Signed big shift right
LSR(0b1101), // Logical shift right AR
ROL(0b1110), // Rotate AR left
BRL(0b1111); // Rotate ARHR left (32-bit rotate)
companion object {
fun locate(value: Int) = values().first{ it.value == value }
fun matchName(name: String) = values().firstOrNull{ it.name.toLowerCase() == name.toLowerCase() }
}
}
enum class ToBus(val value: Int) {
NONE(0b000),
IR(0b001),
PM(0b010),
PC(0b011),
AR(0b100),
HR(0b101),
GR(0b110),
CONST(0b111);
companion object {
fun locate(value: Int) = values().first{ it.value == value }
fun matchName(name: String) = values().first{ it.name.toLowerCase() == name.toLowerCase() }
}
}
enum class FromBus(val value: Int) {
NONE(0b000),
IR(0b001),
PM(0b010),
PC(0b011),
// No AR
HR(0b101),
GR(0b110),
ASR(0b111);
companion object {
fun locate(value: Int) = values().first{ it.value == value }
}
}
enum class LoopCounter(val value: Int) {
NONE(0b00),
DEC(0b01),
BUS(0b10),
U(0b11);
companion object {
fun locate(value: Int) = values().first{ it.value == value }
}
}
enum class SEQ(val value: Int) {
INC(0b0000),
CAL(0b0110),
RET(0b0111),
BRA(0b0101),
BNZ(0b0100),
BRZ(0b1000),
BRN(0b1001),
BRC(0b1010),
BRO(0b1011),
BLS(0b1100),
BNC(0b1101),
BNO(0b1110),
HALT(0b1111);
companion object {
fun locate(value: Int) = values().first{ it.value == value }
fun matchName(name: String) = values().firstOrNull{ it.name.toLowerCase() == name.toLowerCase() }
}
}
open class MicroInstruction(private val _compiledValue: Int, val isConstInstr: Boolean = false) {
constructor(
aluOP: ALU,
toBus: ToBus,
fromBus: FromBus,
muxControl: Boolean,
increment: Boolean,
lc: LoopCounter,
seq: SEQ
): this((
(aluOP.value shl 21) or
(toBus.value shl 18) or
(fromBus.value shl 15) or
(muxControl.toBit() shl 14) or
(increment.toBit() shl 13) or
(lc.value shl 11) or
(seq.value shl 7)
) and (-1 ushr 7))
constructor(aluOP: ALU, toBus: ToBus): this(aluOP, toBus, FromBus.NONE, false, false, LoopCounter.NONE, SEQ.INC)
constructor(aluOP: ALU, constant: Int): this((aluOP.value shl 21) or (ToBus.CONST.value shl 18) or (constant onlyBits 16), true)
constructor(toBus: ToBus, fromBus: FromBus): this(ALU.NOP, toBus, fromBus, false, false, LoopCounter.NONE, SEQ.INC)
constructor(toBus: ToBus): this(ALU.NOP, toBus, FromBus.NONE, false, false, LoopCounter.BUS, SEQ.INC)
var addressReference: AddressReference? = null
val aluOP: ALU
get() = ALU.locate(_compiledValue ushr 21 onlyBits 4)
val toBus: ToBus
get() = ToBus.locate(_compiledValue ushr 18 onlyBits 2)
val fromBus: FromBus
get() = FromBus.locate(_compiledValue ushr 15 onlyBits 2)
val muxControl = _compiledValue.getBitAt(14)
val increment = _compiledValue.getBitAt(13)
val lc: LoopCounter
get() = LoopCounter.locate(_compiledValue ushr 11 onlyBits 2)
val seq: SEQ
get() = SEQ.locate(_compiledValue ushr 7 onlyBits 4)
val address = _compiledValue onlyBits 7
val usesBus = isConstInstr || toBus != ToBus.NONE
val readsBus = isConstInstr || fromBus != FromBus.NONE
val needsUADR = isConstInstr || (lc == LoopCounter.U) || (seq.value in 0b0100..0b1110)
val compiledValue: Int
get(){
if(addressReference != null && addressReference!!.actualAddress == -1)
throw RuntimeException("Unresolved label reference: ${addressReference!!.labelName}")
return _compiledValue onlyBits 25 or (if(addressReference == null) 0 else addressReference!!.actualAddress)
}
infix fun withReference(ref: AddressReference): MicroInstruction {
addressReference = ref
return this
}
infix fun withReference(name: String): MicroInstruction {
addressReference = AddressReference.makeReference(name)
return this
}
infix fun merge(uInstr: MicroInstruction?): MicroInstruction? {
if(uInstr == null) return this
if(isConstInstr || uInstr.isConstInstr) throw RuntimeException("CONST-instructions are fundamentally un-parallellizable")
if(usesBus && uInstr.usesBus && (toBus != uInstr.toBus)) throw RuntimeException("Instructions cannot share bus (output)") // Instructions can't share bus
if(readsBus && uInstr.readsBus && (fromBus != uInstr.fromBus)) throw RuntimeException("Instructions cannot share bus (input)") // Instructions cannot both read from bus to differing outputs
if(seq != SEQ.INC && uInstr.seq != SEQ.INC && seq != uInstr.seq) throw RuntimeException("SEQ mismatch") // We can merge an INC with something else, but we cannot merge more than this
if(needsUADR && uInstr.needsUADR && address != uInstr.address) throw RuntimeException("uADR mismatch") // Instructions cannot depend on differing uADR values
return MicroInstruction(compiledValue or uInstr.compiledValue)
}
}
enum class Register(val busValue: Int, val canRead: Boolean = true) {
ASR(0b111, false),
IR(0b001),
PM(0b010),
PC(0b011),
AR(0b100),
HR(0b101),
GR(0b110);
companion object {
fun lookup(name: String) = values().first{ it.name.toLowerCase() == name.toLowerCase() }
}
}
class AddressReference{
private val address: Int
val labelName: String?
private constructor(address: Int?, labelName: String?){
this.address = address ?: -1
actualAddress = this.address
this.labelName = labelName
if(address == null && labelName == null) throw RuntimeException("Reference must be accessible by name or address")
}
var actualAddress: Int
private set
fun resolveAddress(target: Int){
if(target > 128 || target < 0) throw RuntimeException("Invalid target address")
if(actualAddress != -1) throw RuntimeException("Address already resolved")
actualAddress = target
}
companion object {
private val registry = ArrayList<AddressReference>()
fun makeReference(address: Int) = AddressReference(address, null)
fun makeReference(labelName: String): AddressReference {
val ref = registry.firstOrNull{ it.labelName == labelName }
if(ref != null) return ref
val reference = AddressReference(null, labelName)
registry.add(reference)
return reference
}
}
}
fun Boolean.toBit() = if(this) 1 else 0
fun Int.toInstruction() = or(0x10000000).toString(16).substring(1)
infix fun Int.onlyBits(bits: Int) = and(-1 ushr (32 - bits))
fun Int.getBitAt(index: Int) = ushr(index).and(1) == 1
fun parseMOV(instr: String): MicroInstruction {
val args = instr.split(" ")
if(args.size != 3) throw RuntimeException("MOV instruction requires two arguments: $instr")
if(args[1] == args[2]) throw RuntimeException("Cannot move from register being moved to: $instr")
val a = Register.lookup(args[1])
if(!a.canRead) throw RuntimeException("Cannot read from register: $instr")
if(args[2] == "lc") return MicroInstruction(ToBus.locate(a.busValue))
val b = Register.lookup(args[2])
if(b == Register.AR) return MicroInstruction(ALU.MOV, ToBus.locate(a.busValue))
return MicroInstruction(ToBus.locate(a.busValue), FromBus.locate(b.busValue))
}
fun readNumber(literal: String, max: Int): Int {
val number = if(literal.startsWith("0") && literal.length > 1){ // Parse special number
if(literal[1] == 'x') Integer.parseInt(literal.substring(2), 16)
else if(literal[1] == 'b') Integer.parseInt(literal.substring(2), 2)
else Integer.parseInt(literal.substring(1), 8)
}else{
Integer.parseInt(literal, 10)
}
if(number > max || number < 0) throw RuntimeException("Value out of range: $literal")
return number
}
fun parseCONST(instr: String): MicroInstruction {
val args = instr.split(" ").toTypedArray()
if(args.size != 2) throw RuntimeException("Unexpected arguments: $instr")
return MicroInstruction(ALU.MOV, readNumber(args[1], 65535))
}
fun parseALU(instr: String): MicroInstruction {
val args = instr.split(" ")
if(args.size != 2) throw RuntimeException("Unexpected arguments: $instr")
println(instr)
val source = Register.lookup(args[1])
if(!source.canRead) throw RuntimeException("Cannot read from source: $instr")
return MicroInstruction(ALU.matchName(args[0])!!, ToBus.locate(source.busValue))
}
fun parseLabelReference(ref: String): AddressReference? {
return if(ref.startsWith("@")) AddressReference.makeReference(ref.substring(1)) else null
}
fun parseCALL(instr: String): MicroInstruction {
val args = instr.split(" ")
if(args.size != 2) throw RuntimeException("Unexpected arguments: $instr")
return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, false, LoopCounter.NONE, SEQ.CAL) withReference parseAddressReference(args[1])
}
fun parseBranch(instr: String): MicroInstruction {
val args = instr.split(" ")
if(args.size != 2) throw RuntimeException("Unexpected arguments: $instr")
return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, false, LoopCounter.NONE, SEQ.matchName(args[0])!!) withReference parseAddressReference(args[1])
}
fun parseLCSet(instr: String): MicroInstruction {
val args = instr.split(" ")
if(args.size != 2) throw RuntimeException("Unexpected arguments: $instr")
return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, false, LoopCounter.U, SEQ.INC) withReference AddressReference.makeReference(readNumber(args[1], 0xFF))
}
fun parseAddressReference(arg: String): AddressReference {
return parseLabelReference(arg) ?: AddressReference.makeReference(readNumber(arg, 0xFF))
}
fun parseInstruction(line: String): MicroInstruction? {
val subInstructions = line.split(";")
if(subInstructions.size == 1){
var shave = subInstructions[0].toLowerCase()
while(shave.startsWith(" ") || shave.startsWith("\t")) shave = shave.substring(1)
while(shave.endsWith(" ") || shave.endsWith("\t")) shave = shave.substring(0, shave.length - 1)
if(shave.length == 0) return null
if(shave.startsWith("mov ")) return parseMOV(shave)
else if(shave.startsWith("const ")) return parseCONST(shave)
else if(shave == "incpc") return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, true, LoopCounter.NONE, SEQ.INC)
else if(shave.indexOf(" ") > 0 && ALU.matchName(shave.substring(0, shave.indexOf(" "))) != null) return parseALU(shave)
else if(shave.startsWith("call ")) return parseCALL(shave)
else if(shave == "ret") return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, false, LoopCounter.NONE, SEQ.RET)
else if(shave.startsWith("b") && shave.indexOf(" ") != -1 && SEQ.matchName(shave.substring(0, shave.indexOf(" "))) != null) return parseBranch(shave)
else if(shave == "halt") return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, false, LoopCounter.NONE, SEQ.HALT)
else if(shave.startsWith("lcset")) return parseLCSet(shave)
else if(shave == "declc") return MicroInstruction(ALU.NOP, ToBus.NONE, FromBus.NONE, false, false, LoopCounter.DEC, SEQ.INC)
else throw RuntimeException("Unknown instruction: $shave")
}else{
var result: MicroInstruction? = null
for(rawInstruction in subInstructions){
val parsed = parseInstruction(rawInstruction)
if(parsed == null) continue
result = parsed merge result
if(result == null) throw RuntimeException("Instructions ($line) could not be merged!")
}
return result!!
}
}
fun main(args: Array<String>){
if(args.size != 1) throw RuntimeException("Bad arguments :(")
val file = java.io.File(args[0])
if(!file.isFile) throw RuntimeException("File not found :(")
val builder = StringBuilder("@u\n")
var currentLine = 0
val insns = ArrayList<MicroInstruction>()
for(line in file.readText().replace("\r", "").split("\n")){
val actualCode = (if(line.indexOf("#") != -1) line.substring(0, line.indexOf("#")) else line).toLowerCase()
if(actualCode.length == 0) continue
if(actualCode.startsWith("$")){
AddressReference.makeReference(actualCode.substring(1)).resolveAddress(currentLine)
continue
}
val instr = parseInstruction(actualCode)
if(instr == null) continue // Blank line
insns.add(instr)
++currentLine
}
for(instr in insns)
builder.append(instr.compiledValue.toInstruction()).append("\n")
print(builder.toString())
}