Utilities/PInvokeDelegateFactory.cs
author moel.mich
Tue, 09 Mar 2010 22:27:10 +0000
changeset 79 9cdbe1d8d12a
permissions -rw-r--r--
Changed the CPU clock calculation. If no invariant TSC is available, then the max CPU clock is estimated at startup under load, otherwise an average over one second is used.
moel@1
     1
/*
moel@1
     2
  
moel@1
     3
  Version: MPL 1.1/GPL 2.0/LGPL 2.1
moel@1
     4
moel@1
     5
  The contents of this file are subject to the Mozilla Public License Version
moel@1
     6
  1.1 (the "License"); you may not use this file except in compliance with
moel@1
     7
  the License. You may obtain a copy of the License at
moel@1
     8
 
moel@1
     9
  http://www.mozilla.org/MPL/
moel@1
    10
moel@1
    11
  Software distributed under the License is distributed on an "AS IS" basis,
moel@1
    12
  WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
moel@1
    13
  for the specific language governing rights and limitations under the License.
moel@1
    14
moel@1
    15
  The Original Code is the Open Hardware Monitor code.
moel@1
    16
moel@1
    17
  The Initial Developer of the Original Code is 
moel@1
    18
  Michael Möller <m.moeller@gmx.ch>.
moel@1
    19
  Portions created by the Initial Developer are Copyright (C) 2009-2010
moel@1
    20
  the Initial Developer. All Rights Reserved.
moel@1
    21
moel@1
    22
  Contributor(s):
moel@1
    23
moel@1
    24
  Alternatively, the contents of this file may be used under the terms of
moel@1
    25
  either the GNU General Public License Version 2 or later (the "GPL"), or
moel@1
    26
  the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
moel@1
    27
  in which case the provisions of the GPL or the LGPL are applicable instead
moel@1
    28
  of those above. If you wish to allow use of your version of this file only
moel@1
    29
  under the terms of either the GPL or the LGPL, and not to allow others to
moel@1
    30
  use your version of this file under the terms of the MPL, indicate your
moel@1
    31
  decision by deleting the provisions above and replace them with the notice
moel@1
    32
  and other provisions required by the GPL or the LGPL. If you do not delete
moel@1
    33
  the provisions above, a recipient may use your version of this file under
moel@1
    34
  the terms of any one of the MPL, the GPL or the LGPL.
moel@1
    35
 
moel@1
    36
*/
moel@1
    37
moel@1
    38
using System;
moel@1
    39
using System.Collections.Generic;
moel@1
    40
using System.Reflection;
moel@1
    41
using System.Reflection.Emit;
moel@1
    42
using System.Runtime.InteropServices;
moel@1
    43
moel@1
    44
namespace OpenHardwareMonitor.Hardware {
moel@1
    45
moel@1
    46
  public sealed class PInvokeDelegateFactory {
moel@1
    47
moel@1
    48
    private static AssemblyBuilder assemblyBuilder;
moel@1
    49
    private static ModuleBuilder moduleBuilder;
moel@1
    50
moel@1
    51
    private static IDictionary<DllImportAttribute, Type> wrapperTypes =
moel@1
    52
      new Dictionary<DllImportAttribute, Type>();
moel@1
    53
moel@1
    54
    static PInvokeDelegateFactory() {
moel@1
    55
moel@1
    56
      AssemblyName assemblyName = new AssemblyName();
moel@1
    57
      assemblyName.Name = "PInvokeDelegateFactoryInternalAssembly";
moel@1
    58
moel@1
    59
      assemblyBuilder =
moel@1
    60
        AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName,
moel@1
    61
        AssemblyBuilderAccess.Run);
moel@1
    62
moel@1
    63
      moduleBuilder = assemblyBuilder.DefineDynamicModule(
moel@1
    64
        "PInvokeDelegateFactoryInternalModule");
moel@1
    65
    }
moel@1
    66
moel@1
    67
    private PInvokeDelegateFactory() { }
moel@1
    68
moel@1
    69
    public static void CreateDelegate<T>(DllImportAttribute dllImportAttribute,
moel@1
    70
      out T newDelegate) where T : class 
moel@1
    71
    {
moel@1
    72
      Type wrapperType;
moel@1
    73
      wrapperTypes.TryGetValue(dllImportAttribute, out wrapperType);
moel@1
    74
moel@1
    75
      if (wrapperType == null) {
moel@1
    76
        wrapperType = CreateWrapperType(typeof(T), dllImportAttribute);
moel@1
    77
        wrapperTypes.Add(dllImportAttribute, wrapperType);
moel@1
    78
      }
moel@1
    79
moel@1
    80
      newDelegate = Delegate.CreateDelegate(typeof(T), wrapperType,
moel@1
    81
        dllImportAttribute.EntryPoint) as T;
moel@1
    82
    }
moel@1
    83
moel@1
    84
moel@1
    85
    private static Type CreateWrapperType(Type delegateType,
moel@1
    86
      DllImportAttribute dllImportAttribute) {
moel@1
    87
moel@1
    88
      TypeBuilder typeBuilder = moduleBuilder.DefineType(
moel@1
    89
        "PInvokeDelegateFactoryInternalWrapperType" + wrapperTypes.Count);
moel@1
    90
moel@1
    91
      MethodInfo methodInfo = delegateType.GetMethod("Invoke");
moel@1
    92
moel@1
    93
      ParameterInfo[] parameterInfos = methodInfo.GetParameters();
moel@1
    94
      int parameterCount = parameterInfos.GetLength(0);
moel@1
    95
moel@1
    96
      Type[] parameterTypes = new Type[parameterCount];
moel@1
    97
      for (int i = 0; i < parameterCount; i++)
moel@1
    98
        parameterTypes[i] = parameterInfos[i].ParameterType;
moel@1
    99
moel@1
   100
      MethodBuilder methodBuilder = typeBuilder.DefinePInvokeMethod(
moel@1
   101
        dllImportAttribute.EntryPoint, dllImportAttribute.Value,
moel@1
   102
        MethodAttributes.Public | MethodAttributes.Static |
moel@1
   103
        MethodAttributes.PinvokeImpl, CallingConventions.Standard,
moel@1
   104
        methodInfo.ReturnType, parameterTypes,
moel@1
   105
        dllImportAttribute.CallingConvention,
moel@1
   106
        dllImportAttribute.CharSet);
moel@1
   107
moel@1
   108
      foreach (ParameterInfo parameterInfo in parameterInfos)
moel@1
   109
        methodBuilder.DefineParameter(parameterInfo.Position + 1,
moel@1
   110
          parameterInfo.Attributes, parameterInfo.Name);
moel@1
   111
moel@1
   112
      if (dllImportAttribute.PreserveSig)
moel@1
   113
        methodBuilder.SetImplementationFlags(MethodImplAttributes.PreserveSig);
moel@1
   114
moel@1
   115
      return typeBuilder.CreateType();
moel@1
   116
    }
moel@1
   117
  }
moel@1
   118
}