12#include "arcane/utils/PlatformUtils.h"
13#include "arcane/utils/String.h"
14#include "arcane/utils/CommandLineArguments.h"
15#include "arcane/utils/FatalErrorException.h"
16#include "arcane/utils/Exception.h"
17#include "arcane/utils/CheckedConvert.h"
18#include "arcane/core/Directory.h"
33#include <coreclr_delegates.h>
38using string_t = std::basic_string<char_t>;
54 return string_t((
const char_t*)(s.
utf16().
data()));
56 char_t* _duplicate(
const char_t* x)
63 const UChar* ux =
reinterpret_cast<const UChar*
>(x);
64 size_t slen = wcslen(x);
80#define MAX_PATH PATH_MAX
88 return string_t((
const char_t*)(s.
utf8().
data()));
90 char_t* _duplicate(
const char_t* x)
110int dotnet_verbose = 0;
113using LibHandle = HMODULE;
115using LibHandle =
void*;
124 LibHandle m_lib_handle = (LibHandle)0;
125 bool m_has_valid_lib_handle =
false;
127 hostfxr_initialize_for_runtime_config_fn init_fptr =
nullptr;
128 hostfxr_initialize_for_dotnet_command_line_fn init_command_line_fptr =
nullptr;
129 hostfxr_get_runtime_delegate_fn get_delegate_fptr =
nullptr;
130 hostfxr_run_app_fn run_app_fptr =
nullptr;
131 hostfxr_close_fn close_fptr =
nullptr;
138#define PRINT_FORMAT(level,str,...) \
139 if (dotnet_verbose>=level)\
140 std::cout << String::format("[coreclr] " str "\n",__VA_ARGS__);
144#if defined(ARCANE_DOTNET_ROOT)
145const char* arcane_dotnet_root = ARCANE_DOTNET_ROOT;
147const char* arcane_dotnet_root =
nullptr;
152std::string arcane_dotnet_root_env_variable;
157bool load_hostfxr(
const string_t& assembly_name);
161load_assembly_and_get_function_pointer_fn
162getDotnetLoadAssembly(
const String& assembly)
164 string_t assembly1 = _toString(assembly);
166 void* load_assembly_and_get_function_pointer =
nullptr;
167 hostfxr_handle cxt =
nullptr;
169 hostfxr_initialize_parameters params;
170 params.size =
sizeof(params);
172 params.host_path = assembly1.c_str();
174 int rc = lib_info.init_fptr(assembly1.c_str(), ¶ms, &cxt);
175 if (rc != 0 || cxt ==
nullptr) {
176 auto flags = std::cerr.flags();
177 std::cerr <<
"Init failed: " << std::hex << std::showbase << rc << std::endl;
178 std::cerr.setf(flags);
179 lib_info.close_fptr(cxt);
184 rc = lib_info.get_delegate_fptr(cxt, hdt_load_assembly_and_get_function_pointer,
185 &load_assembly_and_get_function_pointer);
186 if (rc != 0 || load_assembly_and_get_function_pointer ==
nullptr)
187 std::cerr <<
"Get delegate failed: " << std::hex << std::showbase << rc << std::endl;
189 lib_info.close_fptr(cxt);
190 return (load_assembly_and_get_function_pointer_fn)load_assembly_and_get_function_pointer;
199 const String& orig_assembly_name)
201 using const_char_t =
const char_t*;
203 int argc = *(cmd_args.commandLineArgc());
207 string_t root_path1 = _toString(root_path);
208 string_t orig_assembly_name1 = _toString(orig_assembly_name);
209 std::cerr <<
"ENTERING _execDirect root_path=" << root_path <<
"\n";
211 string_t dotnet_root = _toString(
String(arcane_dotnet_root));
212 hostfxr_initialize_parameters params;
213 params.size =
sizeof(params);
214 params.host_path = root_path1.c_str();
215 params.dotnet_root = dotnet_root.c_str();
216 const_char_t* argv =
new const_char_t[1];
217 char_t* argv0_str = _duplicate((
const char_t*)(orig_assembly_name1.c_str()));
219 std::cerr <<
"_execDirect argv[0] =" << orig_assembly_name <<
"\n";
222 hostfxr_handle host_context_handle;
223 int rc = lib_info.init_command_line_fptr(argc, (
const char_t**)argv, ¶ms, &host_context_handle);
224 std::cerr <<
"_execDirect init_command_line R = " << rc <<
"\n";
229 size_t buffer_used = 0;
230 if (hostfxr_get_runtime_property(host_context_handle,
"TEST_PROPERTY",
nullptr, 0, &buffer_used) == HostApiMissingProperty) {
231 hostfxr_set_runtime_property(host_context_handle,
"TEST_PROPERTY",
"TRUE");
235 std::cerr <<
"Launching '.Net'\n";
236 int r = lib_info.run_app_fptr(host_context_handle);
237 std::cerr <<
"End '.Net': R=" << r <<
"\n";
240 lib_info.close_fptr(host_context_handle);
256 if (!verbose_str.
null())
264 if (!dotnet_root_env.
null()){
266 arcane_dotnet_root = arcane_dotnet_root_env_variable.c_str();
270 PRINT_FORMAT(1,
"ARCANE_DOTNET_CORECLR_MAIN assembly_name={0}",orig_assembly_name);
272 if (orig_assembly_name.
empty())
275 string_t orig_assembly_name1 = _toString(orig_assembly_name);
279 PRINT_FORMAT(1,
"ENTERING CORECLR_MAIN root_path={0}",root_path);
284 if (!load_hostfxr(orig_assembly_name1))
287 const bool do_direct_exec =
true;
289 return _execDirect(cmd_args, orig_assembly_name);
297 load_assembly_and_get_function_pointer_fn load_assembly_and_get_function_pointer =
nullptr;
298 load_assembly_and_get_function_pointer = getDotnetLoadAssembly(config_path);
299 assert(load_assembly_and_get_function_pointer !=
nullptr &&
"Failure: get_dotnet_load_assembly()");
305 const char_t* dotnet_type = STR(
"ArcaneMainExec, Arcane.Main");
306 const char_t* dotnet_type_method = STR(
"CoreClrComponentEntryPoint");
309 component_entry_point_fn dll_entry_point_func =
nullptr;
312 int rc = load_assembly_and_get_function_pointer(orig_assembly_name1.c_str(),
317 (
void**)&dll_entry_point_func);
320 ARCANE_FATAL(
"load_assembly_and_get_function_pointer: rc={0}", rc);
321 if (!dll_entry_point_func)
322 ARCANE_FATAL(
"Failure: load_assembly_and_get_function_pointer()");
329 const char_t* message;
332 lib_args args{ STR(
"from host!"), 0 };
334 return dll_entry_point_func(&args,
sizeof(args));
341extern "C" ARCANE_EXPORT
int
347 ret = _arcaneCoreClrMainInternal(cmd_args,orig_assembly_name);
350 ret = arcanePrintArcaneException(ex,
nullptr);
352 catch(
const std::exception& ex){
353 ret = arcanePrintStdException(ex,
nullptr);
356 ret = arcanePrintAnyException(
nullptr);
372LibHandle load_library(
const char_t*);
373void* get_export(LibHandle,
const char*);
376LibHandle load_library(
const char_t* path)
378 HMODULE h = ::LoadLibraryW(path);
380 ARCANE_FATAL(
"Can not load library '{0}'",_toArcaneString(path));
383void free_library(LibHandle h)
387void* get_export(LibHandle h,
const char* name)
389 void* f = ::GetProcAddress(h, name);
395LibHandle load_library(
const char_t* path)
397 void* h = dlopen(path, RTLD_LAZY | RTLD_LOCAL);
399 ARCANE_FATAL(
"Can not load library '{0}' error='{1}'",path,dlerror());
402void free_library(LibHandle h)
406void* get_export(LibHandle h,
const char* name)
408 void* f = dlsym(h, name);
409 PRINT_FORMAT(1,
"get_export name={0} f={1}",name,f);
411 PRINT_FORMAT(0,
"Can not get library symbol '{0}' error='{1}'",name,dlerror());
419load_hostfxr(
const string_t& assembly_name)
425 string_t dotnet_root = _toString(
String(arcane_dotnet_root));
426 get_hostfxr_parameters hostfxr_parameters;
427 hostfxr_parameters.size =
sizeof(get_hostfxr_parameters);
428 hostfxr_parameters.assembly_path = assembly_name.c_str();
429 hostfxr_parameters.dotnet_root = dotnet_root.c_str();
431 PRINT_FORMAT(1,
"Entering load_hostfxr() dotnet_root={0}",arcane_dotnet_root);
433 const int BUF_LEN = 12000;
434 char_t buffer[BUF_LEN];
435 size_t buffer_size =
sizeof(buffer) /
sizeof(char_t);
442 int rc = get_hostfxr_path(buffer, &buffer_size, &hostfxr_parameters);
443 PRINT_FORMAT(1,
"Return value of 'get_hostfxr_path' = '{0}'",rc);
445 PRINT_FORMAT(0,
"Error or warning calling 'get_hostfxr_path' = '{0}'",rc);
450 LibHandle lib = load_library(buffer);
451 PRINT_FORMAT(1,
"LIB_PTR={0} path={1}",lib,_toArcaneString(buffer));
452 lib_info.m_lib_handle = lib;
453 lib_info.m_has_valid_lib_handle =
true;
454 lib_info.init_fptr = (hostfxr_initialize_for_runtime_config_fn)get_export(lib,
"hostfxr_initialize_for_runtime_config");
455 lib_info.get_delegate_fptr = (hostfxr_get_runtime_delegate_fn)get_export(lib,
"hostfxr_get_runtime_delegate");
456 lib_info.close_fptr = (hostfxr_close_fn)get_export(lib,
"hostfxr_close");
457 lib_info.init_command_line_fptr = (hostfxr_initialize_for_dotnet_command_line_fn)get_export(lib,
"hostfxr_initialize_for_dotnet_command_line");
458 lib_info.run_app_fptr = (hostfxr_run_app_fn)get_export(lib,
"hostfxr_run_app");
459 return (lib_info.init_fptr && lib_info.get_delegate_fptr && lib_info.close_fptr && lib_info.init_command_line_fptr && lib_info.run_app_fptr);
470 if (m_has_valid_lib_handle){
471 free_library(m_lib_handle);
472 m_lib_handle = (LibHandle)0;
474 m_has_valid_lib_handle =
false;
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
Arguments de la ligne de commande.
Vue constante d'un tableau de type T.
constexpr const_pointer data() const noexcept
Pointeur sur la mémoire allouée.
Classe gérant un répertoire.
virtual String file(const String &file_name) const
Retourne le chemin complet du fichier file_name dans le répertoire.
Classe de base d'une exception.
Vue sur une chaîne de caractères UTF-8.
Chaîne de caractères unicode.
bool null() const
Retourne true si la chaîne est nulle.
ByteConstArrayView utf8() const
Retourne la conversion de l'instance dans l'encodage UTF-8.
bool empty() const
Vrai si la chaîne est vide (nulle ou "")
ConstArrayView< UChar > utf16() const
Retourne la conversion de l'instance dans l'encodage UTF-16.
std::string_view toStdStringView() const
Retourne une vue de la STL sur la chaîne actuelle.
Integer len(const char *s)
Retourne la longueur de la chaîne s.
Int32 toInt32(Int64 v)
Converti un Int64 en un Int32.
-*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
unsigned short UChar
Type d'un caractère unicode.