Type a page name and press Enter. You'll jump to the page if it exists, or you can create it if it doesn't.
To create a page in a module other than advapi32, prefix the name with the module name and a period.
RegOpenKeyEx (advapi32)
.
C# Signature:
[DllImport("advapi32.dll", CharSet = CharSet.Auto)]
public static extern int RegOpenKeyEx(
UIntPtr hKey,
string subKey,
int ulOptions,
int samDesired,
out UIntPtr hkResult);
VB Signature:
Declare Auto Function RegOpenKeyEx Lib "advapi32.dll" (
ByVal hKey As IntPtr,
ByVal lpSubKey As String,
ByVal ulOptions As Integer,
ByVal samDesired As Integer,
ByRef phkResult As Integer
) As Integer
User-Defined Types:
None.
Change Notes:
- Changed IntPtr to UIntPtr: When invoking with IntPtr for the handles, you will run into an Overflow. UIntPtr is the right choice if you wish this to work correctly on 32 and 64 bit platforms.
Notes:
The current managed versions of OpenSubKey do not support read/writing from the different registry "views" on 64-bit operating systems. So, you're forced to the the APIs to accomplish this feat. See the example below for a "drop in" replacement for OpenSubKey
If the function fails, the return value is a nonzero error code defined in Winerror.h. You can use the FormatMessage function with the FORMAT_MESSAGE_FROM_SYSTEM flag to get a generic description of the error.
uint options: reserved, must be zero.
int sam: access mask that specifies the desired access rights to the key.
values for hKey (C#):
public static UIntPtr HKEY_LOCAL_MACHINE = new UIntPtr(0x80000002u);
public static UIntPtr HKEY_CURRENT_USER = new UIntPtr(0x80000001u);
For compatibility with 16-bit versions of Windows only 'RegOpenKey' is available.
Tips & Tricks:
Please add some!!
Sample Code (VB):
Public Enum RegWow64Options As Integer
None = 0
KEY_WOW64_64KEY = &H100
KEY_WOW64_32KEY = &H200
End Enum
Public Enum RegistryRights As Integer
ReadKey = 131097
WriteKey = 131078
End Enum
' overloaded, see below
Public Function OpenSubKey(ByVal Name As String) As RegistryKey
Return OpenSubKey(Name, False, RegWow64Options.None)
End Function
' overloaded, see below
Public Function OpenSubKey(ByVal Name As String, ByVal Writeable As Boolean) As RegistryKey
Return OpenSubKey(Name, Writeable, RegWow64Options.None)
End Function
' open a SubKey
Public Function OpenSubKey(ByVal Name As String, ByVal Writeable As Boolean, ByVal Options As RegWow64Options) As RegistryKey
Dim ret, Rights As Integer
Dim hSubKey As IntPtr
' quick sanity check
If hKey.Equals(IntPtr.Zero) Then
Throw New ApplicationException("Cannot access a closed registry key")
End If
Rights = RegistryRights.ReadKey
If Writeable Then
Rights = RegistryRights.WriteKey
End If
' Ignore the 64-bit flags when running from <= Win2k
If CDbl(Environment.OSVersion.Version.Major & "." & Environment.OSVersion.Version.Minor) <= 5.0 Then
Options = RegWow64Options.None
End If
ret = RegOpenKeyEx(hKey, Name, 0, Rights Or Options, hSubKey)
If ret <> 0 Then
Return Nothing
End If
Dim ans As New RegistryKey
ans.hKey = hSubKey
ans.RegKeyName = Name
ans.IsRootHive = False
Return ans
End Function
Note that because this uses undocumented private functions, it could break in the future. Apparently .NET 4.0 will include native functionality for reading 64-bit registry from 32-bit processes etc.
Public Shared Function PointerToRegistryKey(ByVal hKey As IntPtr, ByVal writable As Boolean, ByVal ownsHandle As Boolean) As RegistryKey
' Create a SafeHandles.SafeRegistryHandle from this pointer - this is a private class
Dim privateConstructors = Reflection.BindingFlags.Instance Or Reflection.BindingFlags.NonPublic
Dim safeRegistryHandleType = GetType(SafeHandles.SafeHandleZeroOrMinusOneIsInvalid).Assembly.GetType("Microsoft.Win32.SafeHandles.SafeRegistryHandle")
Dim safeRegistryHandleConstructorTypes As System.Type() = {GetType(IntPtr), GetType(Boolean)}
Dim safeRegistryHandleConstructor = safeRegistryHandleType.GetConstructor(privateConstructors, Nothing, safeRegistryHandleConstructorTypes, Nothing)
Dim safeHandle = safeRegistryHandleConstructor.Invoke(New Object() {hKey, ownsHandle})
' Create a new Registry key using the private constructor using the safeHandle - this should then behave like a .NET natively opened handle and disposed of correctly
Dim registryKeyType = GetType(RegistryKey)
Dim registryKeyConstructorTypes As System.Type() = {safeRegistryHandleType, GetType(Boolean)}
Dim registryKeyConstructor = registryKeyType.GetConstructor(privateConstructors, Nothing, registryKeyConstructorTypes, Nothing)
Dim result = DirectCast(registryKeyConstructor.Invoke(New Object() {safeHandle, writable}), RegistryKey)
Return result
End Function
Sample Code (C#) to read a string value
string example = ReadRegKey(HKEY_LOCAL_MACHINE, @"Software\SomeCompany\OurProduct", "InstalledVersion");
The code above didn't compile for me - this code compiles and works for my needs:
' Basis from http://blogs.msdn.com/cumgranosalis/archive/2005/12/09/Win64RegistryPart1.aspx
Private Function GetRegistryKeyHandle(ByVal RegisteryKey As Microsoft.Win32.RegistryKey) As System.IntPtr
Dim Type As System.Type = System.Type.GetType("Microsoft.Win32.RegistryKey")
Dim Info As System.Reflection.FieldInfo = Type.GetField("hkey", System.Reflection.BindingFlags.NonPublic Or System.Reflection.BindingFlags.Instance)
Dim Handle As System.Runtime.InteropServices.SafeHandle = CType(Info.GetValue(RegisteryKey), System.Runtime.InteropServices.SafeHandle)
Dim RealHandle As System.IntPtr = Handle.DangerousGetHandle
Return Handle.DangerousGetHandle
End Function
' Basis from: http://www.pinvoke.net/default.aspx/advapi32/RegOpenKeyEx.html
Public Enum RegWow64Options As System.Int32
None = 0
KEY_WOW64_64KEY = &H100
KEY_WOW64_32KEY = &H200
End Enum
Private Enum RegistryRights As System.Int32
ReadKey = 131097
WriteKey = 131078
End Enum
Private Declare Auto Function RegOpenKeyEx Lib "advapi32.dll" (ByVal hKey As System.IntPtr, ByVal lpSubKey As System.String, ByVal ulOptions As System.Int32, ByVal samDesired As System.Int32, ByRef phkResult As System.Int32) As System.Int32
Public Function OpenSubKey(ByVal ParentKey As Microsoft.Win32.RegistryKey, ByVal SubKeyName As String, ByVal Writeable As Boolean, ByVal Options As RegWow64Options) As Microsoft.Win32.RegistryKey
If ParentKey Is Nothing OrElse GetRegistryKeyHandle(ParentKey).Equals(System.IntPtr.Zero) Then
Throw New System.Exception("OpenSubKey: Parent key is not open")
End If
Dim Rights As System.Int32 = RegistryRights.ReadKey
If Writeable Then
Rights = RegistryRights.WriteKey
End If
Dim SubKeyHandle As System.Int32
Dim Result As System.Int32 = RegOpenKeyEx(GetRegistryKeyHandle(ParentKey), SubKeyName, 0, Rights Or Options, SubKeyHandle)
If Result <> 0 Then
Dim W32ex As New System.ComponentModel.Win32Exception()
Throw New System.Exception("OpenSubKey: Exception encountered opening key", W32ex)
End If
Return PointerToRegistryKey(CType(SubKeyHandle, System.IntPtr), Writeable, False)
End Function
Private Function PointerToRegistryKey(ByVal hKey As System.IntPtr, ByVal writable As Boolean, ByVal ownsHandle As Boolean) As Microsoft.Win32.RegistryKey
' Create a SafeHandles.SafeRegistryHandle from this pointer - this is a private class
Dim privateConstructors As System.Reflection.BindingFlags = System.Reflection.BindingFlags.Instance Or System.Reflection.BindingFlags.NonPublic
Dim safeRegistryHandleType As System.Type = GetType(Microsoft.Win32.SafeHandles.SafeHandleZeroOrMinusOneIsInvalid).Assembly.GetType("Microsoft.Win32.SafeHandles.SafeRegistryHandle")
Dim safeRegistryHandleConstructorTypes As System.Type() = {GetType(System.IntPtr), GetType(System.Boolean)}
Dim safeRegistryHandleConstructor As System.Reflection.ConstructorInfo = safeRegistryHandleType.GetConstructor(privateConstructors, Nothing, safeRegistryHandleConstructorTypes, Nothing)
Dim safeHandle As System.Object = safeRegistryHandleConstructor.Invoke(New System.Object() {hKey, ownsHandle})
' Create a new Registry key using the private constructor using the safeHandle - this should then behave like a .NET natively opened handle and disposed of correctly
Dim registryKeyType As System.Type = GetType(Microsoft.Win32.RegistryKey)
Dim registryKeyConstructorTypes As System.Type() = {safeRegistryHandleType, GetType(System.Boolean)}
Dim registryKeyConstructor As System.Reflection.ConstructorInfo = registryKeyType.GetConstructor(privateConstructors, Nothing, registryKeyConstructorTypes, Nothing)
Dim result As Microsoft.Win32.RegistryKey = DirectCast(registryKeyConstructor.Invoke(New Object() {safeHandle, writable}), Microsoft.Win32.RegistryKey)
Return result
End Function
Sample usage:
Dim Config As Microsoft.Win32.RegistryKey = Nothing
If Is64Bit() Then
Config = OpenSubKey(Microsoft.Win32.Registry.LocalMachine, "SOFTWARE\" & BaseName, False, RegWow64Options.KEY_WOW64_64KEY)
Else
Config = Microsoft.Win32.Registry.LocalMachine.OpenSubKey("SOFTWARE\" & BaseName, False)
End If
An IntPtr is a pointer to a memory location (unmanaged) that adapts to the platform it is running on (64-bit, etc.) UNLIKE a standard int/Integer. You should always use this type for unmanaged calls that require it, even though an int will appear to work on your development machine.
1/13/2008 4:00:13 AM - Damon Carr-72.43.165.29
Click to read this page
6/25/2010 2:17:25 PM - -90.152.60.34
An IntPtr is a pointer to a memory location (unmanaged) that adapts to the platform it is running on (64-bit, etc.) UNLIKE a standard int/Integer. You should always use this type for unmanaged calls that require it, even though an int will appear to work on your development machine.
1/13/2008 4:00:13 AM - Damon Carr-72.43.165.29
Click to read this page
6/25/2010 2:17:25 PM - -90.152.60.34
TODO - a short description
3/16/2007 8:11:30 AM - -61.11.98.124
The FormatMessage API
12/27/2020 2:24:15 AM - -84.110.53.106
Opens the specified registry key
6/12/2012 9:11:53 AM - born2c0de@dreamincode.net-125.99.58.184
An IntPtr is a pointer to a memory location (unmanaged) that adapts to the platform it is running on (64-bit, etc.) UNLIKE a standard int/Integer. You should always use this type for unmanaged calls that require it, even though an int will appear to work on your development machine.
1/13/2008 4:00:13 AM - Damon Carr-72.43.165.29
Please edit this page!
Do you have...
helpful tips or sample code to share for using this API in managed code?
corrections to the existing content?
variations of the signature you want to share?
additional languages you want to include?
Select "Edit This Page" on the right hand toolbar and edit it! Or add new pages containing supporting types needed for this API (structures, delegates, and more).