Utilities/PInvokeDelegateFactory.cs
author moel.mich
Sat, 20 Feb 2010 19:51:10 +0000
changeset 56 5cb7eb5bf628
permissions -rw-r--r--
Improved Winbond temperature reading. Temperatures create by adding PECI Agent values (delta to TCC Activation Temperature) to a (possibly uncalibrated) TBase are not read. Direct reading temperatures from sensor report register if available. Added lower bound for temperatures on Winbond chips. Nvidia GPUs are now displayed even if they do not have any sensors.
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
}