// OpenTK.Rewrite: IL rewriter for OpenTK.dll // Copyright (C) 2013 Stefanos Apostolopoulos // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Text; using Mono.Cecil; using Mono.Cecil.Cil; using Mono.Cecil.Rocks; namespace OpenTK.Rewrite { // Replaces OpenTK.InteropHelper method instances // with the s IL instructions. class Program { static void Main(string[] args) { if (args.Length == 0) { Console.WriteLine("Usage: rewrite [file.dll] [file.snk]"); return; } var program = new Program(); var file = args[0]; var key = args.Length >= 2 ? args[1] : null; program.Rewrite(file, key); } static AssemblyDefinition mscorlib; void Rewrite(string file, string keyfile) { // Specify assembly read and write parameters // We want to keep a valid symbols file (pdb or mdb) var read_params = new ReaderParameters(); var write_params = new WriterParameters(); #if false // Disabled because symbols file is locked during AfterBuild var pdb = Path.ChangeExtension(file, "pdb"); var mdb = Path.ChangeExtension(file, "mdb"); ISymbolReaderProvider provider = null; if (File.Exists(pdb)) { provider = new Mono.Cecil.Pdb.PdbReaderProvider(); } else if (File.Exists(mdb)) { provider = new Mono.Cecil.Mdb.MdbReaderProvider(); } read_params.SymbolReaderProvider = provider; read_params.ReadSymbols = true; write_params.WriteSymbols = true; #endif if (!String.IsNullOrEmpty(keyfile) && File.Exists(keyfile)) { keyfile = Path.GetFullPath(keyfile); var fs = new FileStream(keyfile, FileMode.Open); var keypair = new System.Reflection.StrongNameKeyPair(fs); fs.Close(); write_params.StrongNameKeyPair = keypair; } else { Console.Error.WriteLine("No keyfile specified or keyfile missing."); } // Load assembly and process all modules var assembly = AssemblyDefinition.ReadAssembly(file, read_params); var rewritten = assembly.CustomAttributes.FirstOrDefault(a => a.AttributeType.Name == "RewrittenAttribute"); if (rewritten == null) { foreach (var module in assembly.Modules) { foreach (var reference in module.AssemblyReferences) { var resolved = module.AssemblyResolver.Resolve(reference); if (reference.Name == "mscorlib") { mscorlib = resolved; } } } if (mscorlib == null) { Console.Error.WriteLine("Falied to locate mscorlib"); return; } foreach (var module in assembly.Modules) { foreach (var type in module.Types) { Rewrite(type); } } } else { Console.Error.WriteLine("Error: assembly has already been rewritten"); } // Save rewritten assembly assembly.Write(file, write_params); } void Rewrite(TypeDefinition type) { var entry_points = type.Fields.FirstOrDefault(f => f.Name == "EntryPoints"); if (entry_points != null) { // Build list of entry point signatures (one per entry point) var entry_signatures = new List(); entry_signatures.AddRange(type.Methods .Where(t => t.CustomAttributes.Any(a => a.AttributeType.Name == "SlotAttribute"))); Rewrite(type, entry_points, entry_signatures); RemoveNativeSignatures(type, entry_signatures); } if (type.Name == "RewrittenAttribute") { var rewritten_constructor = type.GetConstructors().First(); var rewritten = new CustomAttribute(rewritten_constructor, BitConverter.GetBytes(true)); type.Module.Assembly.CustomAttributes.Add(rewritten); } } void Rewrite(TypeDefinition type, FieldDefinition entry_points, List entry_signatures) { // Rewrite all wrapper methods var wrapper_signatures = new List(); wrapper_signatures.AddRange(type.Methods .Where(m => m.IsPublic && m.CustomAttributes.Any(a => a.AttributeType.Name == "AutoGeneratedAttribute"))); foreach (var wrapper in wrapper_signatures) { var autogenerated = wrapper.CustomAttributes .Where(a => a.AttributeType.Name == "AutoGeneratedAttribute"); if (autogenerated.Count() > 0) { var signature_name = (string)autogenerated.First() .Fields.First(f => f.Name == "EntryPoint").Argument.Value; var signature = entry_signatures.First(s => s.Name == signature_name); var slot = (int)signature.CustomAttributes .First(a => a.AttributeType.Name == "SlotAttribute") .ConstructorArguments[0].Value; ProcessMethod(wrapper, signature, slot, entry_points); } } RemoveSupportingAttributes(type); if (type.NestedTypes.Count > 0) { foreach (var nested_type in type.NestedTypes) { Rewrite(nested_type, entry_points, entry_signatures); } } } void RemoveNativeSignatures(TypeDefinition type, List methods) { while (methods.Count > 0) { type.Methods.Remove(methods.Last()); methods.RemoveAt(methods.Count - 1); } } void RemoveSupportingAttributes(TypeDefinition type) { foreach (var method in type.Methods) { var attr = method.CustomAttributes; for (int i = 0; i < attr.Count; i++) { if (attr[i].AttributeType.Name == "AutoGeneratedAttribute") { attr.RemoveAt(i); i--; } } } } // Create body for method static void ProcessMethod(MethodDefinition wrapper, MethodDefinition native, int slot, FieldDefinition entry_points) { var nint = wrapper.DeclaringType.Module.Import(mscorlib.MainModule.GetType("System.IntPtr")); //var nint = new TypeReference("System", "IntPtr", wrapper.DeclaringType.Module, null); var body = wrapper.Body; var il = body.GetILProcessor(); var instructions = body.Instructions; instructions.Clear(); // Declare pinned variables for every reference and array parameter // and push each parameter on the stack EmitParameters(wrapper, nint, body, il); // Patch convenience wrappers if (wrapper.Parameters.Count < native.Parameters.Count) { int parameter_count = EmitParameters(wrapper, nint, body, il); int difference = native.Parameters.Count - parameter_count; if (difference == 2) { // Convert sized out-array/reference to return value, for example: // void GenTextures(int n, int[] textures) -> int GenTexture() // { // const int n = 1; // int buffers; // calli GenTextures(n, &textures); // return result; // } body.Variables.Add(new VariableDefinition(native.Parameters.Last().ParameterType)); il.Emit(OpCodes.Ldc_I4, 1); // const int n = 1 il.Emit(OpCodes.Ldloca, body.Variables.Count - 1); // &buffers } else if (difference == 1 && wrapper.ReturnType.Name != "Void") { // Convert unsized out-array/reference to return value, for example: // void GetBoolean(GetPName pname, out bool data) -> bool GetBoolean(GetPName pname) // { // bool result; // GetBooleanv(pname, &result); // return result; // } body.Variables.Add(new VariableDefinition(native.ReturnType)); il.Emit(OpCodes.Ldloca, body.Variables.Count - 1); } else if (difference == 1 && wrapper.ReturnType.Name == "Void") { // Convert in-array/reference to single element, for example: // void DeleteTextures(int n, ref int textures) -> void DeleteTexture(int texture) // { // const int n = 1; // calli DeleteTextures(n, &textures); // } il.Emit(OpCodes.Ldc_I4, 1); // const int n = 1 il.Emit(OpCodes.Ldarga, wrapper.Parameters.Last()); // &textures } else { Console.Error.WriteLine("Unknown wrapper type for ({0})", native.Name); } } // push the entry point address on the stack EmitEntryPoint(entry_points, il, slot); // issue calli EmitCall(il, native); if (wrapper.ReturnType.Name != "Void") { if (wrapper.Parameters.Count < native.Parameters.Count) { // Convenience wrapper. The result is stored in the last local variable il.Emit(OpCodes.Ldloc, body.Variables.Count - 1); } else if (wrapper.ReturnType != native.ReturnType) { if (wrapper.ReturnType.Name == "String") { // String return-type wrapper // return new string((sbyte*)((void*)GetString())); var intptr_to_voidpointer = wrapper.Module.Import(typeof(IntPtr).GetMethods() .First(m => { return m.Name == "op_Explicit" && m.ReturnType.Name == "Void*"; })); var string_constructor = wrapper.Module.Import(typeof(string).GetConstructors() .First(m => { var p = m.GetParameters(); return p.Length > 0 && p[0].ParameterType.Name == "SByte*"; })); il.Emit(OpCodes.Call, intptr_to_voidpointer); il.Emit(OpCodes.Newobj, string_constructor); } else { Console.Error.WriteLine("Return wrappers not implemented yet ({0})", native.Name); } } else { // nothing to do, the native call leaves the return value // on the stack and we return that unmodified to the caller. } } // return il.Emit(OpCodes.Ret); body.OptimizeMacros(); } static int EmitParameters(MethodDefinition method, TypeReference nint, MethodBody body, ILProcessor il) { int i; for (i = 0; i < method.Parameters.Count; i++) { var p = method.Parameters[i]; il.Emit(OpCodes.Ldarg, i); if (p.ParameterType.IsArray || p.ParameterType.IsByReference) { body.Variables.Add(new VariableDefinition(new PinnedType(nint))); var index = body.Variables.Count - 1; il.Emit(OpCodes.Stloc, index); il.Emit(OpCodes.Ldloc, index); //il.Emit(OpCodes.Conv_I); } } return i; } static void EmitEntryPoint(FieldDefinition entry_points, ILProcessor il, int slot) { il.Emit(OpCodes.Ldsfld, entry_points); il.Emit(OpCodes.Ldc_I4, slot); il.Emit(OpCodes.Ldelem_I); } static void EmitCall(ILProcessor il, MethodReference reference) { var signature = new CallSite(reference.ReturnType) { CallingConvention = MethodCallingConvention.Default, }; foreach (var p in reference.Parameters) { signature.Parameters.Add(p); } // Since the last parameter is always the entry point address, // we do not need any special preparation before emiting calli. il.Emit(OpCodes.Calli, signature); } } }