-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathobjdump-x86.go
171 lines (136 loc) · 3.55 KB
/
objdump-x86.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
package main
import (
"bytes"
"debug/elf"
"encoding/binary"
"fmt"
scv "strconv"
str "strings"
)
type DasOpsX86 struct {
p *DasParser
}
func describeX86Insn(name, args string) string {
desc := Insn_x86_64[name]
if len(desc) > 0 {
return desc
}
// there are two kinds of 'movsd' instructions
if name == "movsd" && str.Contains(args, "%xmm") {
return "Move Scalar Double-FP Values [SSE]"
}
// check 'LOCK' prefix
if str.HasPrefix(name, "lock ") {
return describeX86Insn(name[5:], args)
}
// check suffix for conditional instructions
for _, insn := range CondInsn_x86_64 {
cond := ""
if !str.HasPrefix(name, insn) {
continue
}
for cc, cdesc := range Cond_x86_64 {
if name[len(insn):len(name)] == cc {
cond = cdesc
}
}
if len(cond) > 0 {
return fmt.Sprintf("%s If %s", Insn_x86_64[insn], cond)
}
}
namelen := len(name)
iname := name[:namelen-1]
suffix := name[namelen-1:]
// if it has a size suffix (bwdlq), try to match again without it
if size, ok := Size_x86_64[suffix]; ok {
if name, ok2 := SizeInsn_x86_64[iname]; ok2 {
return str.ReplaceAll(name, "${SIZE}", size)
}
if desc, ok2 := Insn_x86_64[iname]; ok2 {
return desc
}
}
return "unknown"
}
func updateComment(p *DasParser, dl *DasLine) {
if dl.mnemonic == "movsd" && str.Contains(dl.args, "%xmm") {
cmts := str.Split(dl.comment, " ")
ofs, err := scv.ParseInt(cmts[0], 16, 64)
if err != nil || (ofs%8) != 0 {
return
}
var val float64
buf := make([]byte, 8)
p.file.ReadAt(buf, ofs)
b := bytes.NewReader(buf)
if binary.Read(b, binary.LittleEndian, &val) == nil {
dl.comment = fmt.Sprintf("%-8x (%f)", ofs, val)
}
return
}
if dl.mnemonic == "movss" && str.Contains(dl.args, "%xmm") {
cmts := str.Split(dl.comment, " ")
ofs, err := scv.ParseInt(cmts[0], 16, 64)
if err != nil || (ofs%4) != 0 {
return
}
var val float32
buf := make([]byte, 4)
p.file.ReadAt(buf, ofs)
b := bytes.NewReader(buf)
if binary.Read(b, binary.LittleEndian, &val) == nil {
dl.comment = fmt.Sprintf("%-8x (%f)", ofs, val)
}
}
}
func (o DasOpsX86) parseInsn(insn interface{}, sym *elf.Symbol) *DasLine {
dl := new(DasLine)
raw_line := insn.(string)
dl.rawline = raw_line
tmp := str.SplitN(raw_line, " ", 2)
dl.mnemonic = str.TrimSpace(tmp[0])
if str.HasPrefix(dl.mnemonic, "ret") {
dl.optype = OPTYPE_RETURN
}
if len(tmp) == 1 {
return dl
}
dl.args = str.TrimSpace(tmp[1])
// some C++ symbols have '#' in the middle
if !str.HasPrefix(dl.mnemonic, "j") && str.Contains(dl.args, "#") {
tmp = str.SplitN(dl.args, "#", 2)
dl.args = str.TrimSpace(tmp[0])
// format comment lines
cmt := str.SplitN(str.TrimSpace(tmp[1]), " ", 2)
dl.comment = fmt.Sprintf("%-8s %s", cmt[0], cmt[1])
// probably inaccurate symbol
if str.Contains(dl.comment, "+0x") {
updateComment(o.p, dl)
}
}
if str.HasPrefix(dl.mnemonic, "j") ||
str.HasPrefix(dl.mnemonic, "call") {
dl.optype = OPTYPE_BRANCH
tmp = str.SplitN(dl.args, " ", 2)
if len(tmp) == 2 {
dl.target, _ = scv.ParseUint(tmp[0], 16, 64)
dl.args = tmp[1]
// if it's a jump in a same function, just save the offset
if str.HasPrefix(dl.args, sym.Name[0:len(sym.Name)-1]) &&
(dl.args[len(sym.Name)-1] == '+' ||
dl.args[len(sym.Name)-1] == '>') {
dl.args = fmt.Sprintf("%#x", dl.target-uint64(sym.Value))
dl.local = true
}
}
}
return dl
}
func (o DasOpsX86) parsePLT() {
}
func (o DasOpsX86) describe(dl *DasLine) string {
return describeX86Insn(dl.mnemonic, dl.args)
}
func getArchOpsX86(p *DasParser) DasArchOps {
return DasOpsX86{p}
}