Creating a new detour looks like this most of the time:
// Create a Detour (or a Hook).
Detour d = new Detour(methodInfoFrom, methodInfoTo);
// When you want to undo and free it, dispose it.
d.Dispose();
Take a look at the DetourTest and HookTest classes for more usage details.
If you want to know more about how it works, take a look at the technical details further down below.
HookGen makes runtime detouring easy to use and flexible.
- Run
MonoMod.RuntimeDetour.HookGen.exe Celeste.exe
, which generatesMMHOOK_Celeste.dll
. Either automate this step in your "mod installer", ship the .dll with your modding API, or merge your API .dll with this .dll using il-repack. Make sure to ship MonoMod.RuntimeDetour.dll and MonoMod.Utils.dll! - Add
MMHOOK_Celeste.dll
to your assembly references in your mod project.
// Let's hook Celeste.Player.GetTrailColor
// Note that we can also -= this (and the IL. manipulation) by using separate methods.
On.Celeste.Player.GetTrailColor += (orig, player, wasDashB) => {
Console.WriteLine("1 - Hello, World!");
// Get the "original" color and manipulate it.
// This step is optional - we can return anything we want.
// We can also pass anything to the orig method.
Color color = orig(player, wasDashB);
// If the player is facing left, display a modified color.
if (player.Facing == Facings.Left)
return new Color(0xFF, color.G, color.B, color.A);
return color;
};
// Or...
IL.Celeste.Player.GetTrailColor += (il) => {
ILCursor c = new ILCursor(il);
// The new cursor starts out at the beginning of the method.
// You can either set .Index directly or perform basic pattern matching using .Goto*
// Insert Console.WriteLine(...)
c.Emit(OpCodes.Ldstr, "2 - Hello, IL manipulation!");
c.Emit(OpCodes.Call, typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }));
// After that, emit an inline delegate call.
c.EmitDelegate<Action>(() => {
Console.WriteLine("3 - Hello, C# code in IL!");
});
// There are also many other helpers to f.e. quickly advance to a region or
// push + invoke any arbitrary delegate accepting any arguments, returning anything.
// Take a look at the il. and c. autocomplete recommendations.
// Leave the rest of the method unmodified.
};
HookGen doesn't modify the original assembly. The generated MMHOOK .dll doesn't contain the original code - it only contains events using RuntimeDetour behind behind the scene.
For every non-generic method in the input assembly, HookGen generates an event and two delegate types with the "On." namespace prefix.
The first delegate type, orig_MethodName, matches the original method's signature, adding a "self" parameter for instance methods.
The second type, hook_MethodName, is the event type. The only difference between orig_ and hook_ is that latter takes an orig_ delegate as the first parameter.
It also generates an event with the "IL." namespace prefix to manipulate the IL at runtime. This feature is still WIP, but it's proven to work already.
MonoMod.RuntimeDetour.HookGen and MonoMod.RuntimeDetour handle all the dirty detouring work transparently. You can even remove your hooks the same way you'd remove event handlers:
// Add a hook.
On.Celeste.PlayerHair.GetHairColor += OnGetHairColor;
// Remove a hook.
On.Celeste.PlayerHair.GetHairColor -= OnGetHairColor;
The RuntimeDetour namespace is split up into the following "layers", bottom to top:
The "platform layer" is an abstraction layer that makes porting RuntimeDetour to new native platforms (ARM) and new runtime platforms (.NET Core) much easier. The performance penalty of using an abstraction layer only applies during the detour creation / application process.
- IDetourNativePlatform is responsible for freeing, copying and allocating memory, setting page flags (read-write-execute) and applying the actual native detour, using the struct NativeDetourData.
As it's operating completely on the native level, it isn't restricted to .NET runtime methods and can also detour native functions.
This allows maintaining the x86 / x86-64 native code separately from the ARM native code and any other platform-specific fixes. - IDetourRuntimePlatform is responsible for pinning methods, creating IL-copies at runtime and getting the starting address of the JIT's resulting native code.
The Mono and .NET Framework detour platforms inherit from the shared DetourRuntimeILPlatform, with the only main differences being how the RuntimeMethodHandle is obtained and how DynamicMethods are JITed.
The following classes implement this interface and allow you to create detours by just instantiating them. The instances are also the way how you generate trampolines and how you undo detours.
- NativeDetour is the lowest-level "managed detour." Even though you can directly use the native and runtime platforms and the NativeDetourData struct, one should use this class instead.
No multi-detour management happens on this level. A "from" pointer can be NativeDetoured only once at a time if you need deterministic results.
If you apply a NativeDetour on a "managed function" (System.Reflection.MethodBase), it creates an IL-copy of the method and uses it as the trampoline. Otherwise, the trampoline temporarily undoes the detour and calls the original method. This means that NativeDetour can be used on large enough native functions. - Detour is the first fully-managed detour level. It thus doesn't allow you to pass a pointer as a method to detour "from", but it manages a dynamic detour chain. Calling the trampoline calls the previous detour, if it exist, or the original method, if none exists. It holds an internal NativeDetour for the top of the detour chain.
- Hook takes detours to the next level and are the driving force behind HookGen. It allows you to detour from a method to any arbitrary delegate with a matching signature, allowing you to receive the trampoline as the first parameter.