-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpe_import.c
178 lines (171 loc) · 5.78 KB
/
pe_import.c
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
#include <stdio.h>
#include <stdlib.h>
#include <inttypes.h>
#include "x86_regs.h"
#include "pe_import.h"
#include "pe_libs.h"
#include "dynamic_memory.h"
#include "dmem_utils.h"
typedef struct {
int is_ord;
char* name;
uint16_t hint_or_ord;
} func_info;
typedef struct {
uint32_t int_addr, iat_addr, iat_size;
int has_int;
char* name;
func_info* funcs;
uint32_t func_num;
} imported_lib_info;
static imported_lib_info* imported_libs = NULL;
static uint32_t imported_lib_count = 0;
static uint32_t read_num(const uint8_t* data, int size) {
uint32_t ret = 0;
int i;
for (i = 0; i < size; i++) {
ret |= data[i] << (i * 8);
}
return ret;
}
static void write_num(uint8_t* out, uint32_t value, int size) {
int i;
for (i = 0; i < size; i++) {
out[i] = (value >> (8 * i)) & 0xff;
}
}
int pe_import_initialize(const pe_import_params* params, uint32_t work_start, uint32_t argc, uint32_t argv) {
uint32_t iat_end = params->iat_addr + (params->iat_size > 0 ? params->iat_size - 1 : 0);
if (!pe_libs_initialize(work_start, argc, argv)) return 0;
free(imported_libs);
imported_libs = NULL;
imported_lib_count = 0;
if (params->import_size >= 20) {
uint32_t i;
for (i = 0; i * 20 <= params->import_size - 20; i++) {
uint32_t this_addr = params->import_addr + i * 20;
uint8_t data[20];
uint32_t name_ptr;
uint32_t j;
int all_zero = 1;
imported_lib_info* new_libs;
int lib_id;
/* 情報をロードする */
if (!dmemory_is_allocated(this_addr, 20)) {
fprintf(stderr, "PE import descriptor %"PRIu32" not allocated!\n", i);
return 0;
}
dmemory_read(data, this_addr, 20);
for (j = 0; j < 20; j++) {
if (data[j] != 0) all_zero = 0;
}
if (all_zero) break;
new_libs = realloc(imported_libs, sizeof(*imported_libs) * (i + 1));
if (new_libs == NULL) {
perror("realloc");
return 0;
}
/* 情報をデコードする */
imported_libs = new_libs;
imported_libs[i].int_addr = read_num(data + 0, 4);
imported_libs[i].iat_addr = read_num(data + 16, 4);
name_ptr = read_num(data + 12, 4);
if (UINT32_MAX - params->image_base < imported_libs[i].int_addr ||
UINT32_MAX - params->image_base < imported_libs[i].iat_addr ||
UINT32_MAX - params->image_base < name_ptr) {
fprintf(stderr, "some address in PE import descriptor %"PRIu32" out of address space!\n", i);
return 0;
}
imported_libs[i].has_int = (imported_libs[i].int_addr != 0);
imported_libs[i].int_addr += params->image_base;
imported_libs[i].iat_addr += params->image_base;
imported_libs[i].iat_size = 0;
name_ptr += params->image_base;
/* インポート対象の名前情報を得る */
imported_libs[i].name = dmem_read_string(name_ptr);
if (imported_libs[i].name == NULL) {
fprintf(stderr, "name in PE import descriptor %"PRIu32" is invalid!\n", i);
return 0;
}
lib_id = get_lib_id(imported_libs[i].name);
/* INT/IATから情報を得る */
if (!imported_libs[i].has_int) imported_libs[i].int_addr = imported_libs[i].iat_addr;
imported_libs[i].funcs = NULL;
imported_libs[i].func_num = 0;
if (iat_end >= 3) {
for (j = 0; imported_libs[i].iat_addr + j * 4 <= iat_end - 3; j++) {
uint8_t entry[4];
uint32_t entry_value;
func_info* new_funcs;
func_info* new_func;
if (UINT32_MAX - imported_libs[i].int_addr < j * 4 ||
!dmemory_is_allocated(imported_libs[i].int_addr + j * 4, 4)) {
fprintf(stderr, "unallocated entry exists in INT/IAT of PE import descriptor %"PRIu32"\n", i);
return 0;
}
dmemory_read(entry, imported_libs[i].int_addr + j * 4, 4);
entry_value = read_num(entry, 4);
if (entry_value == 0) break;
new_funcs = realloc(imported_libs[i].funcs, sizeof(*new_funcs) * (j + 1));
if (new_funcs == NULL) {
perror("realloc");
return 0;
}
imported_libs[i].funcs = new_funcs;
new_func = &imported_libs[i].funcs[j];
if (entry_value & UINT32_C(0x80000000)) {
new_func->is_ord = 1;
new_func->name = NULL;
new_func->hint_or_ord = entry_value & 0xffff;
} else {
uint8_t hint[2];
if (UINT32_MAX - params->image_base < entry_value + 2 ||
!dmemory_is_allocated(params->image_base + entry_value, 2) ||
(new_func->name = dmem_read_string(params->image_base + entry_value + 2)) == NULL) {
fprintf(stderr, "invalid entry exists in INT/IAT of PE import descriptor %"PRIu32"\n", i);
return 0;
}
dmemory_read(hint, params->image_base + entry_value, 2);
new_func->is_ord = 0;
new_func->hint_or_ord = read_num(hint, 2);
}
write_num(entry, get_buffer_address(lib_id, new_func->name, imported_libs[i].iat_addr + j * 4), 4);
dmemory_write(entry, imported_libs[i].iat_addr + j * 4, 4);
}
imported_libs[i].func_num = j;
imported_libs[i].iat_size = j * 4;
}
}
imported_lib_count = i;
}
return 1;
}
int pe_import(uint32_t* eip, uint32_t regs[]) {
uint32_t i;
const imported_lib_info* called_lib = NULL;
const func_info* called_func = NULL;
uint32_t stack_remove_size;
for (i = 0; i < imported_lib_count; i++) {
if (imported_libs[i].iat_addr <= *eip && *eip - imported_libs[i].iat_addr < imported_libs[i].iat_size) {
uint32_t offset = *eip - imported_libs[i].iat_addr;
called_lib = &imported_libs[i];
if (offset % 4 == 0) {
called_func = &imported_libs[i].funcs[offset / 4];
}
break;
}
}
if (called_lib == NULL) {
fprintf(stderr, "library not found for EIP %08"PRIx32"\n", *eip);
return -1;
}
stack_remove_size = pe_lib_exec(regs, called_lib->name,
called_func->is_ord ? NULL : called_func->name,
called_func->is_ord ? called_func->hint_or_ord : 0);
if (stack_remove_size == PE_LIB_EXEC_FAILED) return -1;
if (stack_remove_size == PE_LIB_EXEC_EXIT) return 0;
/* ret */
dmemory_read(eip, regs[ESP], 4);
regs[ESP] += 4 + stack_remove_size;
return 1;
}