Allow new-style export registration.
This commit is contained in:
parent
08770a4ec0
commit
ce3359389b
|
@ -417,9 +417,8 @@ uint64_t UndefinedCallExtern(void* raw_context, uint64_t symbol_info_ptr) {
|
||||||
}
|
}
|
||||||
void X64Emitter::CallExtern(const hir::Instr* instr,
|
void X64Emitter::CallExtern(const hir::Instr* instr,
|
||||||
const FunctionInfo* symbol_info) {
|
const FunctionInfo* symbol_info) {
|
||||||
if (!symbol_info->extern_handler()) {
|
if (symbol_info->behavior() == FunctionBehavior::kBuiltin &&
|
||||||
CallNative(UndefinedCallExtern, reinterpret_cast<uint64_t>(symbol_info));
|
symbol_info->builtin_handler()) {
|
||||||
} else if (symbol_info->behavior() == FunctionBehavior::kBuiltin) {
|
|
||||||
// rcx = context
|
// rcx = context
|
||||||
// rdx = target host function
|
// rdx = target host function
|
||||||
// r8 = arg0
|
// r8 = arg0
|
||||||
|
@ -433,7 +432,8 @@ void X64Emitter::CallExtern(const hir::Instr* instr,
|
||||||
ReloadECX();
|
ReloadECX();
|
||||||
ReloadEDX();
|
ReloadEDX();
|
||||||
// rax = host return
|
// rax = host return
|
||||||
} else if (symbol_info->behavior() == FunctionBehavior::kExtern) {
|
} else if (symbol_info->behavior() == FunctionBehavior::kExtern &&
|
||||||
|
symbol_info->extern_handler()) {
|
||||||
// rcx = context
|
// rcx = context
|
||||||
// rdx = target host function
|
// rdx = target host function
|
||||||
mov(rdx, reinterpret_cast<uint64_t>(symbol_info->extern_handler()));
|
mov(rdx, reinterpret_cast<uint64_t>(symbol_info->extern_handler()));
|
||||||
|
@ -444,6 +444,8 @@ void X64Emitter::CallExtern(const hir::Instr* instr,
|
||||||
ReloadECX();
|
ReloadECX();
|
||||||
ReloadEDX();
|
ReloadEDX();
|
||||||
// rax = host return
|
// rax = host return
|
||||||
|
} else {
|
||||||
|
CallNative(UndefinedCallExtern, reinterpret_cast<uint64_t>(symbol_info));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -19,23 +19,21 @@ ExportResolver::ExportResolver() {}
|
||||||
|
|
||||||
ExportResolver::~ExportResolver() {}
|
ExportResolver::~ExportResolver() {}
|
||||||
|
|
||||||
void ExportResolver::RegisterTable(const std::string& library_name,
|
void ExportResolver::RegisterTable(
|
||||||
Export* exports, const size_t count) {
|
const std::string& library_name,
|
||||||
tables_.emplace_back(library_name, exports, count);
|
const std::vector<xe::cpu::Export*>* exports) {
|
||||||
|
tables_.emplace_back(library_name, exports);
|
||||||
}
|
}
|
||||||
|
|
||||||
Export* ExportResolver::GetExportByOrdinal(const std::string& library_name,
|
Export* ExportResolver::GetExportByOrdinal(const std::string& library_name,
|
||||||
uint16_t ordinal) {
|
uint16_t ordinal) {
|
||||||
for (const auto& table : tables_) {
|
for (const auto& table : tables_) {
|
||||||
if (table.name == library_name || table.simple_name == library_name) {
|
if (table.name == library_name || table.simple_name == library_name) {
|
||||||
// TODO(benvanik): binary search?
|
if (ordinal > table.exports->size()) {
|
||||||
for (size_t n = 0; n < table.count; n++) {
|
|
||||||
if (table.exports[n].ordinal == ordinal) {
|
|
||||||
return &table.exports[n];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
return table.exports->at(ordinal);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
|
@ -89,8 +89,8 @@ class ExportResolver {
|
||||||
ExportResolver();
|
ExportResolver();
|
||||||
~ExportResolver();
|
~ExportResolver();
|
||||||
|
|
||||||
void RegisterTable(const std::string& library_name, Export* exports,
|
void RegisterTable(const std::string& library_name,
|
||||||
const size_t count);
|
const std::vector<Export*>* exports);
|
||||||
|
|
||||||
Export* GetExportByOrdinal(const std::string& library_name, uint16_t ordinal);
|
Export* GetExportByOrdinal(const std::string& library_name, uint16_t ordinal);
|
||||||
|
|
||||||
|
@ -105,10 +105,9 @@ class ExportResolver {
|
||||||
struct ExportTable {
|
struct ExportTable {
|
||||||
std::string name;
|
std::string name;
|
||||||
std::string simple_name; // without extension
|
std::string simple_name; // without extension
|
||||||
Export* exports;
|
const std::vector<Export*>* exports;
|
||||||
size_t count;
|
ExportTable(const std::string& name, const std::vector<Export*>* exports)
|
||||||
ExportTable(const std::string& name, Export* exports, size_t count)
|
: name(name), exports(exports) {
|
||||||
: name(name), exports(exports), count(count) {
|
|
||||||
auto dot_pos = name.find_last_of('.');
|
auto dot_pos = name.find_last_of('.');
|
||||||
if (dot_pos != std::string::npos) {
|
if (dot_pos != std::string::npos) {
|
||||||
simple_name = name.substr(0, dot_pos);
|
simple_name = name.substr(0, dot_pos);
|
||||||
|
|
|
@ -234,8 +234,13 @@ bool XexModule::SetupLibraryImports(const xe_xex2_import_library_t* library) {
|
||||||
|
|
||||||
FunctionInfo::ExternHandler handler = 0;
|
FunctionInfo::ExternHandler handler = 0;
|
||||||
if (kernel_export) {
|
if (kernel_export) {
|
||||||
|
if (kernel_export->function_data.trampoline) {
|
||||||
|
handler = (FunctionInfo::ExternHandler)
|
||||||
|
kernel_export->function_data.trampoline;
|
||||||
|
} else {
|
||||||
handler =
|
handler =
|
||||||
(FunctionInfo::ExternHandler)kernel_export->function_data.shim;
|
(FunctionInfo::ExternHandler)kernel_export->function_data.shim;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
handler = UndefinedImport;
|
handler = UndefinedImport;
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,21 +33,35 @@ XamModule::XamModule(Emulator* emulator, KernelState* kernel_state)
|
||||||
xam::RegisterVoiceExports(export_resolver_, kernel_state_);
|
xam::RegisterVoiceExports(export_resolver_, kernel_state_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<xe::cpu::Export*> xam_exports;
|
||||||
|
|
||||||
|
xe::cpu::Export* RegisterExport_xam(xe::cpu::Export* export) {
|
||||||
|
if (xam_exports.size() <= export->ordinal) {
|
||||||
|
xam_exports.resize(xe::round_up(export->ordinal, 256));
|
||||||
|
}
|
||||||
|
xam_exports[export->ordinal] = export;
|
||||||
|
return export;
|
||||||
|
}
|
||||||
|
|
||||||
void XamModule::RegisterExportTable(xe::cpu::ExportResolver* export_resolver) {
|
void XamModule::RegisterExportTable(xe::cpu::ExportResolver* export_resolver) {
|
||||||
assert_not_null(export_resolver);
|
assert_not_null(export_resolver);
|
||||||
|
|
||||||
if (!export_resolver) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build the export table used for resolution.
|
// Build the export table used for resolution.
|
||||||
#include "xenia/kernel/util/export_table_pre.inc"
|
#include "xenia/kernel/util/export_table_pre.inc"
|
||||||
static xe::cpu::Export xam_export_table[] = {
|
static xe::cpu::Export xam_export_table[] = {
|
||||||
#include "xenia/kernel/xam_table.inc"
|
#include "xenia/kernel/xam_table.inc"
|
||||||
};
|
};
|
||||||
#include "xenia/kernel/util/export_table_post.inc"
|
#include "xenia/kernel/util/export_table_post.inc"
|
||||||
export_resolver->RegisterTable("xam.xex", xam_export_table,
|
for (size_t i = 0; i < xe::countof(xam_export_table); ++i) {
|
||||||
xe::countof(xam_export_table));
|
auto& export = xam_export_table[i];
|
||||||
|
if (xam_exports.size() <= export.ordinal) {
|
||||||
|
xam_exports.resize(xe::round_up(export.ordinal, 256));
|
||||||
|
}
|
||||||
|
if (!xam_exports[export.ordinal]) {
|
||||||
|
xam_exports[export.ordinal] = &export;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
export_resolver->RegisterTable("xam.xex", &xam_exports);
|
||||||
}
|
}
|
||||||
|
|
||||||
XamModule::~XamModule() {}
|
XamModule::~XamModule() {}
|
||||||
|
|
|
@ -23,7 +23,6 @@ class XamModule : public XKernelModule {
|
||||||
virtual ~XamModule();
|
virtual ~XamModule();
|
||||||
|
|
||||||
static void RegisterExportTable(xe::cpu::ExportResolver* export_resolver);
|
static void RegisterExportTable(xe::cpu::ExportResolver* export_resolver);
|
||||||
static void RegisterExport(xe::cpu::Export* export);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
};
|
};
|
||||||
|
|
|
@ -18,6 +18,8 @@ namespace kernel {
|
||||||
|
|
||||||
class KernelState;
|
class KernelState;
|
||||||
|
|
||||||
|
xe::cpu::Export* RegisterExport_xam(xe::cpu::Export* export);
|
||||||
|
|
||||||
namespace xam {
|
namespace xam {
|
||||||
// Registration functions, one per file.
|
// Registration functions, one per file.
|
||||||
void RegisterContentExports(xe::cpu::ExportResolver* export_resolver,
|
void RegisterContentExports(xe::cpu::ExportResolver* export_resolver,
|
||||||
|
|
|
@ -149,22 +149,36 @@ XboxkrnlModule::XboxkrnlModule(Emulator* emulator, KernelState* kernel_state)
|
||||||
WT_EXECUTEINTIMERTHREAD);
|
WT_EXECUTEINTIMERTHREAD);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<xe::cpu::Export*> xboxkrnl_exports;
|
||||||
|
|
||||||
|
xe::cpu::Export* RegisterExport_xboxkrnl(xe::cpu::Export* export) {
|
||||||
|
if (xboxkrnl_exports.size() <= export->ordinal) {
|
||||||
|
xboxkrnl_exports.resize(xe::round_up(export->ordinal, 256));
|
||||||
|
}
|
||||||
|
xboxkrnl_exports[export->ordinal] = export;
|
||||||
|
return export;
|
||||||
|
}
|
||||||
|
|
||||||
void XboxkrnlModule::RegisterExportTable(
|
void XboxkrnlModule::RegisterExportTable(
|
||||||
xe::cpu::ExportResolver* export_resolver) {
|
xe::cpu::ExportResolver* export_resolver) {
|
||||||
assert_not_null(export_resolver);
|
assert_not_null(export_resolver);
|
||||||
|
|
||||||
if (!export_resolver) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build the export table used for resolution.
|
// Build the export table used for resolution.
|
||||||
#include "xenia/kernel/util/export_table_pre.inc"
|
#include "xenia/kernel/util/export_table_pre.inc"
|
||||||
static xe::cpu::Export xboxkrnl_export_table[] = {
|
static xe::cpu::Export xboxkrnl_export_table[] = {
|
||||||
#include "xenia/kernel/xboxkrnl_table.inc"
|
#include "xenia/kernel/xboxkrnl_table.inc"
|
||||||
};
|
};
|
||||||
#include "xenia/kernel/util/export_table_post.inc"
|
#include "xenia/kernel/util/export_table_post.inc"
|
||||||
export_resolver->RegisterTable("xboxkrnl.exe", xboxkrnl_export_table,
|
for (size_t i = 0; i < xe::countof(xboxkrnl_export_table); ++i) {
|
||||||
xe::countof(xboxkrnl_export_table));
|
auto& export = xboxkrnl_export_table[i];
|
||||||
|
if (xboxkrnl_exports.size() <= export.ordinal) {
|
||||||
|
xboxkrnl_exports.resize(xe::round_up(export.ordinal, 256));
|
||||||
|
}
|
||||||
|
if (!xboxkrnl_exports[export.ordinal]) {
|
||||||
|
xboxkrnl_exports[export.ordinal] = &export;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
export_resolver->RegisterTable("xboxkrnl.exe", &xboxkrnl_exports);
|
||||||
}
|
}
|
||||||
|
|
||||||
XboxkrnlModule::~XboxkrnlModule() {
|
XboxkrnlModule::~XboxkrnlModule() {
|
||||||
|
|
|
@ -28,7 +28,6 @@ class XboxkrnlModule : public XKernelModule {
|
||||||
virtual ~XboxkrnlModule();
|
virtual ~XboxkrnlModule();
|
||||||
|
|
||||||
static void RegisterExportTable(xe::cpu::ExportResolver* export_resolver);
|
static void RegisterExportTable(xe::cpu::ExportResolver* export_resolver);
|
||||||
static void RegisterExport(xe::cpu::Export* export);
|
|
||||||
|
|
||||||
int LaunchModule(const char* path);
|
int LaunchModule(const char* path);
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,8 @@ namespace kernel {
|
||||||
|
|
||||||
class KernelState;
|
class KernelState;
|
||||||
|
|
||||||
|
xe::cpu::Export* RegisterExport_xboxkrnl(xe::cpu::Export* export);
|
||||||
|
|
||||||
namespace xboxkrnl {
|
namespace xboxkrnl {
|
||||||
// Registration functions, one per file.
|
// Registration functions, one per file.
|
||||||
void RegisterAudioExports(xe::cpu::ExportResolver* export_resolver,
|
void RegisterAudioExports(xe::cpu::ExportResolver* export_resolver,
|
||||||
|
|
Loading…
Reference in New Issue