Skip to content

Commit

Permalink
Add 'RegBinary' behavioural signature to allow detection of PE images…
Browse files Browse the repository at this point in the history
… written to registry (for #164)
  • Loading branch information
kevoreilly committed Oct 17, 2018
1 parent b6536d8 commit 3a41983
Showing 1 changed file with 68 additions and 40 deletions.
108 changes: 68 additions & 40 deletions modules/signatures/CAPE.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,48 @@

PLUGX_SIGNATURE = 0x5658

def IsPEImage(buf, size):
dos_header = buf[:DOS_HEADER_LIMIT]
nt_headers = None

if size < PE_HEADER_LIMIT:
return False

# Check for sane value in e_lfanew
e_lfanew, = struct.unpack("<L", dos_header[60:64])
if not e_lfanew or e_lfanew > PE_HEADER_LIMIT:
offset = 0
while offset < PE_HEADER_LIMIT-86:
machine_probe = struct.unpack("<H", buf[offset:offset+2])[0]
if machine_probe == IMAGE_FILE_MACHINE_I386 or machine_probe == IMAGE_FILE_MACHINE_AMD64:
nt_headers = buf[offset-4:offset+252]
break
offset = offset + 2
else:
nt_headers = buf[e_lfanew:e_lfanew+256]

if nt_headers == None:
return False

#if ((pNtHeader->FileHeader.Machine == 0) || (pNtHeader->FileHeader.SizeOfOptionalHeader == 0 || pNtHeader->OptionalHeader.SizeOfHeaders == 0))
if struct.unpack("<H", nt_headers[4:6]) == 0 or struct.unpack("<H", nt_headers[20:22]) == 0 or struct.unpack("<H", nt_headers[84:86]) == 0:
return False

#if (!(pNtHeader->FileHeader.Characteristics & IMAGE_FILE_EXECUTABLE_IMAGE))
if (struct.unpack("<H", nt_headers[22:24])[0] & IMAGE_FILE_EXECUTABLE_IMAGE) == 0:
return False

#if (pNtHeader->FileHeader.SizeOfOptionalHeader & (sizeof (ULONG_PTR) - 1))
if struct.unpack("<H", nt_headers[20:22])[0] & 3 != 0:
return False

#if ((pNtHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR32_MAGIC) && (pNtHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR64_MAGIC))
if struct.unpack("<H", nt_headers[24:26])[0] != OPTIONAL_HEADER_MAGIC_PE and struct.unpack("<H", nt_headers[24:26])[0] != OPTIONAL_HEADER_MAGIC_PE_PLUS:
return False

# To pass the above tests it should now be safe to assume it's a PE image
return True

class CAPE_Compression(Signature):
name = "Compression"
description = "Behavioural detection: Decompression of executable module(s)."
Expand All @@ -51,51 +93,37 @@ def on_call(self, call, process):
if call["api"] == "RtlDecompressBuffer":
buf = self.get_raw_argument(call, "UncompressedBuffer")
size = int(self.get_raw_argument(call, "UncompressedBufferLength"), 0)
dos_header = buf[:DOS_HEADER_LIMIT]
nt_headers = None
self.compressed_binary = IsPEImage(buf, size)

if size < PE_HEADER_LIMIT:
return

# Check for sane value in e_lfanew
e_lfanew, = struct.unpack("<L", dos_header[60:64])
if not e_lfanew or e_lfanew > PE_HEADER_LIMIT:
offset = 0
while offset < PE_HEADER_LIMIT-86:
machine_probe = struct.unpack("<H", buf[offset:offset+2])[0]
if machine_probe == IMAGE_FILE_MACHINE_I386 or machine_probe == IMAGE_FILE_MACHINE_AMD64:
nt_headers = buf[offset-4:offset+252]
break
offset = offset + 2
else:
nt_headers = buf[e_lfanew:e_lfanew+256]

if nt_headers == None:
return

#if ((pNtHeader->FileHeader.Machine == 0) || (pNtHeader->FileHeader.SizeOfOptionalHeader == 0 || pNtHeader->OptionalHeader.SizeOfHeaders == 0))
if struct.unpack("<H", nt_headers[4:6]) == 0 or struct.unpack("<H", nt_headers[20:22]) == 0 or struct.unpack("<H", nt_headers[84:86]) == 0:
return

#if (!(pNtHeader->FileHeader.Characteristics & IMAGE_FILE_EXECUTABLE_IMAGE))
if (struct.unpack("<H", nt_headers[22:24])[0] & IMAGE_FILE_EXECUTABLE_IMAGE) == 0:
return

#if (pNtHeader->FileHeader.SizeOfOptionalHeader & (sizeof (ULONG_PTR) - 1))
if struct.unpack("<H", nt_headers[20:22])[0] & 3 != 0:
return

#if ((pNtHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR32_MAGIC) && (pNtHeader->OptionalHeader.Magic != IMAGE_NT_OPTIONAL_HDR64_MAGIC))
if struct.unpack("<H", nt_headers[24:26])[0] != OPTIONAL_HEADER_MAGIC_PE and struct.unpack("<H", nt_headers[24:26])[0] != OPTIONAL_HEADER_MAGIC_PE_PLUS:
return

# To pass the above tests it should now be safe to assume it's a PE image
self.compressed_binary = True

def on_complete(self):
if self.compressed_binary == True:
return True

class CAPE_RegBinary(Signature):
name = "RegBinary"
description = "Behavioural detection: PE binary written to registry."
severity = 1
categories = ["malware"]
authors = ["kevoreilly"]
minimum = "1.3"
evented = True

filter_apinames = set(["RegSetValueExA", "RegSetValueExW", "RegCreateKeyExA", "RegCreateKeyExW"])

def __init__(self, *args, **kwargs):
Signature.__init__(self, *args, **kwargs)
self.reg_binary = False

def on_call(self, call, process):
if call["api"] == "RegSetValueExA" or call["api"] == "RegSetValueExW":
buf = self.get_raw_argument(call, "Buffer")
size = self.get_raw_argument(call, "BufferLength")
self.reg_binary = IsPEImage(buf, size)

def on_complete(self):
if self.reg_binary == True:
return True

class CAPE_Extraction(Signature):
name = "Extraction"
description = "Behavioural detection: Executable code extraction"
Expand Down

2 comments on commit 3a41983

@enzok
Copy link
Contributor

@enzok enzok commented on 3a41983 Oct 17, 2018

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ERROR:lib.cuckoo.core.plugins:Failed to run signature RegBinary: 'int' object has no attribute '__getitem__'
Traceback (most recent call last):
  File "/opt/cuckoo/utils/../lib/cuckoo/core/plugins.py", line 414, in run
    result = sig.on_call(call, proc)
  File "/opt/cuckoo/utils/../modules/signatures/CAPE.py", line 121, in on_call
    self.reg_binary = IsPEImage(buf, size)
  File "/opt/cuckoo/utils/../modules/signatures/CAPE.py", line 36, in IsPEImage
    dos_header = buf[:DOS_HEADER_LIMIT]
TypeError: 'int' object has no attribute '__getitem__'
2018-10-17 14:27:46,310 [lib.cuckoo.core.plugins] ERROR: Failed to run signature RegBinary: 'int' object has no attribute '__getitem__'
Traceback (most recent call last):
  File "/opt/cuckoo/utils/../lib/cuckoo/core/plugins.py", line 414, in run
    result = sig.on_call(call, proc)
  File "/opt/cuckoo/utils/../modules/signatures/CAPE.py", line 121, in on_call
    self.reg_binary = IsPEImage(buf, size)
  File "/opt/cuckoo/utils/../modules/signatures/CAPE.py", line 36, in IsPEImage
    dos_header = buf[:DOS_HEADER_LIMIT]
TypeError: 'int' object has no attribute '__getitem__'

VT sample: 22e74d1a3d2044c813d816b1766cb24ded03f3388397a9e4ca2ddf1b7b0a1582 (Smokeloader)

Writing something large to the registry, probably not a PE.

@enzok
Copy link
Contributor

@enzok enzok commented on 3a41983 Oct 17, 2018

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

e5e70439c88a07277414d973b11f4e572ca89cbf2b14987b4093cbf5d78c52a7

This sample writes a script w/ a binary.

Please sign in to comment.