Crate netcorehost[][src]

Expand description

A Rust library for hosting the dotnet core runtime.

It utilizes the dotnet core hosting API to load and execute managed code from withing the current process.

Usage

Running an application

The example below will setup the runtime, load Test.dll and run its Main method:

let hostfxr = nethost::load_hostfxr().unwrap();
let context = hostfxr.initialize_for_dotnet_command_line(pdcstr!("tests/Test/bin/Debug/net5.0/Test.dll")).unwrap();
let result = context.run_app().value();

The full example can be found in examples/run-app.

Calling a managed function

A function pointer to a managed method can be aquired using an AssemblyDelegateLoader. This is only supported for HostfxrContext’s that are initialized using Hostfxr::initialize_for_runtime_config. The runtimeconfig.json is automatically generated for executables, for libraries it is neccessary to add <GenerateRuntimeConfigurationFiles>True</GenerateRuntimeConfigurationFiles> to the projects .csproj file.

Using the default signature

The default method signature is defined as follows:

public delegate int ComponentEntryPoint(IntPtr args, int sizeBytes);

A method with the default signature (see code below) can be loaded using AssemblyDelegateLoader::get_function_pointer_with_default_signature.

C#

using System;

namespace Test {
    public static class Program {
        public static int Hello(IntPtr args, int sizeBytes) {
            Console.WriteLine("Hello from C#!");
            return 42;
        }
    }
}

Rust

let hostfxr = nethost::load_hostfxr().unwrap();
let context =
    hostfxr.initialize_for_runtime_config(pdcstr!("tests/Test/bin/Debug/net5.0/Test.runtimeconfig.json")).unwrap();
let fn_loader =
    context.get_delegate_loader_for_assembly(pdcstr!("tests/Test/bin/Debug/net5.0/Test.dll")).unwrap();
let hello = fn_loader.get_function_pointer_with_default_signature(
    pdcstr!("Test.Program, Test"),
    pdcstr!("Hello"),
).unwrap();
let result = unsafe { hello(std::ptr::null(), 0) };
assert_eq!(result, 42);

Using UnmanagedCallersOnly

A function pointer to a method annotated with UnmanagedCallersOnly can be loaded without specifying its signature (as these methods cannot be overloaded).

C#

using System;
using System.Runtime.InteropServices;

namespace Test {
    public static class Program {
        [UnmanagedCallersOnly]
        public static void UnmanagedHello() {
            Console.WriteLine("Hello from C#!");
        }
    }
}

Rust

let hostfxr = nethost::load_hostfxr().unwrap();
let context =
    hostfxr.initialize_for_runtime_config(pdcstr!("tests/Test/bin/Debug/net5.0/Test.runtimeconfig.json")).unwrap();
let fn_loader =
    context.get_delegate_loader_for_assembly(pdcstr!("tests/Test/bin/Debug/net5.0/Test.dll")).unwrap();
let hello = fn_loader.get_function_pointer_for_unmanaged_callers_only_method(
    pdcstr!("Test.Program, Test"),
    pdcstr!("UnmanagedHello"),
).unwrap();
let hello = unsafe { cast_managed_fn!(hello, fn()) };
hello();

Specifying the delegate type

Another option is to define a custom delegate type and passing its assembly qualified name to AssemblyDelegateLoader::get_function_pointer.

C#

using System;

namespace Test {
    public static class Program {
        public delegate void CustomHelloFunc();
     
        public static void CustomHello() {
            Console.WriteLine("Hello from C#!");
        }
    }
}

Rust

let hostfxr = nethost::load_hostfxr().unwrap();
let context =
    hostfxr.initialize_for_runtime_config(pdcstr!("tests/Test/bin/Debug/net5.0/Test.runtimeconfig.json")).unwrap();
let fn_loader =
    context.get_delegate_loader_for_assembly(pdcstr!("tests/Test/bin/Debug/net5.0/Test.dll")).unwrap();
let hello = fn_loader.get_function_pointer(
    pdcstr!("Test.Program, Test"),
    pdcstr!("CustomHello"),
    pdcstr!("Test.Program+CustomHelloFunc, Test")
).unwrap();
let hello = unsafe { cast_managed_fn!(hello, fn()) };
hello();

The full examples can be found in examples/call-managed-function.

Passing complex parameters

Examples for passing non-primitive parameters can be found in examples/passing-parameters.

Modules

Module for the raw bindings for hostfxr and nethost.

Module containing error enums.

Module for abstractions of the hostfxr library.

Module for abstractions of the nethost library.

Module for a platform dependent c-like string type.

Macros

Macro for casting a MethodWithUnknownSignature to a concrete function signature with the correct calling convention.

A macro for creating a PdCStr at compile time.