/** ****************************************************************************** * Xenia : Xbox 360 Emulator Research Project * ****************************************************************************** * Copyright 2013 Ben Vanik. All rights reserved. * * Released under the BSD license - see LICENSE in the root for more details. * ****************************************************************************** */ #include "kernel/modules/xboxkrnl/objects/xmodule.h" #include "kernel/modules/xboxkrnl/objects/xthread.h" using namespace xe; using namespace xe::kernel; using namespace xe::kernel::xboxkrnl; namespace { } XModule::XModule(KernelState* kernel_state, const char* path) : XObject(kernel_state, kTypeModule), xex_(NULL) { XEIGNORE(xestrcpy(path_, XECOUNT(path_), path)); const xechar_t *slash = xestrrchr(path, '/'); if (slash) { XEIGNORE(xestrcpy(name_, XECOUNT(name_), slash + 1)); } } XModule::~XModule() { xe_xex2_release(xex_); } const char* XModule::path() { return path_; } const char* XModule::name() { return name_; } xe_xex2_ref XModule::xex() { return xe_xex2_retain(xex_); } const xe_xex2_header_t* XModule::xex_header() { return xe_xex2_get_header(xex_); } X_STATUS XModule::LoadFromFile(const char* path) { // TODO(benvanik): load from virtual filesystem. XEASSERTALWAYS(); return X_STATUS_UNSUCCESSFUL; } X_STATUS XModule::LoadFromMemory(const void* addr, const size_t length) { // Load the XEX into memory and decrypt. xe_xex2_options_t xex_options; xex_ = xe_xex2_load(kernel_state()->memory(), addr, length, xex_options); XEEXPECTNOTNULL(xex_); // Prepare the module for execution. XEEXPECTZERO(kernel_state()->processor()->PrepareModule( name_, path_, xex_, runtime()->export_resolver())); return X_STATUS_SUCCESS; XECLEANUP: return X_STATUS_UNSUCCESSFUL; } X_STATUS XModule::GetSection(const char* name, uint32_t* out_data, uint32_t* out_size) { const PESection* section = xe_xex2_get_pe_section(xex_, name); if (!section) { return X_STATUS_UNSUCCESSFUL; } *out_data = section->address; *out_size = section->size; return X_STATUS_SUCCESS; } void* XModule::GetProcAddressByOrdinal(uint16_t ordinal) { // TODO(benvanik): check export tables. XELOGE(XT("GetProcAddressByOrdinal not implemented")); return NULL; } X_STATUS XModule::Launch(uint32_t flags) { const xe_xex2_header_t* header = xex_header(); // Set as the main module, while running. kernel_state()->SetExecutableModule(this); // Create a thread to run in. XThread* thread = new XThread( kernel_state(), header->exe_stack_size, 0, header->exe_entry_point, NULL, 0); X_STATUS result = thread->Create(); if (XFAILED(result)) { XELOGE(XT("Could not create launch thread: %.8X"), result); return result; } // Wait until thread completes. // XLARGE_INTEGER timeout = XINFINITE; // xekNtWaitForSingleObjectEx(thread_handle, TRUE, &timeout); while (true) { sleep(1); } kernel_state()->SetExecutableModule(NULL); thread->Release(); return X_STATUS_SUCCESS; } void XModule::Dump() { ExportResolver* export_resolver = runtime()->export_resolver().get(); const xe_xex2_header_t* header = xe_xex2_get_header(xex_); // XEX info. printf("Module %s:\n\n", path_); printf(" Module Flags: %.8X\n", header->module_flags); printf(" System Flags: %.8X\n", header->system_flags); printf("\n"); printf(" Address: %.8X\n", header->exe_address); printf(" Entry Point: %.8X\n", header->exe_entry_point); printf(" Stack Size: %.8X\n", header->exe_stack_size); printf(" Heap Size: %.8X\n", header->exe_heap_size); printf("\n"); printf(" Execution Info:\n"); printf(" Media ID: %.8X\n", header->execution_info.media_id); printf(" Version: %d.%d.%d.%d\n", header->execution_info.version.major, header->execution_info.version.minor, header->execution_info.version.build, header->execution_info.version.qfe); printf(" Base Version: %d.%d.%d.%d\n", header->execution_info.base_version.major, header->execution_info.base_version.minor, header->execution_info.base_version.build, header->execution_info.base_version.qfe); printf(" Title ID: %.8X\n", header->execution_info.title_id); printf(" Platform: %.8X\n", header->execution_info.platform); printf(" Exec Table: %.8X\n", header->execution_info.executable_table); printf(" Disc Number: %d\n", header->execution_info.disc_number); printf(" Disc Count: %d\n", header->execution_info.disc_count); printf(" Savegame ID: %.8X\n", header->execution_info.savegame_id); printf("\n"); printf(" Loader Info:\n"); printf(" Image Flags: %.8X\n", header->loader_info.image_flags); printf(" Game Regions: %.8X\n", header->loader_info.game_regions); printf(" Media Flags: %.8X\n", header->loader_info.media_flags); printf("\n"); printf(" TLS Info:\n"); printf(" Slot Count: %d\n", header->tls_info.slot_count); printf(" Data Size: %db\n", header->tls_info.data_size); printf(" Address: %.8X, %db\n", header->tls_info.raw_data_address, header->tls_info.raw_data_size); printf("\n"); printf(" Headers:\n"); for (size_t n = 0; n < header->header_count; n++) { const xe_xex2_opt_header_t* opt_header = &header->headers[n]; printf(" %.8X (%.8X, %4db) %.8X = %11d\n", opt_header->key, opt_header->offset, opt_header->length, opt_header->value, opt_header->value); } printf("\n"); // Resources. printf("Resources:\n"); printf(" %.8X, %db\n", header->resource_info.address, header->resource_info.size); printf(" TODO\n"); printf("\n"); // Section info. printf("Sections:\n"); for (size_t n = 0, i = 0; n < header->section_count; n++) { const xe_xex2_section_t* section = &header->sections[n]; const char* type = "UNKNOWN"; switch (section->info.type) { case XEX_SECTION_CODE: type = "CODE "; break; case XEX_SECTION_DATA: type = "RWDATA "; break; case XEX_SECTION_READONLY_DATA: type = "RODATA "; break; } const size_t start_address = header->exe_address + (i * xe_xex2_section_length); const size_t end_address = start_address + (section->info.page_count * xe_xex2_section_length); printf(" %3d %s %3d pages %.8X - %.8X (%d bytes)\n", (int)n, type, section->info.page_count, (int)start_address, (int)end_address, section->info.page_count * xe_xex2_section_length); i += section->info.page_count; } printf("\n"); // Static libraries. printf("Static Libraries:\n"); for (size_t n = 0; n < header->static_library_count; n++) { const xe_xex2_static_library_t *library = &header->static_libraries[n]; printf(" %-8s : %d.%d.%d.%d\n", library->name, library->major, library->minor, library->build, library->qfe); } printf("\n"); // Imports. printf("Imports:\n"); for (size_t n = 0; n < header->import_library_count; n++) { const xe_xex2_import_library_t* library = &header->import_libraries[n]; xe_xex2_import_info_t* import_infos; size_t import_info_count; if (!xe_xex2_get_import_infos(xex_, library, &import_infos, &import_info_count)) { printf(" %s - %d imports\n", library->name, (int)import_info_count); printf(" Version: %d.%d.%d.%d\n", library->version.major, library->version.minor, library->version.build, library->version.qfe); printf(" Min Version: %d.%d.%d.%d\n", library->min_version.major, library->min_version.minor, library->min_version.build, library->min_version.qfe); printf("\n"); // Counts. int known_count = 0; int unknown_count = 0; int impl_count = 0; int unimpl_count = 0; for (size_t m = 0; m < import_info_count; m++) { const xe_xex2_import_info_t* info = &import_infos[m]; KernelExport* kernel_export = export_resolver->GetExportByOrdinal(library->name, info->ordinal); if (kernel_export) { known_count++; if (kernel_export->is_implemented) { impl_count++; } } else { unknown_count++; unimpl_count++; } } printf(" Total: %4zu\n", import_info_count); printf(" Known: %3d%% (%d known, %d unknown)\n", (int)(known_count / (float)import_info_count * 100.0f), known_count, unknown_count); printf(" Implemented: %3d%% (%d implemented, %d unimplemented)\n", (int)(impl_count / (float)import_info_count * 100.0f), impl_count, unimpl_count); printf("\n"); // Listing. for (size_t m = 0; m < import_info_count; m++) { const xe_xex2_import_info_t* info = &import_infos[m]; KernelExport* kernel_export = export_resolver->GetExportByOrdinal( library->name, info->ordinal); const char *name = "UNKNOWN"; bool implemented = false; if (kernel_export) { name = kernel_export->name; implemented = kernel_export->is_implemented; } if (info->thunk_address) { printf(" F %.8X %.8X %.3X (%3d) %s %s\n", info->value_address, info->thunk_address, info->ordinal, info->ordinal, implemented ? " " : "!!", name); } else { printf(" V %.8X %.3X (%3d) %s %s\n", info->value_address, info->ordinal, info->ordinal, implemented ? " " : "!!", name); } } xe_free(import_infos); } printf("\n"); } // Exports. printf("Exports:\n"); printf(" TODO\n"); printf("\n"); }