mirror of
https://github.com/Ryujinx/Opentk.git
synced 2025-01-09 22:55:39 +00:00
ea09f56740
Also extends Rewrite to understand the new syntax of "*name".
1076 lines
46 KiB
C#
1076 lines
46 KiB
C#
// 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 <http://www.gnu.org/licenses/>.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using CommandLine;
|
|
using Mono.Cecil;
|
|
using Mono.Cecil.Cil;
|
|
using Mono.Cecil.Rocks;
|
|
|
|
namespace OpenTK.Rewrite
|
|
{
|
|
// Replaces OpenTK.InteropHelper method instances
|
|
// with the s IL instructions.
|
|
internal class Program
|
|
{
|
|
private static Options Options;
|
|
|
|
private static void Main(string[] args)
|
|
{
|
|
Parser.Default.ParseArguments<Options>(args)
|
|
.WithParsed(result => Options = result)
|
|
.WithNotParsed(error => Environment.Exit(-1));
|
|
|
|
// Argument error checking
|
|
if (!File.Exists(Options.TargetAssembly))
|
|
{
|
|
Console.Error.WriteLine($"Target assembly not found. \n" +
|
|
$"Please check the given path ({Options.TargetAssembly}).");
|
|
}
|
|
|
|
if (!File.Exists(Path.ChangeExtension(Options.TargetAssembly, "pdb")))
|
|
{
|
|
Console.Error.WriteLine("Debugging symbols for target assembly not found. \n" +
|
|
"Please make sure that debugging symbols are being generated.");
|
|
}
|
|
|
|
var program = new Program();
|
|
program.Rewrite();
|
|
}
|
|
|
|
// mscorlib types
|
|
private static AssemblyDefinition mscorlib;
|
|
|
|
private static TypeDefinition TypeMarshal;
|
|
private static TypeDefinition TypeVoid;
|
|
private static TypeDefinition TypeIntPtr;
|
|
private static TypeDefinition TypeInt32;
|
|
|
|
// OpenTK.BindingsBase
|
|
private static TypeDefinition TypeBindingsBase;
|
|
|
|
private void Rewrite()
|
|
{
|
|
// 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();
|
|
|
|
read_params.ReadSymbols = true;
|
|
read_params.ReadWrite = true;
|
|
write_params.WriteSymbols = true;
|
|
|
|
if (!String.IsNullOrEmpty(Options.StrongNameKey) && File.Exists(Options.StrongNameKey))
|
|
{
|
|
string absoluteKeyFilePath = Path.GetFullPath(Options.StrongNameKey);
|
|
|
|
using (var fs = new FileStream(absoluteKeyFilePath, FileMode.Open, FileAccess.Read))
|
|
{
|
|
var keypair = new System.Reflection.StrongNameKeyPair(fs);
|
|
write_params.StrongNameKeyPair = keypair;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Console.Error.WriteLine("No keyfile specified or keyfile missing.");
|
|
}
|
|
|
|
// Load assembly and process all modules
|
|
try
|
|
{
|
|
using (AssemblyDefinition assembly = AssemblyDefinition.ReadAssembly(Options.TargetAssembly, 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)
|
|
{
|
|
try
|
|
{
|
|
var resolved = module.AssemblyResolver.Resolve(reference);
|
|
if (reference.Name == "mscorlib")
|
|
{
|
|
mscorlib = resolved;
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Console.Error.WriteLine(e.ToString());
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mscorlib == null)
|
|
{
|
|
Console.Error.WriteLine("Failed to locate mscorlib");
|
|
return;
|
|
}
|
|
TypeMarshal = mscorlib.MainModule.GetType("System.Runtime.InteropServices.Marshal");
|
|
TypeVoid = mscorlib.MainModule.GetType("System.Void");
|
|
TypeIntPtr = mscorlib.MainModule.GetType("System.IntPtr");
|
|
TypeInt32 = mscorlib.MainModule.GetType("System.Int32");
|
|
|
|
TypeBindingsBase = assembly.Modules.Select(m => m.GetType("OpenTK.BindingsBase")).First();
|
|
|
|
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(write_params);
|
|
}
|
|
}
|
|
catch (InvalidOperationException inex)
|
|
{
|
|
Console.WriteLine("Failed to load the assembly. It may already have been rewritten, and the debug symbols no longer match.");
|
|
Console.WriteLine(inex);
|
|
}
|
|
}
|
|
|
|
private 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<MethodDefinition>();
|
|
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);
|
|
rewritten.ConstructorArguments.Add(new CustomAttributeArgument(
|
|
type.Module.ImportReference(mscorlib.MainModule.GetType("System.Boolean")), true));
|
|
type.Module.Assembly.CustomAttributes.Add(rewritten);
|
|
}
|
|
}
|
|
|
|
private static int GetSlot(MethodDefinition signature)
|
|
{
|
|
// Pretend there is no slots if we want to force everything to work through DllImport (Android & iOS)
|
|
if (Options.UseDLLImport)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
var slot_attribute = signature.CustomAttributes
|
|
.FirstOrDefault(a => a.AttributeType.Name == "SlotAttribute");
|
|
int slot =
|
|
slot_attribute != null ?
|
|
(int)slot_attribute.ConstructorArguments[0].Value :
|
|
-1;
|
|
|
|
return slot;
|
|
}
|
|
|
|
private void Rewrite(TypeDefinition type, FieldDefinition entry_points,
|
|
List<MethodDefinition> entry_signatures)
|
|
{
|
|
// Rewrite all wrapper methods
|
|
var wrapper_signatures = new List<MethodDefinition>();
|
|
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").ToList();
|
|
if (autogenerated.Any())
|
|
{
|
|
var signature_name = (string)autogenerated.First()
|
|
.Fields.First(f => f.Name == "EntryPoint").Argument.Value;
|
|
var signature = entry_signatures.FirstOrDefault(s => s.Name == signature_name);
|
|
int slot = GetSlot(signature);
|
|
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void RemoveNativeSignatures(TypeDefinition type, IEnumerable<MethodDefinition> methods)
|
|
{
|
|
// Remove all DllImports for functions called through calli, since
|
|
// their signatures are embedded directly into the calli callsite.
|
|
// This reduces dll size by ~400KB.
|
|
foreach (var m in methods.Where(s => GetSlot(s) != -1))
|
|
{
|
|
type.Methods.Remove(m);
|
|
}
|
|
}
|
|
|
|
private static 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--;
|
|
}
|
|
}
|
|
|
|
foreach (var parameter in method.Parameters)
|
|
{
|
|
var pattr = parameter.CustomAttributes;
|
|
for (int i = 0; i < pattr.Count; i++)
|
|
{
|
|
if (pattr[i].AttributeType.Name == "CountAttribute")
|
|
{
|
|
pattr.RemoveAt(i);
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Create body for method
|
|
private static void ProcessMethod(MethodDefinition wrapper, MethodDefinition native, int slot,
|
|
FieldDefinition entry_points)
|
|
{
|
|
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
|
|
|
|
DebugVariables vars = null;
|
|
if (Options.EnableDebugCalls)
|
|
{
|
|
vars = EmitDebugPrologue(wrapper, il);
|
|
}
|
|
|
|
// Patch convenience wrappers
|
|
List<GeneratedVariableIdentifier> generatedVariables = new List<GeneratedVariableIdentifier>();
|
|
if (wrapper.Parameters.Count == native.Parameters.Count)
|
|
{
|
|
generatedVariables = EmitParameters(wrapper, native, body, il);
|
|
}
|
|
else
|
|
{
|
|
int difference = native.Parameters.Count - wrapper.Parameters.Count;
|
|
generatedVariables = EmitConvenienceWrapper(wrapper, native, difference, body, il);
|
|
}
|
|
|
|
if (slot == -1 || Options.UseDLLImport)
|
|
{
|
|
// issue DllImport call
|
|
EmitCall(il, native);
|
|
}
|
|
else
|
|
{
|
|
// push the entry point address on the stack
|
|
EmitEntryPoint(entry_points, il, slot);
|
|
|
|
// issue calli
|
|
EmitCalli(il, native);
|
|
}
|
|
|
|
if (wrapper.ReturnType.Name != "Void")
|
|
{
|
|
EmitReturnTypeWrapper(wrapper, native, body, il);
|
|
}
|
|
|
|
EmitParameterEpilogues(wrapper, native, body, il, generatedVariables);
|
|
|
|
if (Options.EnableDebugCalls)
|
|
{
|
|
EmitDebugEpilogue(wrapper, il, vars);
|
|
}
|
|
|
|
// return
|
|
il.Emit(OpCodes.Ret);
|
|
|
|
if (body.Variables.Count > 0)
|
|
{
|
|
// Required for verifiable executables
|
|
// (otherwise peverify complains bitterly)
|
|
body.InitLocals = true;
|
|
}
|
|
body.OptimizeMacros();
|
|
}
|
|
|
|
private class DebugVariables
|
|
{
|
|
public TypeDefinition ErrorHelperType;
|
|
public VariableDefinition ErrorHelperLocal;
|
|
public MethodReference Get_CurrentContext;
|
|
public MethodReference Set_ErrorChecking;
|
|
public Instruction BeginTry;
|
|
}
|
|
|
|
private static DebugVariables EmitDebugPrologue(MethodDefinition wrapper, ILProcessor il)
|
|
{
|
|
|
|
DebugVariables vars = null;
|
|
if (il.Body.Method.Name != "GetError")
|
|
{
|
|
// Pull out the namespace name, method fullname will look
|
|
// something like "type namespace.class::method(type arg)"
|
|
var module = il.Body.Method.FullName;
|
|
module = module.Substring(module.IndexOf(' ') + 1);
|
|
module = module.Substring(0, module.IndexOf("::", StringComparison.Ordinal));
|
|
module = module.Substring(0, module.LastIndexOf('.'));
|
|
|
|
// Only works for Graphics modules due to hardcoded use of
|
|
// OpenTK.Graphics.GraphicsContext
|
|
if (module == "OpenTK.Graphics.OpenGL4" ||
|
|
module == "OpenTK.Graphics.OpenGL" ||
|
|
module == "OpenTK.Graphics.ES10" ||
|
|
module == "OpenTK.Graphics.ES11" ||
|
|
module == "OpenTK.Graphics.ES20" ||
|
|
module == "OpenTK.Graphics.ES30")
|
|
{
|
|
var errorHelperType = wrapper.Module.GetType(module, "ErrorHelper");
|
|
|
|
if (errorHelperType != null)
|
|
{
|
|
vars = new DebugVariables();
|
|
vars.ErrorHelperType = errorHelperType;
|
|
|
|
// GraphicsContext type
|
|
var graphicsContext = wrapper.Module.Types.First(
|
|
type => type.FullName == "OpenTK.Graphics.GraphicsContext");
|
|
|
|
// IGraphicsContext type
|
|
var iGraphicsContext = wrapper.Module.Types.First(
|
|
type => type.FullName == "OpenTK.Graphics.IGraphicsContext");
|
|
|
|
// Get the constructor that takes a GraphicsContext parameter
|
|
var ctor = vars.ErrorHelperType.GetConstructors().FirstOrDefault(
|
|
c => c.Parameters.Count == 1 &&
|
|
c.Parameters[0].ParameterType.FullName == iGraphicsContext.FullName);
|
|
|
|
if (ctor == null)
|
|
{
|
|
throw new InvalidOperationException(
|
|
String.Format(
|
|
"{0} does needs a constructor taking {1}",
|
|
errorHelperType,
|
|
graphicsContext));
|
|
}
|
|
|
|
// GraphicsContext.CurrentContext property getter
|
|
vars.Get_CurrentContext = graphicsContext.Methods.First(
|
|
method => method.Name == "get_CurrentContext");
|
|
|
|
vars.Set_ErrorChecking = graphicsContext.Methods.First(
|
|
method => method.Name == "set_ErrorChecking");
|
|
|
|
vars.ErrorHelperLocal = new VariableDefinition(vars.ErrorHelperType);
|
|
|
|
// using (new ErrorHelper(GraphicsContext.CurrentContext)) { ...
|
|
il.Body.Variables.Add(vars.ErrorHelperLocal);
|
|
il.Emit(OpCodes.Ldloca, vars.ErrorHelperLocal);
|
|
il.Emit(OpCodes.Call, vars.Get_CurrentContext);
|
|
il.Emit(OpCodes.Call, ctor);
|
|
|
|
vars.BeginTry = Instruction.Create(OpCodes.Nop);
|
|
il.Append(vars.BeginTry);
|
|
|
|
// Special case Begin to turn off error checking.
|
|
if (il.Body.Method.Name == "Begin")
|
|
{
|
|
il.Emit(OpCodes.Call, vars.Get_CurrentContext);
|
|
il.Emit(OpCodes.Ldc_I4_0);
|
|
il.Emit(OpCodes.Conv_I1);
|
|
il.Emit(OpCodes.Call, vars.Set_ErrorChecking);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return vars;
|
|
}
|
|
|
|
private static void EmitDebugEpilogue(MethodDefinition wrapper, ILProcessor il, DebugVariables vars)
|
|
{
|
|
if (vars != null)
|
|
{
|
|
var disposeMethod = vars.ErrorHelperType.Methods.First(
|
|
method => method.Name == "Dispose");
|
|
|
|
// Store then reload the result from the call
|
|
var resultLocal = new VariableDefinition(wrapper.ReturnType);
|
|
if (resultLocal.VariableType.FullName != Program.TypeVoid.FullName)
|
|
{
|
|
il.Body.Variables.Add(resultLocal);
|
|
il.Emit(OpCodes.Stloc, resultLocal);
|
|
}
|
|
|
|
// Special case End to turn on error checking.
|
|
if (il.Body.Method.Name == "End")
|
|
{
|
|
il.Emit(OpCodes.Call, vars.Get_CurrentContext);
|
|
il.Emit(OpCodes.Ldc_I4_1);
|
|
il.Emit(OpCodes.Conv_I1);
|
|
il.Emit(OpCodes.Call, vars.Set_ErrorChecking);
|
|
}
|
|
|
|
// We need a NOP to set up the finally handler range correctly.
|
|
var nopInstruction = Instruction.Create(OpCodes.Nop);
|
|
var loadInstruction = Instruction.Create(OpCodes.Ldloca, vars.ErrorHelperLocal);
|
|
var disposeInstruction = Instruction.Create(OpCodes.Call, disposeMethod);
|
|
var endFinallyInstruction = Instruction.Create(OpCodes.Endfinally);
|
|
var endTryInstruction = Instruction.Create(OpCodes.Leave, nopInstruction);
|
|
|
|
il.Append(endTryInstruction);
|
|
il.Append(loadInstruction);
|
|
il.Append(disposeInstruction);
|
|
il.Append(endFinallyInstruction);
|
|
il.Append(nopInstruction);
|
|
|
|
var finallyHandler = new ExceptionHandler(ExceptionHandlerType.Finally);
|
|
finallyHandler.TryStart = vars.BeginTry;
|
|
finallyHandler.TryEnd = loadInstruction;
|
|
finallyHandler.HandlerStart = loadInstruction;
|
|
finallyHandler.HandlerEnd = nopInstruction;
|
|
|
|
il.Body.ExceptionHandlers.Add(finallyHandler);
|
|
|
|
if (resultLocal.VariableType.FullName != Program.TypeVoid.FullName)
|
|
{
|
|
il.Emit(OpCodes.Ldloc, resultLocal);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void EmitReturnTypeWrapper(MethodDefinition wrapper, MethodDefinition native, MethodBody body, ILProcessor il)
|
|
{
|
|
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.ImportReference(mscorlib.MainModule.GetType("System.IntPtr").GetMethods()
|
|
.First(m =>
|
|
{
|
|
return
|
|
m.Name == "op_Explicit" &&
|
|
m.ReturnType.Name == "Void*";
|
|
}));
|
|
|
|
var string_constructor = wrapper.Module.ImportReference(mscorlib.MainModule.GetType("System.String").GetConstructors()
|
|
.First(m =>
|
|
{
|
|
var p = m.Parameters;
|
|
return p.Count > 0 && p[0].ParameterType.Name == "SByte*";
|
|
}));
|
|
|
|
il.Emit(OpCodes.Call, intptr_to_voidpointer);
|
|
il.Emit(OpCodes.Newobj, string_constructor);
|
|
}
|
|
else if (wrapper.ReturnType.Resolve().IsEnum)
|
|
{
|
|
// Nothing to do
|
|
}
|
|
else if (wrapper.ReturnType.Name == "Boolean" && native.ReturnType.Name == "Byte")
|
|
{
|
|
// Nothing to do
|
|
// It appears that a byte with 1 = true (GL_TRUE) and 0 = false (GL_FALSE)
|
|
// can be reinterpreted as a bool without a problem.
|
|
// Todo: maybe we should return (value == 0 ? false : true) just to be
|
|
// on the safe side?
|
|
}
|
|
else
|
|
{
|
|
Console.Error.WriteLine("Return wrapper for '{1}' not implemented yet ({0})", native.Name, wrapper.ReturnType.Name);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// nothing to do, the native call leaves the return value
|
|
// on the stack and we return that unmodified to the caller.
|
|
}
|
|
}
|
|
|
|
private static void EmitParameterEpilogues(MethodDefinition wrapper, MethodDefinition native, MethodBody body, ILProcessor il,
|
|
List<GeneratedVariableIdentifier> generatedVariables)
|
|
{
|
|
foreach (var p in wrapper.Parameters)
|
|
{
|
|
if (!p.ParameterType.IsArray && p.ParameterType.Name == "String&")
|
|
{
|
|
EmitStringOutEpilogue(wrapper, native, p, body, il, GetGeneratedVariable(generatedVariables, p.Name + "_string_ptr", body));
|
|
}
|
|
|
|
if (!p.ParameterType.IsArray && p.ParameterType.Name == "String")
|
|
{
|
|
EmitStringEpilogue(wrapper, p, body, il, GetGeneratedVariable(generatedVariables, p.Name + "_string_ptr", body));
|
|
}
|
|
|
|
if (p.ParameterType.IsArray && p.ParameterType.GetElementType().Name == "String")
|
|
{
|
|
EmitStringArrayEpilogue(wrapper, p, body, il, GetGeneratedVariable(generatedVariables, p.Name + "_string_array_ptr", body));
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Retrieves a generated variable by searching the given list by the variable's name and associated method body.
|
|
/// </summary>
|
|
/// <param name="variableIdentifiers"></param>
|
|
/// <param name="name"></param>
|
|
/// <param name="body"></param>
|
|
/// <returns></returns>
|
|
private static GeneratedVariableIdentifier GetGeneratedVariable(IEnumerable<GeneratedVariableIdentifier> variableIdentifiers, string name, MethodBody body)
|
|
{
|
|
return variableIdentifiers.FirstOrDefault(v => v.Name == name && v.Body == body &&
|
|
body.Variables.Contains(v.Definition));
|
|
}
|
|
|
|
private static GeneratedVariableIdentifier EmitStringOutParameter(MethodDefinition method, ParameterDefinition parameter, MethodBody body, ILProcessor il)
|
|
{
|
|
// void GetShaderInfoLog(..., out String foo)
|
|
// IntPtr foo_string_ptr;
|
|
// try {
|
|
// foo_string_ptr = Marshal.AllocHGlobal(count + 1);
|
|
// glGetShaderInfoLog(..., foo_string_ptr);
|
|
// foo = MarshalPtrToString(foo_string_ptr);
|
|
// }
|
|
// finally {
|
|
// Marshal.FreeHGlobal(foo_string_ptr);
|
|
// }
|
|
|
|
// Pop off the string parameter that would of just been loaded
|
|
il.Emit(OpCodes.Pop);
|
|
|
|
// Make sure we have imported Marshal::AllocHGlobal
|
|
var alloc_hglobal = method.Module.ImportReference(TypeMarshal.Methods.First(m => m.Name == "AllocHGlobal"));
|
|
|
|
// IntPtr ptr;
|
|
var variableDefinition = new VariableDefinition(TypeIntPtr);
|
|
body.Variables.Add(variableDefinition);
|
|
int stringPtrIndex = body.Variables.Count - 1;
|
|
|
|
GeneratedVariableIdentifier stringPtrVar = new GeneratedVariableIdentifier(body, variableDefinition, parameter.Name + "_string_ptr");
|
|
|
|
// ptr = Marshal.AllocHGlobal(count + 1);
|
|
var count = GetCountAttribute(parameter);
|
|
if (count == null)
|
|
{
|
|
// We need a count attribute so we know what size to make the
|
|
// string buffer. Currently every string out parameter has a
|
|
// count attribute but this check is in place to make things
|
|
// clearer if this case is ever hit.
|
|
throw new InvalidOperationException(string.Format("{0}({1}) doesn't have a count attribute", method.Name, parameter.Name));
|
|
}
|
|
|
|
if (count.Count != 0)
|
|
{
|
|
// Fixed size
|
|
il.Emit(OpCodes.Ldc_I4, count.Count);
|
|
}
|
|
else if (count.Parameter != null)
|
|
{
|
|
// Parameter sized
|
|
var countVariable = EmitCountVariable(method, body, il, count.Parameter);
|
|
il.Emit(OpCodes.Ldloc, countVariable.Index);
|
|
}
|
|
else if (count.Computed != null)
|
|
{
|
|
if (method.Name == "GetActiveVarying")
|
|
{
|
|
// GetActiveVaryingNV's name parameter has a count of "COMPSIZE(program,index,bufSize)" but really it should be bufSize.
|
|
var countVariable = EmitCountVariable(method, body, il, "bufSize");
|
|
il.Emit(OpCodes.Ldloc, countVariable.Index);
|
|
}
|
|
else
|
|
{
|
|
// Computed counts are hard and require manual reading of the specification for each one.
|
|
throw new NotSupportedException(string.Format("{0}({1}) requires a computed count: {2}", method.Name, parameter.Name, count.Computed));
|
|
}
|
|
}
|
|
|
|
il.Emit(OpCodes.Ldc_I4, 1);
|
|
il.Emit(OpCodes.Add);
|
|
il.Emit(OpCodes.Call, alloc_hglobal);
|
|
il.Emit(OpCodes.Stloc, stringPtrIndex);
|
|
il.Emit(OpCodes.Ldloc, stringPtrIndex);
|
|
|
|
// We'll emit the try-finally block in the epilogue implementation,
|
|
// because we haven't yet emitted all necessary instructions here.
|
|
|
|
return stringPtrVar;
|
|
}
|
|
|
|
private static void EmitStringOutEpilogue(MethodDefinition wrapper, MethodDefinition native,
|
|
ParameterDefinition parameter, MethodBody body, ILProcessor il, GeneratedVariableIdentifier generatedPtrVar)
|
|
{
|
|
if (generatedPtrVar == null)
|
|
{
|
|
throw new ArgumentNullException(nameof(generatedPtrVar));
|
|
}
|
|
|
|
// void GetShaderInfoLog(..., out String foo)
|
|
// IntPtr foo_string_ptr;
|
|
// try {
|
|
// foo_string_ptr = Marshal.AllocHGlobal(count + 1);
|
|
// glGetShaderInfoLog(..., foo_string_ptr);
|
|
// foo = MarshalPtrToString(foo_string_ptr);
|
|
// }
|
|
// finally {
|
|
// Marshal.FreeHGlobal(foo_string_ptr);
|
|
// }
|
|
|
|
// Make sure we have imported BindingsBase::MasrhalPtrToString and Marshal::FreeHGlobal
|
|
var ptr_to_str = wrapper.Module.ImportReference(TypeBindingsBase.Methods.First(m => m.Name == "MarshalPtrToString"));
|
|
var free_hglobal = wrapper.Module.ImportReference(TypeMarshal.Methods.First(m => m.Name == "FreeHGlobal"));
|
|
|
|
var block = new ExceptionHandler(ExceptionHandlerType.Finally);
|
|
block.TryStart = body.Instructions[0];
|
|
|
|
il.Emit(OpCodes.Ldarg, parameter.Index);
|
|
il.Emit(OpCodes.Ldloc, generatedPtrVar.Definition.Index);
|
|
il.Emit(OpCodes.Call, ptr_to_str);
|
|
il.Emit(OpCodes.Stind_Ref);
|
|
|
|
block.TryEnd = body.Instructions.Last();
|
|
block.HandlerStart = body.Instructions.Last();
|
|
|
|
il.Emit(OpCodes.Ldloc, generatedPtrVar.Definition.Index);
|
|
il.Emit(OpCodes.Call, free_hglobal);
|
|
|
|
block.HandlerEnd = body.Instructions.Last();
|
|
}
|
|
|
|
private static GeneratedVariableIdentifier EmitStringParameter(MethodDefinition wrapper, ParameterDefinition parameter, MethodBody body,
|
|
ILProcessor il)
|
|
{
|
|
// string marshaling:
|
|
// IntPtr ptr = MarshalStringToPtr(str);
|
|
// try { calli }
|
|
// finally { Marshal.FreeHGlobal(ptr); }
|
|
var marshal_str_to_ptr = wrapper.Module.ImportReference(TypeBindingsBase.Methods.First(m => m.Name == "MarshalStringToPtr"));
|
|
|
|
// IntPtr ptr;
|
|
var variableDefinition = new VariableDefinition(TypeIntPtr);
|
|
body.Variables.Add(variableDefinition);
|
|
int generatedPointerVarIndex = body.Variables.Count - 1;
|
|
|
|
GeneratedVariableIdentifier stringPtrVar = new GeneratedVariableIdentifier(body, variableDefinition, parameter.Name + "_string_ptr");
|
|
|
|
// ptr = Marshal.StringToHGlobalAnsi(str);
|
|
il.Emit(OpCodes.Call, marshal_str_to_ptr);
|
|
il.Emit(OpCodes.Stloc, generatedPointerVarIndex);
|
|
il.Emit(OpCodes.Ldloc, generatedPointerVarIndex);
|
|
|
|
// The finally block will be emitted in the function epilogue
|
|
return stringPtrVar;
|
|
}
|
|
|
|
private static void EmitStringEpilogue(MethodDefinition wrapper, ParameterDefinition parameter, MethodBody body,
|
|
ILProcessor il, GeneratedVariableIdentifier generatedPtrVar)
|
|
{
|
|
var free = wrapper.Module.ImportReference(TypeBindingsBase.Methods.First(m => m.Name == "FreeStringPtr"));
|
|
|
|
// FreeStringPtr(ptr)
|
|
il.Emit(OpCodes.Ldloc, generatedPtrVar.Definition.Index);
|
|
il.Emit(OpCodes.Call, free);
|
|
}
|
|
|
|
private static GeneratedVariableIdentifier EmitStringArrayParameter(MethodDefinition wrapper, ParameterDefinition parameter, MethodBody body,
|
|
ILProcessor il)
|
|
{
|
|
// string[] masrhaling:
|
|
// IntPtr ptr = MarshalStringArrayToPtr(strings);
|
|
// try { calli }
|
|
// finally { FreeStringArrayPtr(ptr); }
|
|
var marshal_str_array_to_ptr = wrapper.Module.ImportReference(TypeBindingsBase.Methods.First(m => m.Name == "MarshalStringArrayToPtr"));
|
|
|
|
// IntPtr ptr;
|
|
var variableDefinition = new VariableDefinition(TypeIntPtr);
|
|
body.Variables.Add(variableDefinition);
|
|
int generatedPointerVarIndex = body.Variables.Count - 1;
|
|
|
|
GeneratedVariableIdentifier stringArrayPtrVar = new GeneratedVariableIdentifier(body, variableDefinition, parameter.Name + "_string_array_ptr");
|
|
|
|
// ptr = MarshalStringArrayToPtr(strings);
|
|
il.Emit(OpCodes.Call, marshal_str_array_to_ptr);
|
|
il.Emit(OpCodes.Stloc, generatedPointerVarIndex);
|
|
il.Emit(OpCodes.Ldloc, generatedPointerVarIndex);
|
|
|
|
// The finally block will be emitted in the function epilogue
|
|
|
|
return stringArrayPtrVar;
|
|
}
|
|
|
|
private static void EmitStringArrayEpilogue(MethodDefinition wrapper, ParameterDefinition parameter, MethodBody body,
|
|
ILProcessor il, GeneratedVariableIdentifier generatedPtrVar)
|
|
{
|
|
if (generatedPtrVar == null)
|
|
{
|
|
throw new ArgumentNullException(nameof(generatedPtrVar));
|
|
}
|
|
|
|
// Note: only works for string vectors (1d arrays).
|
|
// We do not (and will probably never) support 2d or higher string arrays
|
|
var free = wrapper.Module.ImportReference(TypeBindingsBase.Methods.First(m => m.Name == "FreeStringArrayPtr"));
|
|
|
|
// FreeStringArrayPtr(string_array_ptr, string_array.Length)
|
|
|
|
// load string_array_ptr
|
|
il.Emit(OpCodes.Ldloc, generatedPtrVar.Definition.Index);
|
|
|
|
// load string_array.Length
|
|
il.Emit(OpCodes.Ldarg, parameter.Index);
|
|
il.Emit(OpCodes.Ldlen);
|
|
il.Emit(OpCodes.Conv_I4);
|
|
|
|
// call FreeStringArrayPtr
|
|
il.Emit(OpCodes.Call, free);
|
|
}
|
|
|
|
private static List<GeneratedVariableIdentifier> EmitConvenienceWrapper(MethodDefinition wrapper,
|
|
MethodDefinition native, int difference, MethodBody body, ILProcessor il)
|
|
{
|
|
if (wrapper.Parameters.Count > 2)
|
|
{
|
|
// Todo: emit all parameters bar the last two
|
|
throw new NotImplementedException();
|
|
}
|
|
|
|
|
|
List<GeneratedVariableIdentifier> generatedVariables = new List<GeneratedVariableIdentifier>();
|
|
if (wrapper.ReturnType.Name != "Void")
|
|
{
|
|
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(wrapper.ReturnType));
|
|
il.Emit(OpCodes.Ldc_I4, 1); // const int n = 1
|
|
il.Emit(OpCodes.Ldloca, body.Variables.Count - 1); // &buffers
|
|
}
|
|
else if (difference == 1)
|
|
{
|
|
// 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(wrapper.ReturnType));
|
|
|
|
generatedVariables = EmitParameters(wrapper, native, body, il);
|
|
il.Emit(OpCodes.Ldloca, body.Variables.Count - 1);
|
|
}
|
|
else
|
|
{
|
|
Console.Error.WriteLine("Unknown wrapper type for ({0})", native.Name);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (difference == 1)
|
|
{
|
|
// 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);
|
|
}
|
|
}
|
|
|
|
return generatedVariables;
|
|
}
|
|
|
|
private static object GetAttributeField(CustomAttribute attribute, string name)
|
|
{
|
|
try
|
|
{
|
|
var field = attribute.Fields.First(f => f.Name == name);
|
|
return field.Argument.Value;
|
|
}
|
|
catch (InvalidOperationException)
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private static CountAttribute GetCountAttribute(ParameterDefinition parameter)
|
|
{
|
|
var attribute = parameter.CustomAttributes
|
|
.FirstOrDefault(a => a.AttributeType.Name == "CountAttribute");
|
|
|
|
CountAttribute count = null;
|
|
if (attribute != null)
|
|
{
|
|
count = new CountAttribute();
|
|
count.Count = (int)(GetAttributeField(attribute, "Count") ?? 0);
|
|
count.Parameter = (string)(GetAttributeField(attribute, "Parameter"));
|
|
count.Computed = (string)(GetAttributeField(attribute, "Computed"));
|
|
}
|
|
return count;
|
|
}
|
|
|
|
|
|
private static VariableDefinition EmitCountVariable(MethodDefinition method, MethodBody body, ILProcessor il, string countParameter)
|
|
{
|
|
var countVariable = new VariableDefinition(TypeInt32);
|
|
body.Variables.Add(countVariable);
|
|
|
|
// Parameter will either by a simple name, a dereference of a name
|
|
// like "*name" or an expression like "name*5"
|
|
var parameter = method.Parameters.FirstOrDefault(
|
|
param => param.Name == countParameter);
|
|
if (parameter != null)
|
|
{
|
|
il.Emit(OpCodes.Ldarg, parameter.Index);
|
|
il.Emit(OpCodes.Stloc, countVariable.Index);
|
|
}
|
|
else if (countParameter[0] == '*')
|
|
{
|
|
var pointerParam = method.Parameters.FirstOrDefault(
|
|
param => param.Name == countParameter.Substring(1));
|
|
|
|
il.Emit(OpCodes.Ldarg, pointerParam.Index);
|
|
il.Emit(OpCodes.Ldind_I4);
|
|
}
|
|
else
|
|
{
|
|
var operands = countParameter.Split('*');
|
|
parameter = method.Parameters.FirstOrDefault(
|
|
param => param.Name == operands[0]);
|
|
var scale = int.Parse(operands[1]);
|
|
|
|
il.Emit(OpCodes.Ldarg, parameter.Index);
|
|
il.Emit(OpCodes.Ldc_I4, scale);
|
|
il.Emit(OpCodes.Mul);
|
|
il.Emit(OpCodes.Stloc, countVariable.Index);
|
|
}
|
|
return countVariable;
|
|
}
|
|
|
|
private static List<GeneratedVariableIdentifier> EmitParameters(MethodDefinition method, MethodDefinition native, MethodBody body, ILProcessor il)
|
|
{
|
|
List<GeneratedVariableIdentifier> generatedVariables = new List<GeneratedVariableIdentifier>();
|
|
for (int i = 0; i < method.Parameters.Count; i++)
|
|
{
|
|
var parameter = method.Parameters[i];
|
|
var p = method.Module.ImportReference(method.Parameters[i].ParameterType);
|
|
il.Emit(OpCodes.Ldarg, i);
|
|
|
|
if (p.Name.Contains("Int32") && native.Parameters[i].ParameterType.Name.Contains("IntPtr"))
|
|
{
|
|
// This is a convenience Int32 overload for an IntPtr (size_t) parameter.
|
|
// We need to convert the loaded argument to IntPtr.
|
|
il.Emit(OpCodes.Conv_I);
|
|
}
|
|
else if (p.Name == "String&" && !p.IsArray)
|
|
{
|
|
generatedVariables.Add(EmitStringOutParameter(method, parameter, body, il));
|
|
}
|
|
else if (p.Name == "String" && !p.IsArray)
|
|
{
|
|
generatedVariables.Add(EmitStringParameter(method, parameter, body, il));
|
|
}
|
|
else if (p.IsByReference)
|
|
{
|
|
body.Variables.Add(new VariableDefinition(new PinnedType(p)));
|
|
var index = body.Variables.Count - 1;
|
|
il.Emit(OpCodes.Stloc, index);
|
|
il.Emit(OpCodes.Ldloc, index);
|
|
il.Emit(OpCodes.Conv_I);
|
|
}
|
|
else if (p.IsArray)
|
|
{
|
|
if (p.Name != "String[]")
|
|
{
|
|
// .Net treats 1d arrays differently than higher rank arrays.
|
|
// 1d arrays are directly supported by instructions such as ldlen and ldelema.
|
|
// Higher rank arrays must be accessed through System.Array methods such as get_Length.
|
|
// 1d array:
|
|
// check array is not null
|
|
// check ldlen array > 0
|
|
// ldc.i4.0
|
|
// ldelema
|
|
// 2d array:
|
|
// check array is not null
|
|
// check array.get_Length() > 0
|
|
// ldc.i4.0
|
|
// ldc.i4.0
|
|
// call instance T& T[0..., 0...]::Address(int32, int32)
|
|
// Mono treats everything as a 1d array.
|
|
// Interestingly, the .Net approach works on both Mono and .Net.
|
|
// The Mono approach fails when using high-rank arrays on .Net.
|
|
// We should report a bug to http://bugzilla.xamarin.com
|
|
|
|
// Pin the array and pass the address
|
|
// of its first element.
|
|
var array = (ArrayType)p;
|
|
var element_type = p.GetElementType();
|
|
body.Variables.Add(new VariableDefinition(new PinnedType(new ByReferenceType(element_type))));
|
|
int pinned_index = body.Variables.Count - 1;
|
|
|
|
var empty = il.Create(OpCodes.Ldc_I4, 0);
|
|
var pin = il.Create(OpCodes.Ldarg, i);
|
|
var end = il.Create(OpCodes.Stloc, pinned_index);
|
|
|
|
// if (array == null) goto empty
|
|
il.Emit(OpCodes.Brfalse, empty);
|
|
|
|
// else if (array.Length != 0) goto pin
|
|
il.Emit(OpCodes.Ldarg, i);
|
|
if (array.Rank == 1)
|
|
{
|
|
il.Emit(OpCodes.Ldlen);
|
|
il.Emit(OpCodes.Conv_I4);
|
|
}
|
|
else
|
|
{
|
|
var get_length = method.Module.ImportReference(
|
|
mscorlib.MainModule.GetType("System.Array").Methods.First(m => m.Name == "get_Length"));
|
|
il.Emit(OpCodes.Callvirt, get_length);
|
|
}
|
|
il.Emit(OpCodes.Brtrue, pin);
|
|
|
|
// empty: IntPtr ptr = IntPtr.Zero
|
|
il.Append(empty);
|
|
il.Emit(OpCodes.Conv_U);
|
|
il.Emit(OpCodes.Br, end);
|
|
|
|
// pin: &array[0]
|
|
il.Append(pin);
|
|
if (array.Rank == 1)
|
|
{
|
|
// 1d array (vector), address is taken by ldelema
|
|
il.Emit(OpCodes.Ldc_I4, 0);
|
|
il.Emit(OpCodes.Ldelema, element_type);
|
|
}
|
|
else
|
|
{
|
|
// 2d-3d array, address must be taken as follows:
|
|
// call instance T& T[0..., 0..., 0...]::Address(int, int, int)
|
|
ByReferenceType t_ref = array.ElementType.MakeByReferenceType();
|
|
MethodReference get_address = new MethodReference("Address", t_ref, array);
|
|
for (int r = 0; r < array.Rank; r++)
|
|
{
|
|
get_address.Parameters.Add(new ParameterDefinition(TypeInt32));
|
|
}
|
|
get_address.HasThis = true;
|
|
|
|
// emit the get_address call
|
|
for (int r = 0; r < array.Rank; r++)
|
|
{
|
|
il.Emit(OpCodes.Ldc_I4, 0);
|
|
}
|
|
il.Emit(OpCodes.Call, get_address);
|
|
}
|
|
|
|
// end: fixed (IntPtr ptr = &array[0])
|
|
il.Append(end);
|
|
il.Emit(OpCodes.Ldloc, pinned_index);
|
|
il.Emit(OpCodes.Conv_I);
|
|
}
|
|
else
|
|
{
|
|
generatedVariables.Add(EmitStringArrayParameter(method, parameter, body, il));
|
|
}
|
|
}
|
|
}
|
|
|
|
return generatedVariables;
|
|
}
|
|
|
|
private 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);
|
|
}
|
|
|
|
private static void EmitCalli(ILProcessor il, MethodReference reference)
|
|
{
|
|
var signature = new CallSite(reference.ReturnType)
|
|
{
|
|
CallingConvention = MethodCallingConvention.StdCall,
|
|
};
|
|
|
|
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);
|
|
}
|
|
|
|
private static void EmitCall(ILProcessor il, MethodReference reference)
|
|
{
|
|
il.Emit(OpCodes.Call, reference);
|
|
}
|
|
}
|
|
}
|