Better .rdata organization

This commit is contained in:
David Gonzalez Martin 2024-10-09 17:18:49 -06:00 committed by David
parent 2a904d1ea8
commit 95e5d62203
3 changed files with 118 additions and 89 deletions

View File

@ -835,6 +835,14 @@ may_be_unused fn String string_base(String string)
auto index = cast(u64, s64, maybe_index);
result = s_get_slice(u8, string, index + 1, string.length);
}
#if _WIN32
if (!result.pointer)
{
auto maybe_index = string_last_ch(string, '\\');
auto index = cast(u64, s64, maybe_index);
result = s_get_slice(u8, string, index + 1, string.length);
}
#endif
return result;
}
@ -3286,6 +3294,7 @@ may_be_unused fn void run_command(Arena* arena, CStringSlice arguments, char* en
DWORD exit_code;
if (GetExitCodeProcess(process_information.hProcess, &exit_code))
{
print("Process ran with exit code: {u32:x}\n", exit_code);
if (exit_code != 0)
{
fail();

View File

@ -10664,8 +10664,14 @@ typedef enum COFFDebugType : u32
COFF_DEBUG_OMAP_FROM_SRC = 8,
COFF_DEBUG_BORLAND = 9,
COFF_DEBUG_RESERVED10 = 10,
COFF_DEBUG_BBT = 10,
COFF_DEBUG_CLSID = 11,
COFF_DEBUG_VC_FEATURE = 12,
COFF_DEBUG_POGO = 13,
COFF_DEBUG_ILTCG = 14,
COFF_DEBUG_MPX = 15,
COFF_DEBUG_REPRO = 16,
COFF_DEBUG_SPGO = 18,
COFF_DEBUG_EXTENDED_DLL_CHARACTERISTICS = 20,
} COFFDebugType;
@ -10676,9 +10682,9 @@ STRUCT(COFFDebugDirectory)
u16 major_version;
u16 minor_version;
COFFDebugType type;
u32 data_size;
u32 data_rva;
u32 data_offset;
u32 size;
u32 rva;
u32 file_offset;
};
static_assert(sizeof(COFFDebugDirectory) == 28);
@ -10758,24 +10764,6 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
};
vb_copy_scalar(&file, dos_header);
u8 dos_code[] = { 0x0E, 0x1F, 0xBA, 0x0E, 0x00, 0xB4, 0x09, 0xCD, 0x21, 0xB8, 0x01, 0x4C, 0xCD, 0x21, };
vb_copy_any_array(&file, dos_code);
auto dos_string = strlit("This program cannot be run in DOS mode.\r\r\n");
vb_copy_string(&file, dos_string);
*vb_add(&file, 1) = '$';
vb_align(&file, 8);
u8 rich_header[] = {
0xDD, 0x6A, 0x05, 0xC7, 0x99, 0x0B, 0x6B, 0x94, 0x99, 0x0B, 0x6B, 0x94, 0x99, 0x0B, 0x6B, 0x94,
0xD2, 0x73, 0x6A, 0x95, 0x9A, 0x0B, 0x6B, 0x94, 0x99, 0x0B, 0x6A, 0x94, 0x98, 0x0B, 0x6B, 0x94,
0xD1, 0x8E, 0x6F, 0x95, 0x98, 0x0B, 0x6B, 0x94, 0xD1, 0x8E, 0x69, 0x95, 0x98, 0x0B, 0x6B, 0x94,
0x52, 0x69, 0x63, 0x68, 0x99, 0x0B, 0x6B, 0x94, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
};
vb_copy_any_array(&file, rich_header);
u32 data_directory_count = 16;
const u32 virtual_section_alignment = 0x1000;
@ -10814,7 +10802,6 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
rva = cast(u32, u64, align_forward(rva, virtual_section_alignment));
vb_align(&file, file_section_alignment);
auto file_offset = file.length;
assert(file_offset == 0x400);
u8 text_content[] = { 0x48, 0x83, 0xEC, 0x28, 0x33, 0xC9, 0xFF, 0x15, 0xF4, 0x0F, 0x00, 0x00, 0x90, 0x48, 0x83, 0xC4, 0x28, 0xC3, };
entry_point_rva = rva;
vb_copy_any_array(&file, text_content);
@ -10850,34 +10837,31 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
{
// .rdata
rva = cast(u32, u64, align_forward(rva, virtual_section_alignment));
assert(rva == 0x2000);
vb_align(&file, file_section_alignment);
auto file_offset = file.length;
assert(file_offset == 0x600);
u8 rdata_chunk_0[] = {
0xF0, 0x21, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00,
};
iat_rva = rva;
iat_size = sizeof(rdata_chunk_0);
vb_copy_any_array(&file, rdata_chunk_0);
auto iat_file_offset = file.length;
auto iat_section_offset = iat_file_offset - file_offset;
iat_rva = rva + iat_section_offset;
const u32 import_address_count = 2;
auto* import_address_table = vb_add_any_array(&file, COFFImportAddress, import_address_count);
iat_size = file.length - iat_file_offset;
auto debug_directory_file_offset = file.length;
auto debug_directory_section_offset = debug_directory_file_offset - file_offset;
debug_directory_rva = rva + debug_directory_section_offset;
auto* debug_directory = vb_add_scalar(&file, COFFDebugDirectory);
auto* debug_directories = vb_add_any_array(&file, COFFDebugDirectory, 3);
debug_directory_size = file.length - debug_directory_file_offset;
u8 rdata_chunk_1[] = {
0x00, 0x00, 0x00, 0x00, 0x70, 0x63, 0xFD, 0x66, 0x00, 0x00, 0x00, 0x00, 0x0C, 0x00, 0x00, 0x00,
0x14, 0x00, 0x00, 0x00, 0xC0, 0x20, 0x00, 0x00, 0xC0, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x70, 0x63, 0xFD, 0x66, 0x00, 0x00, 0x00, 0x00, 0x0D, 0x00, 0x00, 0x00, 0xDC, 0x00, 0x00, 0x00,
0xD4, 0x20, 0x00, 0x00, 0xD4, 0x06, 0x00, 0x00, 0x18, 0x00, 0x00, 0x00, 0x03, 0x80, 0x03, 0x80,
0x18, 0x00, 0x00, 0x00, 0x03, 0x80, 0x03, 0x80,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x7C, 0x20, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00,
0x00, 0x10, 0x00, 0x00, 0x12, 0x00, 0x00, 0x00,
};
vb_copy_any_array(&file, rdata_chunk_1);
vb_copy_array(&file, rdata_chunk_1);
auto debug_directory_data_file_offset = file.length;
auto debug_directory_data_section_offset = debug_directory_data_file_offset - file_offset;
@ -10907,69 +10891,96 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
0x01, 0x04, 0x01, 0x00, 0x04, 0x42, 0x00, 0x00,
};
vb_copy_any_array(&file, rdata_chunk_2);
*debug_directory = (COFFDebugDirectory){
.characteristics = 0,
.timestamp = 1727882096,
.major_version = 0,
.minor_version = 0,
.type = COFF_DEBUG_CODEVIEW,
.data_size = debug_directory_data_size,
.data_rva = debug_directory_data_rva,
.data_offset = debug_directory_data_file_offset,
u32 timestamp = 0x66fd6370;
COFFDebugDirectory sample_debug_directories[] = {
{
.characteristics = 0,
.timestamp = timestamp,
.major_version = 0,
.minor_version = 0,
.type = COFF_DEBUG_CODEVIEW,
.size = debug_directory_data_size,
.rva = debug_directory_data_rva,
.file_offset = debug_directory_data_file_offset,
},
{
.characteristics = 0,
.timestamp = timestamp,
.major_version = 0,
.minor_version = 0,
.type = COFF_DEBUG_VC_FEATURE,
.size = 0x14,
.rva = rva + 0xc0,
.file_offset = file_offset + 0xc0,
},
{
.characteristics = 0,
.timestamp = timestamp,
.major_version = 0,
.minor_version = 0,
.type = COFF_DEBUG_POGO,
.size = 0xdc,
.rva = rva + 0xd4,
.file_offset = file_offset + 0xd4,
},
};
debug_directory_size = sizeof(*debug_directory) + debug_directory->data_size;
assert(file.length == 0x7b8);
memcpy(debug_directories, sample_debug_directories, sizeof(sample_debug_directories));
auto import_directory_file_offset = file.length;
auto import_directory_section_offset = file.length - file_offset;
import_directory_rva = rva + import_directory_section_offset;
// IAT
COFFImportDirectory import_directories[] = {
{
.lookup_table_rva = 0x21e0,
.time_date_stamp = 0,
.forwarder_chain = 0,
.dll_name_rva = 0x21fe,
.address_table_rva = 0x2000,
},
};
const u32 import_directory_count = 2;
auto* import_directories = vb_add_any_array(&file, COFFImportDirectory, import_directory_count);
assert(import_directories[array_length(import_directories) - 1].forwarder_chain == 0);
vb_copy_any_array(&file, import_directories);
import_directory_size = file.length - import_directory_file_offset;
u8 weird_padding[] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, };
vb_copy_any_array(&file, weird_padding);
import_directory_size= file.length - import_directory_file_offset;
COFFImportLookup import_lookups[] = {
{
.name_table_rva = 0x21f0,
},
};
assert(array_length(import_directories) == array_length(import_lookups));
vb_copy_any_array(&file, import_lookups);
assert(file.length - import_directory_file_offset != 40);
// This goes back to the first .rdata offset
// COFFImportAddress import_addresses[] = {
// {
// .name_table_rva = 0x21f0,
// },
// };
vb_align(&file, 0x10);
assert(file.length == 0x7f0);
auto import_lookup_file_offset = file.length;
auto import_lookup_section_offset = file.length - file_offset;
auto import_lookup_rva = rva + import_lookup_section_offset;
const u32 import_lookup_count = import_directory_count;
auto* import_lookups = vb_add_any_array(&file, COFFImportLookup, import_lookup_count);
assert(import_lookup_count == import_directory_count);
auto name_table_file_offset = file.length;
auto name_table_section_offset = name_table_file_offset - file_offset;
auto name_table_rva = rva + name_table_section_offset;
coff_import_name(&file, 376, strlit("ExitProcess"));
assert(file.length == 0x7fe);
auto dll_name_file_offset = file.length;
auto dll_name_section_offset = dll_name_file_offset - file_offset;
auto dll_name_rva = rva + dll_name_section_offset;
auto dll_name = strlit("KERNEL32.dll");
vb_copy_string_zero_terminated(&file, dll_name);
*vb_add(&file, 1) = 0;
assert(import_lookup_count == 2);
import_lookups[0] = (COFFImportLookup) {
.name_table_rva = name_table_rva,
};
import_lookups[1] = (COFFImportLookup) {};
assert(import_directory_count == 2);
import_directories[0] = (COFFImportDirectory) {
.lookup_table_rva = import_lookup_rva,
.time_date_stamp = 0,
.forwarder_chain = 0,
.dll_name_rva = dll_name_rva,
.address_table_rva = iat_rva,
};
import_directories[1] = (COFFImportDirectory) {};
assert(import_directories[import_directory_count - 1].forwarder_chain == 0);
assert(import_address_count == 2);
import_address_table[0] = (COFFImportAddress) {
.name_table_rva = name_table_rva,
};
import_address_table[1] = (COFFImportAddress) {};
auto virtual_size = file.length - file_offset;
vb_align(&file, file_section_alignment);
auto file_size = file.length - file_offset;
@ -10997,7 +11008,6 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
{
// .pdata content
vb_align(&file, file_section_alignment);
assert(file.length == 0xa00);
rva = cast(u32, u64, align_forward(rva, virtual_section_alignment));
auto file_offset = file.length;
@ -11006,7 +11016,7 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
{
.start_rva = text_section_header->rva,
.end_rva = text_section_header->rva + text_section_header->virtual_size,
.unwind_information_rva = 0x21b0,
.unwind_information_rva = rdata_section_header->rva + 0x1b0,
},
};
exception_table_rva = rva;
@ -11051,6 +11061,9 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
uninitialized_data_size += section_header->file_size * section_header->flags.contains_uninitialized_data;
}
assert(debug_directory_size % sizeof(COFFDebugDirectory) == 0);
assert(iat_size % sizeof(COFFImportAddress) == 0);
*coff_optional_header = (COFFOptionalHeader) {
.format = COFF_FORMAT_PE32_PLUS,
.major_linker_version = 14,
@ -11100,12 +11113,19 @@ may_be_unused fn String write_pe(Thread* thread, ObjectOptions options)
auto minimal = file_read(thread->arena, strlit("C:/Users/David/dev/minimal/main.exe"));
assert(file.length == minimal.length);
u32 diff_count = 0;
for (u32 i = 0; i < minimal.length; i += 1)
{
auto mine = file.pointer[i];
auto original = minimal.pointer[i];
assert(mine == original);
if (mine != original)
{
print("Diff at {u32}\n", i);
diff_count += 1;
}
}
assert(diff_count == 0);
#else
unused(thread);
#endif

View File

@ -1,4 +1,4 @@
mkdir -p build
mkdir build
clang -o build/build.exe bootstrap/build.c -g -march=native -std=gnu2x -Wall -Wextra -Wpedantic -Wno-nested-anon-types -Wno-keyword-macro -Wno-gnu-auto-type -Wno-auto-decl-extensions -Wno-gnu-empty-initializer -Wno-fixed-enum-extension -pedantic -fno-exceptions -fno-stack-protector -Wl,/INCREMENTAL:no
SET clang_exit_code=%errorlevel%
echo Clang exit code: %clang_exit_code%