delve/pkg/proc/i386_disasm.go

79 lines
3.1 KiB
Go

// TODO: disassembler support should be compiled in unconditionally,
// instead of being decided by the build-target architecture, and be
// part of the Arch object instead.
package proc
import (
"github.com/go-delve/delve/pkg/dwarf/op"
"github.com/go-delve/delve/pkg/dwarf/regnum"
"golang.org/x/arch/x86/x86asm"
)
func i386AsmDecode(asmInst *AsmInstruction, mem []byte, regs *op.DwarfRegisters, memrw MemoryReadWriter, bi *BinaryInfo) error {
return x86AsmDecode(asmInst, mem, regs, memrw, bi, 32)
}
// Possible stacksplit prologues are inserted by stacksplit in
// $GOROOT/src/cmd/internal/obj/x86/obj6.go.
// If 386 on linux when pie, the stacksplit prologue begin with `call __x86.get_pc_thunk.` sometime.
var prologuesI386 []opcodeSeq
func init() {
var i386GetPcIns = opcodeSeq{uint64(x86asm.CALL)}
var tinyStacksplit = opcodeSeq{uint64(x86asm.CMP), uint64(x86asm.JBE)}
var smallStacksplit = opcodeSeq{uint64(x86asm.LEA), uint64(x86asm.CMP), uint64(x86asm.JBE)}
var bigStacksplit = opcodeSeq{uint64(x86asm.MOV), uint64(x86asm.CMP), uint64(x86asm.JE), uint64(x86asm.LEA), uint64(x86asm.SUB), uint64(x86asm.CMP), uint64(x86asm.JBE)}
var unixGetG = opcodeSeq{uint64(x86asm.MOV), uint64(x86asm.MOV)}
prologuesI386 = make([]opcodeSeq, 0, 2*3)
for _, getPcIns := range []opcodeSeq{{}, i386GetPcIns} {
for _, getG := range []opcodeSeq{unixGetG} { // TODO(chainhelen), need to support other OSs.
for _, stacksplit := range []opcodeSeq{tinyStacksplit, smallStacksplit, bigStacksplit} {
prologue := make(opcodeSeq, 0, len(getPcIns)+len(getG)+len(stacksplit))
prologue = append(prologue, getPcIns...)
prologue = append(prologue, getG...)
prologue = append(prologue, stacksplit...)
prologuesI386 = append(prologuesI386, prologue)
}
}
}
}
var i386AsmRegisters = map[int]asmRegister{
// 8-bit
int(x86asm.AL): {regnum.I386_Eax, 0, mask8},
int(x86asm.CL): {regnum.I386_Ecx, 0, mask8},
int(x86asm.DL): {regnum.I386_Edx, 0, mask8},
int(x86asm.BL): {regnum.I386_Ebx, 0, mask8},
int(x86asm.AH): {regnum.I386_Eax, 8, mask8},
int(x86asm.CH): {regnum.I386_Ecx, 8, mask8},
int(x86asm.DH): {regnum.I386_Edx, 8, mask8},
int(x86asm.BH): {regnum.I386_Ebx, 8, mask8},
int(x86asm.SPB): {regnum.I386_Esp, 0, mask8},
int(x86asm.BPB): {regnum.I386_Ebp, 0, mask8},
int(x86asm.SIB): {regnum.I386_Esi, 0, mask8},
int(x86asm.DIB): {regnum.I386_Edi, 0, mask8},
// 16-bit
int(x86asm.AX): {regnum.I386_Eax, 0, mask16},
int(x86asm.CX): {regnum.I386_Ecx, 0, mask16},
int(x86asm.DX): {regnum.I386_Edx, 0, mask16},
int(x86asm.BX): {regnum.I386_Ebx, 0, mask16},
int(x86asm.SP): {regnum.I386_Esp, 0, mask16},
int(x86asm.BP): {regnum.I386_Ebp, 0, mask16},
int(x86asm.SI): {regnum.I386_Esi, 0, mask16},
int(x86asm.DI): {regnum.I386_Edi, 0, mask16},
// 32-bit
int(x86asm.EAX): {regnum.I386_Eax, 0, mask32},
int(x86asm.ECX): {regnum.I386_Ecx, 0, mask32},
int(x86asm.EDX): {regnum.I386_Edx, 0, mask32},
int(x86asm.EBX): {regnum.I386_Ebx, 0, mask32},
int(x86asm.ESP): {regnum.I386_Esp, 0, mask32},
int(x86asm.EBP): {regnum.I386_Ebp, 0, mask32},
int(x86asm.ESI): {regnum.I386_Esi, 0, mask32},
int(x86asm.EDI): {regnum.I386_Edi, 0, mask32},
}