forked from carbonblack/binee
-
Notifications
You must be signed in to change notification settings - Fork 0
/
main.go
139 lines (120 loc) · 3.86 KB
/
main.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
// Package main is the main entry point into using Binee. Provides all
// parameterized options passed in via command line
package main
import (
"flag"
"fmt"
"log"
"github.com/carbonblack/binee/pefile"
"github.com/carbonblack/binee/util"
"github.com/carbonblack/binee/windows"
)
func main() {
isAPISetLookup := flag.String("a", "", "get the real dll name from an apiset name")
listAllAPISets := flag.Bool("A", false, "list all apisets and their mappings")
showDLL := flag.Bool("d", false, "show the dll prfix on all function calls")
configFilePath := flag.String("c", "", "path to configuration file")
listExports := flag.Bool("e", false, "dump pe file's exports table")
listImports := flag.Bool("i", false, "dump a pe file's imports table")
outputJSON := flag.Bool("j", false, "output data as json")
instructionLog := flag.Bool("l", false, "log instructions to a []*Instruction slice, typically this is for programmatic emulation")
verbose2 := flag.Bool("vv", false, "verbose level 2")
verbose1 := flag.Bool("v", false, "verbose level 1")
runDLLMain := flag.Bool("m", false, "call DLLMain while loading DLLs")
rootFolder := flag.String("r", "os/win10_32/", "root path of mock file system, defaults to ./os/win10_32")
maxTicks := flag.Int64("t", 0, "maximum number of instructions to emulate before stopping emulation, default is 0 and will run forever or until other stopping event")
flag.Parse()
verboseLevel := 0
if *verbose1 {
verboseLevel = 1
}
if *verbose2 {
verboseLevel = 2
}
// if apiset dump option, load apisetschema.dll and dump all apisets
if *listAllAPISets {
if *configFilePath != "" {
conf, err := util.ReadGenericConfig(*configFilePath)
if err != nil {
log.Fatal(err)
}
rootFolder = &conf.Root
}
path, err := util.SearchFile([]string{"C:\\Windows\\System32", *rootFolder + "windows/system32"}, "apisetschema.dll")
if err != nil {
log.Fatal(err)
}
apiset, _ := pefile.LoadPeFile(path)
for k, v := range apiset.Apisets {
fmt.Println(k, v)
}
return
}
// if apiset lookup, load apisetschema.dll and look up the apiset name
if *isAPISetLookup != "" {
if *configFilePath != "" {
conf, err := util.ReadGenericConfig(*configFilePath)
if err != nil {
log.Fatal(err)
}
rootFolder = &conf.Root
}
path, err := util.SearchFile([]string{"C:\\Windows\\System32", *rootFolder + "windows/system32"}, "apisetschema.dll")
if err != nil {
log.Fatal(err)
}
apiset, _ := pefile.LoadPeFile(path)
lookup := (*isAPISetLookup)[0 : len(*isAPISetLookup)-6]
if apiset.Apisets[lookup] != nil {
for i := 0; i < len(apiset.Apisets[lookup]); i++ {
fmt.Println(" ", apiset.Apisets[lookup][i])
}
} else {
fmt.Println("apiset not found.")
}
return
}
// quit if no binary is passed in
if flag.NArg() == 0 {
flag.PrintDefaults()
return
}
// print the binaries import table
if *listImports {
if pe, err := pefile.LoadPeFile(flag.Arg(0)); err == nil {
for _, importInfo := range pe.Imports {
fmt.Printf("%s.%s => 0x%x\n", importInfo.DllName, importInfo.FuncName, importInfo.Offset)
}
}
return
}
// print the binaries export table
if *listExports {
if pe, err := pefile.LoadPeFile(flag.Arg(0)); err == nil {
for _, export := range pe.Exports {
fmt.Println(export.Name)
}
}
return
}
options := windows.InitWinEmulatorOptions()
options.VerboseLevel = verboseLevel
options.ConfigPath = *configFilePath
options.RootFolder = *rootFolder
options.ShowDLL = *showDLL
options.RunDLLMain = *runDLLMain
if *outputJSON {
options.LogType = windows.LogTypeJSON
} else if *instructionLog {
options.LogType = windows.LogTypeSlice
} else {
options.LogType = windows.LogTypeStdout
}
options.MaxTicks = *maxTicks
// now start the emulator with the various options
emu, err := windows.Load(flag.Arg(0), flag.Args()[1:], options)
if err != nil {
log.Fatal(err)
}
emu.Start()
}