Hardware/ATI/ADL.cs
author moel.mich
Thu, 20 Jan 2011 21:31:54 +0000
changeset 247 6dc755f1970e
parent 200 50de2faf3336
child 333 c34cbcc7df66
permissions -rw-r--r--
Added a minimal control interface to allow manual fan control and implemented the interface for ATI GPUs.
     1 /*
     2   
     3   Version: MPL 1.1/GPL 2.0/LGPL 2.1
     4 
     5   The contents of this file are subject to the Mozilla Public License Version
     6   1.1 (the "License"); you may not use this file except in compliance with
     7   the License. You may obtain a copy of the License at
     8  
     9   http://www.mozilla.org/MPL/
    10 
    11   Software distributed under the License is distributed on an "AS IS" basis,
    12   WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
    13   for the specific language governing rights and limitations under the License.
    14 
    15   The Original Code is the Open Hardware Monitor code.
    16 
    17   The Initial Developer of the Original Code is 
    18   Michael Möller <m.moeller@gmx.ch>.
    19   Portions created by the Initial Developer are Copyright (C) 2009-2010
    20   the Initial Developer. All Rights Reserved.
    21 
    22   Contributor(s):
    23 
    24   Alternatively, the contents of this file may be used under the terms of
    25   either the GNU General Public License Version 2 or later (the "GPL"), or
    26   the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
    27   in which case the provisions of the GPL or the LGPL are applicable instead
    28   of those above. If you wish to allow use of your version of this file only
    29   under the terms of either the GPL or the LGPL, and not to allow others to
    30   use your version of this file under the terms of the MPL, indicate your
    31   decision by deleting the provisions above and replace them with the notice
    32   and other provisions required by the GPL or the LGPL. If you do not delete
    33   the provisions above, a recipient may use your version of this file under
    34   the terms of any one of the MPL, the GPL or the LGPL.
    35  
    36 */
    37 
    38 using System;
    39 using System.Runtime.InteropServices;
    40 
    41 namespace OpenHardwareMonitor.Hardware.ATI {
    42   
    43   [StructLayout(LayoutKind.Sequential)]
    44   internal struct ADLAdapterInfo {
    45     public int Size;
    46     public int AdapterIndex;
    47     [MarshalAs(UnmanagedType.ByValTStr, SizeConst = ADL.ADL_MAX_PATH)]
    48     public string UDID;
    49     public int BusNumber;
    50     public int DeviceNumber;
    51     public int FunctionNumber;
    52     public int VendorID;
    53     [MarshalAs(UnmanagedType.ByValTStr, SizeConst = ADL.ADL_MAX_PATH)]
    54     public string AdapterName;
    55     [MarshalAs(UnmanagedType.ByValTStr, SizeConst = ADL.ADL_MAX_PATH)]
    56     public string DisplayName;
    57     public int Present;
    58     public int Exist;
    59     [MarshalAs(UnmanagedType.ByValTStr, SizeConst = ADL.ADL_MAX_PATH)]
    60     public string DriverPath;
    61     [MarshalAs(UnmanagedType.ByValTStr, SizeConst = ADL.ADL_MAX_PATH)]
    62     public string DriverPathExt;
    63     [MarshalAs(UnmanagedType.ByValTStr, SizeConst = ADL.ADL_MAX_PATH)]
    64     public string PNPString;
    65     public int OSDisplayIndex;
    66   }
    67 
    68   [StructLayout(LayoutKind.Sequential)]
    69   internal struct ADLPMActivity {
    70     public int Size;
    71     public int EngineClock;
    72     public int MemoryClock;
    73     public int Vddc;
    74     public int ActivityPercent;
    75     public int CurrentPerformanceLevel;
    76     public int CurrentBusSpeed;
    77     public int CurrentBusLanes;
    78     public int MaximumBusLanes;
    79     public int Reserved;
    80   }
    81 
    82   [StructLayout(LayoutKind.Sequential)]
    83   internal struct ADLTemperature {
    84     public int Size;
    85     public int Temperature;
    86   }
    87 
    88   [StructLayout(LayoutKind.Sequential)]
    89   internal struct ADLFanSpeedValue {
    90     public int Size;
    91     public int SpeedType;
    92     public int FanSpeed;
    93     public int Flags;
    94   }
    95 
    96   [StructLayout(LayoutKind.Sequential)]
    97   internal struct ADLFanSpeedInfo {
    98     public int Size;
    99     public int Flags;
   100     public int MinPercent;
   101     public int MaxPercent;
   102     public int MinRPM;
   103     public int MaxRPM;
   104   }
   105 
   106   internal class ADL {
   107     public const int ADL_MAX_PATH = 256;
   108     public const int ADL_MAX_ADAPTERS = 40;
   109     public const int ADL_MAX_DISPLAYS = 40;
   110     public const int ADL_MAX_DEVICENAME = 32;
   111     public const int ADL_OK = 0;
   112     public const int ADL_ERR = -1;
   113     public const int ADL_DRIVER_OK = 0;
   114     public const int ADL_MAX_GLSYNC_PORTS = 8;
   115     public const int ADL_MAX_GLSYNC_PORT_LEDS = 8;
   116     public const int ADL_MAX_NUM_DISPLAYMODES = 1024;
   117 
   118     public const int ADL_DL_FANCTRL_SPEED_TYPE_PERCENT = 1;
   119     public const int ADL_DL_FANCTRL_SPEED_TYPE_RPM = 2;
   120 
   121     public const int ADL_DL_FANCTRL_SUPPORTS_PERCENT_READ = 1;
   122     public const int ADL_DL_FANCTRL_SUPPORTS_PERCENT_WRITE = 2;
   123     public const int ADL_DL_FANCTRL_SUPPORTS_RPM_READ = 4;
   124     public const int ADL_DL_FANCTRL_SUPPORTS_RPM_WRITE = 8;
   125     public const int ADL_DL_FANCTRL_FLAG_USER_DEFINED_SPEED = 1;
   126 
   127     public const int ATI_VENDOR_ID1 = 1002;
   128     public const int ATI_VENDOR_ID2 = 0x1002;
   129 
   130     private delegate int ADL_Main_Control_CreateDelegate(
   131       ADL_Main_Memory_AllocDelegate callback, int enumConnectedAdapters);
   132     private delegate int ADL_Adapter_AdapterInfo_GetDelegate(IntPtr info,
   133       int size);
   134 
   135     public delegate int ADL_Main_Control_DestroyDelegate();
   136     public delegate int ADL_Adapter_NumberOfAdapters_GetDelegate(
   137       ref int numAdapters);    
   138     public delegate int ADL_Adapter_ID_GetDelegate(int adapterIndex,
   139       out int adapterID);
   140     public delegate int ADL_Display_AdapterID_GetDelegate(int adapterIndex,
   141       out int adapterID);      	
   142     public delegate int ADL_Adapter_Active_GetDelegate(int adapterIndex,
   143       out int status);
   144     public delegate int ADL_Overdrive5_CurrentActivity_GetDelegate(
   145       int iAdapterIndex, ref ADLPMActivity activity);
   146     public delegate int ADL_Overdrive5_Temperature_GetDelegate(int adapterIndex,
   147         int thermalControllerIndex, ref ADLTemperature temperature);
   148     public delegate int ADL_Overdrive5_FanSpeed_GetDelegate(int adapterIndex,
   149         int thermalControllerIndex, ref	ADLFanSpeedValue fanSpeedValue);
   150     public delegate int ADL_Overdrive5_FanSpeedInfo_GetDelegate(
   151       int adapterIndex, int thermalControllerIndex,
   152       ref ADLFanSpeedInfo fanSpeedInfo);
   153     public delegate int ADL_Overdrive5_FanSpeedToDefault_SetDelegate(
   154       int adapterIndex, int thermalControllerIndex);
   155     public delegate int ADL_Overdrive5_FanSpeed_SetDelegate(int adapterIndex,
   156       int thermalControllerIndex, ref	ADLFanSpeedValue fanSpeedValue);
   157 
   158     private static ADL_Main_Control_CreateDelegate
   159       _ADL_Main_Control_Create;
   160     private static ADL_Adapter_AdapterInfo_GetDelegate
   161       _ADL_Adapter_AdapterInfo_Get;
   162 
   163     public static ADL_Main_Control_DestroyDelegate
   164       ADL_Main_Control_Destroy;
   165     public static ADL_Adapter_NumberOfAdapters_GetDelegate
   166       ADL_Adapter_NumberOfAdapters_Get;
   167     public static ADL_Adapter_ID_GetDelegate 
   168       _ADL_Adapter_ID_Get;
   169     public static ADL_Display_AdapterID_GetDelegate 
   170       _ADL_Display_AdapterID_Get;
   171     public static ADL_Adapter_Active_GetDelegate 
   172       ADL_Adapter_Active_Get;
   173     public static ADL_Overdrive5_CurrentActivity_GetDelegate
   174       ADL_Overdrive5_CurrentActivity_Get;
   175     public static ADL_Overdrive5_Temperature_GetDelegate
   176       ADL_Overdrive5_Temperature_Get;
   177     public static ADL_Overdrive5_FanSpeed_GetDelegate
   178       ADL_Overdrive5_FanSpeed_Get;
   179     public static ADL_Overdrive5_FanSpeedInfo_GetDelegate
   180       ADL_Overdrive5_FanSpeedInfo_Get;
   181     public static ADL_Overdrive5_FanSpeedToDefault_SetDelegate
   182       ADL_Overdrive5_FanSpeedToDefault_Set;
   183     public static ADL_Overdrive5_FanSpeed_SetDelegate
   184       ADL_Overdrive5_FanSpeed_Set;
   185 
   186     private static string dllName;
   187 
   188     private static void GetDelegate<T>(string entryPoint, out T newDelegate)
   189       where T : class 
   190     {
   191       DllImportAttribute attribute = new DllImportAttribute(dllName);
   192       attribute.CallingConvention = CallingConvention.StdCall;
   193       attribute.PreserveSig = true;
   194       attribute.EntryPoint = entryPoint;
   195       PInvokeDelegateFactory.CreateDelegate(attribute, out newDelegate);
   196     }
   197 
   198     private static void CreateDelegates(string name) {
   199       int p = (int)Environment.OSVersion.Platform;
   200       if ((p == 4) || (p == 128))
   201         dllName = name + ".so";
   202       else
   203         dllName = name + ".dll";
   204 
   205       GetDelegate("ADL_Main_Control_Create",
   206         out _ADL_Main_Control_Create);
   207       GetDelegate("ADL_Adapter_AdapterInfo_Get",
   208         out _ADL_Adapter_AdapterInfo_Get);
   209       GetDelegate("ADL_Main_Control_Destroy",
   210         out ADL_Main_Control_Destroy);
   211       GetDelegate("ADL_Adapter_NumberOfAdapters_Get",
   212         out ADL_Adapter_NumberOfAdapters_Get);
   213       GetDelegate("ADL_Adapter_ID_Get",
   214         out _ADL_Adapter_ID_Get);
   215       GetDelegate("ADL_Display_AdapterID_Get", 
   216         out _ADL_Display_AdapterID_Get);
   217       GetDelegate("ADL_Adapter_Active_Get",
   218         out ADL_Adapter_Active_Get);
   219       GetDelegate("ADL_Overdrive5_CurrentActivity_Get",
   220         out ADL_Overdrive5_CurrentActivity_Get);
   221       GetDelegate("ADL_Overdrive5_Temperature_Get",
   222         out ADL_Overdrive5_Temperature_Get);
   223       GetDelegate("ADL_Overdrive5_FanSpeed_Get",
   224         out ADL_Overdrive5_FanSpeed_Get);
   225       GetDelegate("ADL_Overdrive5_FanSpeedInfo_Get",
   226         out ADL_Overdrive5_FanSpeedInfo_Get);
   227       GetDelegate("ADL_Overdrive5_FanSpeedToDefault_Set",
   228         out ADL_Overdrive5_FanSpeedToDefault_Set);
   229       GetDelegate("ADL_Overdrive5_FanSpeed_Set",
   230         out ADL_Overdrive5_FanSpeed_Set);
   231     }
   232 
   233     static ADL() {
   234       CreateDelegates("atiadlxx");
   235     }
   236 
   237     private ADL() { }
   238 
   239     public static int ADL_Main_Control_Create(int enumConnectedAdapters) {
   240       try {
   241         try {
   242           return _ADL_Main_Control_Create(Main_Memory_Alloc,
   243             enumConnectedAdapters);
   244         } catch {
   245           CreateDelegates("atiadlxy");
   246           return _ADL_Main_Control_Create(Main_Memory_Alloc,
   247             enumConnectedAdapters);
   248         }
   249       } catch {
   250         return ADL_ERR;
   251       }
   252     }
   253 
   254     public static int ADL_Adapter_AdapterInfo_Get(ADLAdapterInfo[] info) {
   255       int elementSize = Marshal.SizeOf(typeof(ADLAdapterInfo));
   256       int size = info.Length * elementSize;
   257       IntPtr ptr = Marshal.AllocHGlobal(size);
   258       int result = _ADL_Adapter_AdapterInfo_Get(ptr, size);
   259       for (int i = 0; i < info.Length; i++)
   260         info[i] = (ADLAdapterInfo)
   261           Marshal.PtrToStructure((IntPtr)((long)ptr + i * elementSize),
   262           typeof(ADLAdapterInfo));
   263       Marshal.FreeHGlobal(ptr);
   264       return result;
   265     }
   266 
   267     public static int ADL_Adapter_ID_Get(int adapterIndex,
   268       out int adapterID) {
   269       try {
   270         return _ADL_Adapter_ID_Get(adapterIndex, out adapterID);
   271       } catch (EntryPointNotFoundException) {
   272         try {
   273           return _ADL_Display_AdapterID_Get(adapterIndex, out adapterID);
   274         } catch (EntryPointNotFoundException) {
   275           adapterID = 1;
   276           return ADL_OK;
   277         }
   278       }
   279     }
   280 
   281     private delegate IntPtr ADL_Main_Memory_AllocDelegate(int size);
   282 
   283     // create a Main_Memory_Alloc delegate and keep it alive
   284     private static ADL_Main_Memory_AllocDelegate Main_Memory_Alloc =
   285       delegate(int size) {
   286         return Marshal.AllocHGlobal(size);
   287       };
   288 
   289     private static void Main_Memory_Free(IntPtr buffer) {
   290       if (IntPtr.Zero != buffer)
   291         Marshal.FreeHGlobal(buffer);
   292     }
   293   }
   294 }