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
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
|
#include<ntifs.h>
// 定义ssdt的硬编码
unsigned char ssdt_chars[] =
{
0x48, 0x8B, 0x45, 0xB0, 0x48, 0x8B, 0x4D, 0xB8, 0x48, 0x8B,
0x55, 0xC0, 0xFB, 0x48, 0x89, 0x8B, 0x88, 0x00, 0x00, 0x00,
0x89, 0x83, 0x80, 0x00, 0x00, 0x00, 0x0F, 0x1F, 0x84, 0x00,
0x00, 0x00, 0x00, 0x00, 0x48, 0x89, 0xA3, 0x90, 0x00, 0x00,
0x00, 0x8B, 0xF8, 0xC1, 0xEF, 0x07, 0x83, 0xE7, 0x20, 0x25,
0xFF, 0x0F, 0x00, 0x00
};
// 定义一个存储模块信息的结构体
typedef struct _MODULE_INFO
{
PULONG64 ModuleBase;
ULONG64 ModuleSize;
}MODULE_INFO, *PMODULE_INFO;
// 定义一个存储区段信息的结构体
typedef struct _SECTION_INFO
{
PULONG64 sectionAddr;
ULONG64 sectionSize;
}SECTION_INFO, *PSECTION_INFO;
// 定义PE文件中的结构体
//0x40 bytes (sizeof)
typedef struct _IMAGE_DOS_HEADER
{
USHORT e_magic; //0x0
USHORT e_cblp; //0x2
USHORT e_cp; //0x4
USHORT e_crlc; //0x6
USHORT e_cparhdr; //0x8
USHORT e_minalloc; //0xa
USHORT e_maxalloc; //0xc
USHORT e_ss; //0xe
USHORT e_sp; //0x10
USHORT e_csum; //0x12
USHORT e_ip; //0x14
USHORT e_cs; //0x16
USHORT e_lfarlc; //0x18
USHORT e_ovno; //0x1a
USHORT e_res[4]; //0x1c
USHORT e_oemid; //0x24
USHORT e_oeminfo; //0x26
USHORT e_res2[10]; //0x28
LONG e_lfanew; //0x3c
}IMAGE_DOS_HEADER, * PIMAGE_DOS_HEADER;
//0x14 bytes (sizeof)
typedef struct _IMAGE_FILE_HEADER
{
USHORT Machine; //0x0
USHORT NumberOfSections; //0x2
ULONG TimeDateStamp; //0x4
ULONG PointerToSymbolTable; //0x8
ULONG NumberOfSymbols; //0xc
USHORT SizeOfOptionalHeader; //0x10
USHORT Characteristics; //0x12
}IMAGE_FILE_HEADER, * PIMAGE_FILE_HEADER;
//0x8 bytes (sizeof)
typedef struct _IMAGE_DATA_DIRECTORY
{
ULONG VirtualAddress; //0x0
ULONG Size; //0x4
}IMAGE_DATA_DIRECTORY, * PIMAGE_DATA_DIRECTORY;
//0xf0 bytes (sizeof)
typedef struct _IMAGE_OPTIONAL_HEADER64
{
USHORT Magic; //0x0
UCHAR MajorLinkerVersion; //0x2
UCHAR MinorLinkerVersion; //0x3
ULONG SizeOfCode; //0x4
ULONG SizeOfInitializedData; //0x8
ULONG SizeOfUninitializedData; //0xc
ULONG AddressOfEntryPoint; //0x10
ULONG BaseOfCode; //0x14
ULONGLONG ImageBase; //0x18
ULONG SectionAlignment; //0x20
ULONG FileAlignment; //0x24
USHORT MajorOperatingSystemVersion; //0x28
USHORT MinorOperatingSystemVersion; //0x2a
USHORT MajorImageVersion; //0x2c
USHORT MinorImageVersion; //0x2e
USHORT MajorSubsystemVersion; //0x30
USHORT MinorSubsystemVersion; //0x32
ULONG Win32VersionValue; //0x34
ULONG SizeOfImage; //0x38
ULONG SizeOfHeaders; //0x3c
ULONG CheckSum; //0x40
USHORT Subsystem; //0x44
USHORT DllCharacteristics; //0x46
ULONGLONG SizeOfStackReserve; //0x48
ULONGLONG SizeOfStackCommit; //0x50
ULONGLONG SizeOfHeapReserve; //0x58
ULONGLONG SizeOfHeapCommit; //0x60
ULONG LoaderFlags; //0x68
ULONG NumberOfRvaAndSizes; //0x6c
struct _IMAGE_DATA_DIRECTORY DataDirectory[16]; //0x70
}IMAGE_OPTIONAL_HEADER64, * PIMAGE_OPTIONAL_HEADER64;
//0x108 bytes (sizeof)
typedef struct _IMAGE_NT_HEADERS64
{
ULONG Signature; //0x0
struct _IMAGE_FILE_HEADER FileHeader; //0x4
struct _IMAGE_OPTIONAL_HEADER64 OptionalHeader; //0x18
}IMAGE_NT_HEADERS64, * PIMAGE_NT_HEADERS64;
//0x28 bytes (sizeof)
typedef struct _IMAGE_SECTION_HEADER
{
UCHAR Name[8]; //0x0
union
{
ULONG PhysicalAddress; //0x8
ULONG VirtualSize; //0x8
} Misc; //0x8
ULONG VirtualAddress; //0xc
ULONG SizeOfRawData; //0x10
ULONG PointerToRawData; //0x14
ULONG PointerToRelocations; //0x18
ULONG PointerToLinenumbers; //0x1c
USHORT NumberOfRelocations; //0x20
USHORT NumberOfLinenumbers; //0x22
ULONG Characteristics; //0x24
}IMAGE_SECTION_HEADER, * PIMAGE_SECTION_HEADER;
//0x120 bytes (sizeof)
typedef struct _LDR_DATA_TABLE_ENTRY
{
struct _LIST_ENTRY InLoadOrderLinks; //0x0
struct _LIST_ENTRY InMemoryOrderLinks; //0x10
struct _LIST_ENTRY InInitializationOrderLinks; //0x20
VOID* DllBase; //0x30
VOID* EntryPoint; //0x38
ULONG SizeOfImage; //0x40
struct _UNICODE_STRING FullDllName; //0x48
struct _UNICODE_STRING BaseDllName; //0x58
}LDR_DATA_TABLE_ENTRY, * PLDR_DATA_TABLE_ENTRY;
// 定义SSDT结构体
typedef struct _ServiceDescriptorEntry {
PULONG32 ServiceTableBase; //SystemService Dispatch Table 的基地址
ULONG64 ServiceCounterTableBase; //包含着 SSDT 中每个服务被调用次数的计数器。这个计数器一般由 sysenter 更新。
ULONG64 NumberOfServices; //由 ServiceTableBase描述的服务的数目。
PUCHAR ParamTableBase; //包含每个系统服务参数字节数表的基地址-系统服务参数表
}SSDT, * PSSDT;
// 工具函数:通过起始地址及硬编码获取对应地址
PVOID findAddrByOpcode(char* opCode, PVOID start, PVOID end, ULONG codeSize)
{
BOOLEAN findOp = FALSE;
char* p;
char* q;
char* r;
for (p = start; p != end; p++)
{
findOp = TRUE;
for (q = p, r = opCode; q != p + codeSize; q++, r++)
{
if (*q != *r)
{
findOp = FALSE;
break;
}
}
if (findOp == TRUE)
{
break;
}
}
return findOp ? p : NULL;
}
// 根据模块名称获取当前进程的下对应模块的信息
PMODULE_INFO getProcessInfoByName(char* processName, PDRIVER_OBJECT pDrvierObject)
{
ANSI_STRING orignName = { 0 };
RtlInitAnsiString(&orignName, processName);
UNICODE_STRING ModuleName = { 0 };
RtlAnsiStringToUnicodeString(&ModuleName, &orignName, TRUE);
//通过当前驱动对象获取镜像对应结构体
PLDR_DATA_TABLE_ENTRY pLdr = NULL;
PLIST_ENTRY pListEntry = NULL;
PLIST_ENTRY pCurrentListEntry = NULL;
PLDR_DATA_TABLE_ENTRY pCurrentModule = NULL;
pLdr = (PLDR_DATA_TABLE_ENTRY)pDrvierObject->DriverSection;
pListEntry = pLdr->InLoadOrderLinks.Flink;
pCurrentListEntry = pListEntry->Flink;
while (pCurrentListEntry != pListEntry) //前后不相等
{
//获取LDR_DATA_TABLE_ENTRY结构
pCurrentModule = CONTAINING_RECORD(pCurrentListEntry, LDR_DATA_TABLE_ENTRY, InLoadOrderLinks);
if (pCurrentModule->BaseDllName.Buffer != 0)
{
if (!RtlCompareUnicodeString(&ModuleName, &(pCurrentModule->BaseDllName), TRUE))
{
DbgPrint("ModuleName = %wZ ModuleBase = %p ModuleEndBase = %p\r\n",
pCurrentModule->BaseDllName,
pCurrentModule->DllBase,
(LONGLONG)pCurrentModule->DllBase + pCurrentModule->SizeOfImage);
MODULE_INFO moduleInfo = { 0 };
moduleInfo.ModuleBase = pCurrentModule->DllBase;
moduleInfo.ModuleSize = pCurrentModule->SizeOfImage;
return &moduleInfo;
}
}
pCurrentListEntry = pCurrentListEntry->Flink;
}
return NULL;
}
// 根据模块地址及区段名称获取区段信息
void getSectionsAddrByImageBase(PVOID imageBase, char* sectionName, PSECTION_INFO sectionInfo)
{
PIMAGE_DOS_HEADER pImageDosHeader = (PIMAGE_DOS_HEADER)imageBase;
PIMAGE_NT_HEADERS64 pImageNtHeader = (PIMAGE_NT_HEADERS64)(pImageDosHeader->e_lfanew + (ULONG64)pImageDosHeader);
//区段头s的地址
PIMAGE_OPTIONAL_HEADER64 pImageOptionalHeaders = &(pImageNtHeader->OptionalHeader);
PIMAGE_SECTION_HEADER pFirstSectionHeader = pImageNtHeader->FileHeader.SizeOfOptionalHeader + (ULONG64)pImageOptionalHeaders;
try
{
for (int i = 0; i < pImageNtHeader->FileHeader.NumberOfSections; i++)
{
PIMAGE_SECTION_HEADER pResSection = pFirstSectionHeader + i;
// 判断如果名称相同,则返回区段地址
if (strstr(pResSection->Name, sectionName))
{
DbgPrint("SectionName = %s SectionAddr = %p SectionSize = %d\r\n",
pResSection->Name,
pResSection->VirtualAddress + (ULONG64)imageBase,
pResSection->SizeOfRawData);
//SECTION_INFO section = { 0 };
sectionInfo->sectionAddr = (PULONG64)(pResSection->VirtualAddress + (ULONG64)imageBase);
sectionInfo->sectionSize = pResSection->SizeOfRawData;
return;
}
}
}
except(1)
{
DbgPrint("faild get sections");
}
return;
}
// 通过区段,模块名称及硬编码获取SSDT基址
PVOID getSSDTBase(PDRIVER_OBJECT pDrvierObject)
{
// 获取模块信息
PMODULE_INFO pModuleInfo = getProcessInfoByName("ntoskrnl.exe", pDrvierObject);
if (pModuleInfo == NULL)
{
return NULL;
}
// 通过模块地址获取区段信息
SECTION_INFO sectionInfo = { 0 };
getSectionsAddrByImageBase(pModuleInfo->ModuleBase, "text",§ionInfo);
// 获取SSDT基址
PVOID pSSDTflag = findAddrByOpcode(
ssdt_chars,
sectionInfo.sectionAddr,
(PVOID)((ULONG64)(sectionInfo.sectionAddr) + sectionInfo.sectionSize),
sizeof(ssdt_chars));
PUCHAR pSSDTBase = (PUCHAR)pSSDTflag + sizeof(ssdt_chars);
if (pSSDTBase[0] == 0x4C && pSSDTBase[1] == 0x8D && pSSDTBase[2] == 0x15
&& pSSDTBase[7] == 0x4C && pSSDTBase[8] == 0x8D && pSSDTBase[9] == 0x1D)
{
PULONG64 pSSDTAddr = *(PULONG32)(pSSDTBase + 3) + pSSDTBase+7;
return pSSDTAddr;
}
return NULL;
}
// 通过SSDT基址获取获取所有数据
PVOID getSSDT(PVOID pSSDTBase, PVOID buff)
{
PSSDT pSSDT = (PSSDT)pSSDTBase;
ULONG64 fucnNumber = pSSDT->NumberOfServices;
PULONG32 pFuncBegin = pSSDT->ServiceTableBase;
PUCHAR pParamTableBase = pSSDT->ParamTableBase;
PULONG64 begin = (PULONG64)buff;
for (ULONG i = 0; i < fucnNumber; i++)
{
// 获取地址值,转换为LONG,位移运算才是算是位移:
ULONG64 addr = (ULONG64)((((LONG32)(pFuncBegin[i])) >> 4) + (ULONG64)pFuncBegin);
begin[i*2] = addr;
// 获取参数个数
ULONG64 param = ((ULONG64)(pParamTableBase[i]))/4;
begin[i*2 + 1] = param;
}
return NULL;
}
|