Disassembler skeleton
This commit is contained in:
@@ -21,7 +21,11 @@ func Assemble(src []byte) (*vm.Program, error) {
|
|||||||
return nil, errors.Wrapf(err, "At function index %d\n", f)
|
return nil, errors.Wrapf(err, "At function index %d\n", f)
|
||||||
}
|
}
|
||||||
|
|
||||||
ret.Functions = append(ret.Functions, instrs)
|
newFunc := &vm.Function{
|
||||||
|
Instructions: instrs,
|
||||||
|
}
|
||||||
|
|
||||||
|
ret.Functions = append(ret.Functions, newFunc)
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret, nil
|
return ret, nil
|
||||||
|
|||||||
34
asm/disassemble.go
Normal file
34
asm/disassemble.go
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
package asm
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
import "github.com/firestuff/subcoding/vm"
|
||||||
|
import "github.com/pkg/errors"
|
||||||
|
|
||||||
|
func Disassemble(prog *vm.Program) ([]byte, error) {
|
||||||
|
for f, fnc := range prog.Functions {
|
||||||
|
_, err := disassembleFunction(fnc)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "At function index %d", f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func disassembleFunction(fnc *vm.Function) ([]byte, error) {
|
||||||
|
for i, instr := range fnc.Instructions {
|
||||||
|
_, err := disassembleInstruction(instr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "At instruction index %d", i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func disassembleInstruction(in *vm.Instruction) ([]byte, error) {
|
||||||
|
fmt.Printf("%s\n", nameByOpCode[in.OpCode])
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
@@ -70,6 +70,48 @@ var opCodeByName = map[string]vm.OpCodeType{
|
|||||||
"retf": vm.OpRetF,
|
"retf": vm.OpRetF,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var nameByOpCode = map[vm.OpCodeType]string{
|
||||||
|
vm.OpNop: "nop",
|
||||||
|
|
||||||
|
vm.OpMov: "mov",
|
||||||
|
|
||||||
|
vm.OpAdd: "add",
|
||||||
|
vm.OpSub: "sub",
|
||||||
|
vm.OpMul: "mul",
|
||||||
|
vm.OpDivU: "divu",
|
||||||
|
vm.OpDivS: "divs",
|
||||||
|
|
||||||
|
vm.OpNot: "not",
|
||||||
|
vm.OpAnd: "and",
|
||||||
|
vm.OpOr: "or",
|
||||||
|
vm.OpXor: "xor",
|
||||||
|
|
||||||
|
vm.OpShR: "shr",
|
||||||
|
vm.OpShL: "shl",
|
||||||
|
|
||||||
|
vm.OpEq: "eq",
|
||||||
|
vm.OpLTU: "ltu",
|
||||||
|
vm.OpLTS: "lts",
|
||||||
|
vm.OpGTU: "gtu",
|
||||||
|
vm.OpGTS: "gts",
|
||||||
|
vm.OpLTEU: "lteu",
|
||||||
|
vm.OpLTES: "ltes",
|
||||||
|
vm.OpGTEU: "gteu",
|
||||||
|
vm.OpGTES: "gtes",
|
||||||
|
|
||||||
|
vm.OpJmp: "jmp",
|
||||||
|
vm.OpJmpT: "jmpt",
|
||||||
|
vm.OpJmpF: "jmpf",
|
||||||
|
|
||||||
|
vm.OpCal: "cal",
|
||||||
|
vm.OpCalT: "calt",
|
||||||
|
vm.OpCalF: "calf",
|
||||||
|
|
||||||
|
vm.OpRet: "ret",
|
||||||
|
vm.OpRetT: "rett",
|
||||||
|
vm.OpRetF: "retf",
|
||||||
|
}
|
||||||
|
|
||||||
type operandType int
|
type operandType int
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
|||||||
20
test/disasm_test.go
Normal file
20
test/disasm_test.go
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
package test
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
import "github.com/firestuff/subcoding/asm"
|
||||||
|
|
||||||
|
func TestRoundTrip(t *testing.T) {
|
||||||
|
prog, err := asm.AssembleString(`
|
||||||
|
functions:
|
||||||
|
- - [nop]
|
||||||
|
`)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = asm.Disassemble(prog)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
5
vm/function.go
Normal file
5
vm/function.go
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
package vm
|
||||||
|
|
||||||
|
type Function struct {
|
||||||
|
Instructions []*Instruction
|
||||||
|
}
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
package vm
|
package vm
|
||||||
|
|
||||||
type Program struct {
|
type Program struct {
|
||||||
Functions [][]*Instruction
|
Functions []*Function
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ func (state *State) stackFrame() *stackFrame {
|
|||||||
return state.stack[len(state.stack)-1]
|
return state.stack[len(state.stack)-1]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (state *State) function() []*Instruction {
|
func (state *State) function() *Function {
|
||||||
return state.program.Functions[state.functionIndex]
|
return state.program.Functions[state.functionIndex]
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,7 +53,7 @@ func (state *State) setError(err error) {
|
|||||||
|
|
||||||
func (state *State) setHandlers() {
|
func (state *State) setHandlers() {
|
||||||
for _, fnc := range state.program.Functions {
|
for _, fnc := range state.program.Functions {
|
||||||
for _, instr := range fnc {
|
for _, instr := range fnc.Instructions {
|
||||||
handler, found := opHandlers[instr.OpCode]
|
handler, found := opHandlers[instr.OpCode]
|
||||||
if !found {
|
if !found {
|
||||||
state.setError(fmt.Errorf("Invalid OpCode: 0x%08x", instr.OpCode))
|
state.setError(fmt.Errorf("Invalid OpCode: 0x%08x", instr.OpCode))
|
||||||
@@ -67,7 +67,7 @@ func (state *State) setHandlers() {
|
|||||||
|
|
||||||
func (state *State) processInstruction() {
|
func (state *State) processInstruction() {
|
||||||
fnc := state.function()
|
fnc := state.function()
|
||||||
instr := fnc[state.instructionIndex]
|
instr := fnc.Instructions[state.instructionIndex]
|
||||||
state.instructionIndex += 1
|
state.instructionIndex += 1
|
||||||
instr.opHandler(state, instr)
|
instr.opHandler(state, instr)
|
||||||
|
|
||||||
@@ -75,7 +75,7 @@ func (state *State) processInstruction() {
|
|||||||
state.ret()
|
state.ret()
|
||||||
}
|
}
|
||||||
|
|
||||||
if state.instructionIndex >= int64(len(fnc)) {
|
if state.instructionIndex >= int64(len(fnc.Instructions)) {
|
||||||
state.ret()
|
state.ret()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user