-
-
Notifications
You must be signed in to change notification settings - Fork 49
/
anomaly.go
218 lines (174 loc) · 8.94 KB
/
anomaly.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
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
// Copyright 2021 Saferwall. All rights reserved.
// Use of this source code is governed by Apache v2 license
// license that can be found in the LICENSE file.
package pe
import (
"encoding/binary"
"time"
)
// Anomalies found in a PE
var (
// AnoPEHeaderOverlapDOSHeader is reported when the PE headers overlaps with the DOS header.
AnoPEHeaderOverlapDOSHeader = "PE header overlaps with DOS header"
// AnoPETimeStampNull is reported when the file header timestamp is 0.
AnoPETimeStampNull = "file header timestamp set to 0"
// AnoPETimeStampFuture is reported when the file header timestamp is more
// than one day ahead of the current date timestamp.
AnoPETimeStampFuture = "file header timestamp set to 0"
// NumberOfSections is reported when number of sections is larger or equal than 10.
AnoNumberOfSections10Plus = "number of sections is 10+"
// AnoNumberOfSectionsNull is reported when sections count's is 0.
AnoNumberOfSectionsNull = "number of sections is 0"
// AnoSizeOfOptionalHeaderNull is reported when size of optional header is 0.
AnoSizeOfOptionalHeaderNull = "size of optional header is 0"
// AnoUncommonSizeOfOptionalHeader32 is reported when size of optional
// header for PE32 is larger than 0xE0.
AnoUncommonSizeOfOptionalHeader32 = "size of optional header is larger than 0xE0 (PE32)"
// AnoUncommonSizeOfOptionalHeader64 is reported when size of optional
// header for PE32+ is larger than 0xF0.
AnoUncommonSizeOfOptionalHeader64 = "size of optional header is larger than 0xF0 (PE32+)"
// AnoAddressOfEntryPointNull is reported when address of entry point is 0.
AnoAddressOfEntryPointNull = "address of entry point is 0"
// AnoAddressOfEPLessSizeOfHeaders is reported when address of entry point
// is smaller than size of headers, the file cannot run under Windows.
AnoAddressOfEPLessSizeOfHeaders = "address of entry point is smaller than size of headers, " +
"the file cannot run under Windows 8"
// AnoImageBaseNull is reported when the image base is null.
AnoImageBaseNull = "image base is 0"
// AnoDanSMagicOffset is reported when the `DanS` magic offset is different than 0x80.
AnoDanSMagicOffset = "`DanS` magic offset is different than 0x80"
// ErrInvalidFileAlignment is reported when file alignment is larger than
// 0x200 and not a power of 2.
ErrInvalidFileAlignment = "FileAlignment larger than 0x200 and not a power of 2"
// ErrInvalidSectionAlignment is reported when file alignment is lesser
// than 0x200 and different from section alignment.
ErrInvalidSectionAlignment = "FileAlignment lesser than 0x200 and different from section alignment"
// AnoMajorSubsystemVersion is reported when MajorSubsystemVersion has a
// value different than the standard 3 --> 6.
AnoMajorSubsystemVersion = "MajorSubsystemVersion is outside 3<-->6 boundary"
// AnonWin32VersionValue is reported when Win32VersionValue is different than 0
AnonWin32VersionValue = "Win32VersionValue is a reserved field, must be set to zero"
// AnoInvalidPEChecksum is reported when the optional header checksum field
// is different from what it should normally be.
AnoInvalidPEChecksum = "optional header checksum is invalid"
// AnoNumberOfRvaAndSizes is reported when NumberOfRvaAndSizes is different than 16.
AnoNumberOfRvaAndSizes = "optional header NumberOfRvaAndSizes != 16"
// AnoReservedDataDirectoryEntry is reported when the last data directory entry is not zero.
AnoReservedDataDirectoryEntry = "last data directory entry is a reserved field, must be set to zero"
// AnoCOFFSymbolsCount is reported when the number of COFF symbols is absurdly high.
AnoCOFFSymbolsCount = "COFF symbols count is absurdly high"
// AnoRelocationEntriesCount is reported when the number of relocation entries is absurdly high.
AnoRelocationEntriesCount = "relocation entries count is absurdly high"
)
// GetAnomalies reportes anomalies found in a PE binary.
// These nomalies does prevent the Windows loader from loading the files but
// is an interesting features for malware analysis.
func (pe *File) GetAnomalies() error {
// ******************** Anomalies in File header ************************
// An application for Windows NT typically has the nine predefined sections
// named: .text, .bss, .rdata, .data, .rsrc, .edata, .idata, .pdata, and
// .debug. Some applications do not need all of these sections, while
// others may define still more sections to suit their specific needs.
// NumberOfSections can be up to 96 under XP.
// NumberOfSections can be up to 65535 under Vista and later.
if pe.NtHeader.FileHeader.NumberOfSections >= 10 {
pe.Anomalies = append(pe.Anomalies, AnoNumberOfSections10Plus)
}
// File header timestamp set to 0.
if pe.NtHeader.FileHeader.TimeDateStamp == 0 {
pe.Anomalies = append(pe.Anomalies, AnoPETimeStampNull)
}
// File header timestamp set to the future.
now := time.Now()
future := uint32(now.Add(24 * time.Hour).Unix())
if pe.NtHeader.FileHeader.TimeDateStamp > future {
pe.Anomalies = append(pe.Anomalies, AnoPETimeStampFuture)
}
// NumberOfSections can be null with low alignment PEs
// and in this case, the values are just checked but not really used (under XP)
if pe.NtHeader.FileHeader.NumberOfSections == 0 {
pe.Anomalies = append(pe.Anomalies, AnoNumberOfSectionsNull)
}
// SizeOfOptionalHeader is not the size of the optional header, but the delta
// between the top of the Optional header and the start of the section table.
// Thus, it can be null (the section table will overlap the Optional Header,
// or can be null when no sections are present)
if pe.NtHeader.FileHeader.SizeOfOptionalHeader == 0 {
pe.Anomalies = append(pe.Anomalies, AnoSizeOfOptionalHeaderNull)
}
// SizeOfOptionalHeader can be bigger than the file
// (the section table will be in virtual space, full of zeroes), but can't be negative.
// Do some check here.
oh32 := ImageOptionalHeader32{}
oh64 := ImageOptionalHeader64{}
// SizeOfOptionalHeader standard value is 0xE0 for PE32.
if pe.Is32 &&
pe.NtHeader.FileHeader.SizeOfOptionalHeader > uint16(binary.Size(oh32)) {
pe.Anomalies = append(pe.Anomalies, AnoUncommonSizeOfOptionalHeader32)
}
// SizeOfOptionalHeader standard value is 0xF0 for PE32+.
if pe.Is64 &&
pe.NtHeader.FileHeader.SizeOfOptionalHeader > uint16(binary.Size(oh64)) {
pe.Anomalies = append(pe.Anomalies, AnoUncommonSizeOfOptionalHeader64)
}
// ***************** Anomalies in Optional header *********************
// Under Windows 8, AddressOfEntryPoint is not allowed to be smaller than
// SizeOfHeaders, except if it's null.
switch pe.Is64 {
case true:
oh64 = pe.NtHeader.OptionalHeader.(ImageOptionalHeader64)
case false:
oh32 = pe.NtHeader.OptionalHeader.(ImageOptionalHeader32)
}
// Use oh for fields which are common for both structures.
oh := oh32
if oh.AddressOfEntryPoint != 0 && oh.AddressOfEntryPoint < oh.SizeOfHeaders {
pe.Anomalies = append(pe.Anomalies, AnoAddressOfEPLessSizeOfHeaders)
}
// AddressOfEntryPoint can be null in DLLs: in this case,
// DllMain is just not called. can be null
if oh.AddressOfEntryPoint == 0 {
pe.Anomalies = append(pe.Anomalies, AnoAddressOfEntryPointNull)
}
// ImageBase can be null, under XP.
// In this case, the binary will be relocated to 10000h
if (pe.Is64 && oh64.ImageBase == 0) ||
(pe.Is32 && oh32.ImageBase == 0) {
pe.Anomalies = append(pe.Anomalies, AnoImageBaseNull)
}
// The msdn states that SizeOfImage must be a multiple of the section
// alignment. This is not a requirement though. Adding it as anomaly.
// Todo: raise an anomaly when SectionAlignment is NULL ?
if oh.SectionAlignment != 0 && oh.SizeOfImage%oh.SectionAlignment != 0 {
pe.Anomalies = append(pe.Anomalies, AnoInvalidSizeOfImage)
}
// For DLLs, MajorSubsystemVersion is ignored until Windows 8. It can have
// any value. Under Windows 8, it needs a standard value (3.10 < 6.30).
if oh.MajorSubsystemVersion < 3 || oh.MajorSubsystemVersion > 6 {
pe.Anomalies = append(pe.Anomalies, AnoMajorSubsystemVersion)
}
// Win32VersionValue officially defined as `reserved` and should be null
// if non null, it overrides MajorVersion/MinorVersion/BuildNumber/PlatformId
// OperatingSystem Versions values located in the PEB, after loading.
if oh.Win32VersionValue != 0 {
pe.Anomalies = append(pe.Anomalies, AnonWin32VersionValue)
}
// Checksums are required for kernel-mode drivers and some system DLLs.
// Otherwise, this field can be 0.
if pe.Checksum() != oh.CheckSum && oh.CheckSum != 0 {
pe.Anomalies = append(pe.Anomalies, AnoInvalidPEChecksum)
}
// This field contains the number of IMAGE_DATA_DIRECTORY entries.
// This field has been 16 since the earliest releases of Windows NT.
if (pe.Is64 && oh64.NumberOfRvaAndSizes == 0xA) ||
(pe.Is32 && oh32.NumberOfRvaAndSizes == 0xA) {
pe.Anomalies = append(pe.Anomalies, AnoNumberOfRvaAndSizes)
}
return nil
}
// addAnomaly appends the given anomaly to the list of anomalies.
func (pe *File) addAnomaly(anomaly string) {
if !stringInSlice(anomaly, pe.Anomalies) {
pe.Anomalies = append(pe.Anomalies, anomaly)
}
}