Dalamud/lib/CoreCLR/CoreCLR.cpp

166 lines
4.8 KiB
C++

#define WIN32_LEAN_AND_MEAN
#include "CoreCLR.h"
#include <Windows.h>
#include <filesystem>
#include <iostream>
#include "nethost/nethost.h"
#include "..\..\Dalamud.Boot\logging.h"
CoreCLR::CoreCLR(void* calling_module)
: m_calling_module(calling_module)
{
}
/* Core public functions */
int CoreCLR::load_hostfxr()
{
return CoreCLR::load_hostfxr(nullptr);
}
int CoreCLR::load_hostfxr(const struct get_hostfxr_parameters* parameters)
{
/*
// Get the path to CoreCLR's hostfxr
std::wstring calling_module_path(MAX_PATH, L'\0');
do
{
calling_module_path.resize(GetModuleFileNameW(static_cast<HMODULE>(m_calling_module), &calling_module_path[0], static_cast<DWORD>(calling_module_path.size())));
}
while (!calling_module_path.empty() && GetLastError() == ERROR_INSUFFICIENT_BUFFER);
if (calling_module_path.empty())
return -1;
calling_module_path = (std::filesystem::path(calling_module_path).parent_path() / L"nethost.dll").wstring();
auto lib_nethost = reinterpret_cast<void*>(load_library(calling_module_path.c_str()));
if (!lib_nethost)
return -1;
auto get_hostfxr_path = reinterpret_cast<get_hostfxr_path_type>(
get_export(lib_nethost, "get_hostfxr_path"));
if (!get_hostfxr_path)
return -1;
*/
wchar_t buffer[MAX_PATH]{};
size_t buffer_size = sizeof buffer / sizeof(wchar_t);
if (int rc = get_hostfxr_path(buffer, &buffer_size, parameters); rc != 0)
return rc;
// Load hostfxr and get desired exports
auto lib_hostfxr = reinterpret_cast<void*>(load_library(buffer));
if (!lib_hostfxr)
return -1;
m_hostfxr_initialize_for_runtime_config_fptr = reinterpret_cast<hostfxr_initialize_for_runtime_config_fn>(
get_export(lib_hostfxr, "hostfxr_initialize_for_runtime_config"));
m_hostfxr_get_runtime_delegate_fptr = reinterpret_cast<hostfxr_get_runtime_delegate_fn>(
get_export(lib_hostfxr, "hostfxr_get_runtime_delegate"));
m_hostfxr_close_fptr = reinterpret_cast<hostfxr_close_fn>(
get_export(lib_hostfxr, "hostfxr_close"));
return m_hostfxr_initialize_for_runtime_config_fptr
&& m_hostfxr_get_runtime_delegate_fptr
&& m_hostfxr_close_fptr ? 0 : -1;
}
int CoreCLR::load_runtime(const std::wstring& runtime_config_path)
{
return CoreCLR::load_runtime(runtime_config_path, nullptr);
}
int CoreCLR::load_runtime(const std::wstring& runtime_config_path, const struct hostfxr_initialize_parameters* parameters)
{
int result;
// Load .NET Core
hostfxr_handle context = nullptr;
result = m_hostfxr_initialize_for_runtime_config_fptr(
runtime_config_path.c_str(),
parameters,
&context);
// Success_HostAlreadyInitialized
if (result == 1)
{
logging::I("Success_HostAlreadyInitialized (0x1)");
result = 0;
}
if (result != 0 || context == nullptr)
{
m_hostfxr_close_fptr(context);
return result;
}
// Get the load assembly function pointer
result = m_hostfxr_get_runtime_delegate_fptr(
context,
hdt_load_assembly_and_get_function_pointer,
reinterpret_cast<void**>(&m_load_assembly_and_get_function_pointer_fptr));
if (result != 0 || m_load_assembly_and_get_function_pointer_fptr == nullptr) {
m_hostfxr_close_fptr(context);
return result;
}
result = m_hostfxr_get_runtime_delegate_fptr(
context,
hdt_get_function_pointer,
reinterpret_cast<void**>(&m_get_function_pointer_fptr));
if (result != 0 || m_get_function_pointer_fptr == nullptr)
{
m_hostfxr_close_fptr(context);
return result;
}
m_hostfxr_close_fptr(context);
return 0;
}
int CoreCLR::load_assembly_and_get_function_pointer(
const wchar_t* assembly_path,
const wchar_t* type_name,
const wchar_t* method_name,
const wchar_t* delegate_type_name,
void* reserved,
void** delegate) const
{
int result = m_load_assembly_and_get_function_pointer_fptr(assembly_path, type_name, method_name, delegate_type_name, reserved, delegate);
if (result != 0)
delegate = nullptr;
return result;
};
int CoreCLR::get_function_pointer(
const wchar_t* type_name,
const wchar_t* method_name,
const wchar_t* delegate_type_name,
void* load_context,
void* reserved,
void** delegate) const
{
int result = m_get_function_pointer_fptr(type_name, method_name, delegate_type_name, load_context, reserved, delegate);
if (result != 0)
delegate = nullptr;
return result;
}
/* Helpers */
uint64_t CoreCLR::load_library(const wchar_t* path)
{
return reinterpret_cast<uint64_t>(LoadLibraryW(path));
}
uint64_t CoreCLR::get_export(void* h, const char* name)
{
return reinterpret_cast<uint64_t>(GetProcAddress(static_cast<HMODULE>(h), name));
}