-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathlibrary.cpp
144 lines (121 loc) · 4.46 KB
/
library.cpp
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
#ifdef _WIN32
#define EXPORT __declspec(dllexport)
#include <windows.h>
#else
#define EXPORT __attribute__((visibility("default")))
#include <dlfcn.h>
#include <thread>
#endif
/// This class helps to manage shared libraries
class Module {
public:
static void *getBaseAddress(const char *library) {
#ifdef _WIN32
auto base = GetModuleHandleA(library);
#else
auto base = dlopen(library, RTLD_LAZY);
#endif
return reinterpret_cast<void *>(base);
}
static void *getExportByName(void *module, const char *name) {
#ifdef _WIN32
auto address = GetProcAddress((HMODULE) module, name);
#else
auto address = dlsym(module, name);
#endif
return reinterpret_cast<void *>(address);
}
template<typename T>
static T getFunctionByName(void *module, const char *name) {
return reinterpret_cast<T>(getExportByName(module, name));
}
};
#include <hostfxr.h>
#include <coreclr_delegates.h>
/// This enums represents possible errors to hide it from others
/// useful for debugging
enum class InitializeResult : uint32_t {
Success,
HostFxrLoadError,
InitializeRuntimeConfigError,
GetRuntimeDelegateError,
EntryPointError,
};
extern "C" EXPORT InitializeResult bootstrapper_load_assembly(
const char_t *runtime_config_path,
const char_t *assembly_path,
const char_t *type_name,
const char_t *method_name
) {
/// Get module base address
#ifdef _WIN32
auto libraryName = "hostfxr.dll";
#else
auto libraryName = "libhostfxr.so";
#endif
void *module = Module::getBaseAddress(libraryName);
if (!module) {
return InitializeResult::HostFxrLoadError;
}
/// Obtaining useful exports
auto hostfxr_initialize_for_runtime_config_fptr =
Module::getFunctionByName<hostfxr_initialize_for_runtime_config_fn>(module, "hostfxr_initialize_for_runtime_config");
auto hostfxr_get_runtime_delegate_fptr =
Module::getFunctionByName<hostfxr_get_runtime_delegate_fn>(module, "hostfxr_get_runtime_delegate");
auto hostfxr_close_fptr =
Module::getFunctionByName<hostfxr_close_fn>(module, "hostfxr_close");
/// Load runtime config
hostfxr_handle ctx = nullptr;
int rc = hostfxr_initialize_for_runtime_config_fptr(runtime_config_path, nullptr, &ctx);
/// Success_HostAlreadyInitialized = 0x00000001
/// @see https://github.com/dotnet/runtime/blob/main/docs/design/features/host-error-codes.md
if (rc != 1 || ctx == nullptr) {
hostfxr_close_fptr(ctx);
return InitializeResult::InitializeRuntimeConfigError;
}
/// From docs: native function pointer to the requested runtime functionality
void *delegate = nullptr;
int ret = hostfxr_get_runtime_delegate_fptr(ctx, hostfxr_delegate_type::hdt_load_assembly_and_get_function_pointer,
&delegate);
if (ret != 0 || delegate == nullptr) {
return InitializeResult::GetRuntimeDelegateError;
}
/// `void *` -> `load_assembly_and_get_function_pointer_fn`, undocumented???
auto load_assembly_fptr = reinterpret_cast<load_assembly_and_get_function_pointer_fn>(delegate);
typedef void (CORECLR_DELEGATE_CALLTYPE *custom_entry_point_fn)();
custom_entry_point_fn custom = nullptr;
ret = load_assembly_fptr(assembly_path, type_name, method_name, UNMANAGEDCALLERSONLY_METHOD, nullptr,
(void **) &custom);
if (ret != 0 || custom == nullptr) {
return InitializeResult::EntryPointError;
}
custom();
hostfxr_close_fptr(ctx);
return InitializeResult::Success;
}
#ifndef _WIN32
std::string getEnvVar(const char *name) {
auto val = std::getenv(name);
return val == nullptr ? std::string() : std::string(val);
}
__attribute__((constructor))
void initialize_library() {
auto runtime_config_path = getEnvVar("RUNTIME_CONFIG_PATH");
auto assembly_path = getEnvVar("ASSEMBLY_PATH");
auto type_name = getEnvVar("TYPE_NAME");
auto method_name = getEnvVar("METHOD_NAME");
if (!runtime_config_path.empty() && !assembly_path.empty() && !type_name.empty() && !method_name.empty()) {
std::thread thread([=] {
sleep(1);
auto ret = bootstrapper_load_assembly(
runtime_config_path.c_str(),
assembly_path.c_str(),
type_name.c_str(),
method_name.c_str()
);
printf("[+] api.inject() => %d\n", (uint32_t) ret);
});
thread.detach();
}
}
#endif