I'm attempting to get a KBDLLHOOKSTRUCT from a keyboard-hook's lParam.
private static IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
{
KBDLLHOOKSTRUCT kbd = new KBDLLHOOKSTRUCT();
Marshal.PtrToStructure(lParam, kbd); // Throws System.ArguementException
...
Unfortunately the PtrToStructure is throwing two
A first chance exception of type 'System.ArgumentException' occurred in myprogram.exe
errors every time a key is pressed. It also stops that method in its tracks.
MSNDA says: http://msdn.microsoft.com/en-us/library/4ca6d5z7.aspx
ArgumentException when:
The structureType parameter layout is not sequential or explicit.
-or-
The structureType parameter is a generic type.
What can I do here to get it working? lParam is coming straight from the keyboard hook so I would expect it to be correct. Do either of those errors make sense here, and what can I do to fix it?
Here is some code that works for me:
The crucial difference from your code is that I am calling the Marshal.PtrToStructure(IntPtr, Type) overload rather than the (IntPtr, object) overload. And I think this is where things have gone awry for you. Because if you call the (IntPtr, object) overload with a struct, you get the following error:
The obvious fix for this error is to change KBDLLHOOKSTRUCT to be a class (reference type) instead of a struct (value type):
However, this results in the error that MSDN means by "The structureType parameter layout is not sequential or explicit.":
I am guessing this is where you are right now, with KBDLLHOOKSTRUCT declared as a class, and getting the "no layout information" error. There are two ways to address this.
First, as per Eric Law's comment, you can keep your Marshal.PtrToStructure call as it is, keep KBDLLHOOKSTRUCT as a class, and add layout information to KBDLLHOOKSTRUCT:
Second, as per my sample code above, you can change KBDLLHOOKSTRUCT to a
struct
instead of aclass
, and change your Marshal.PtrToStructure call to the (IntPtr, Type) overload:(In this case you can still add the
[StructLayout(LayoutKind.Sequential)]
attribute to the KBDLLHOOKSTRUCT struct if you like. It is technically redundant but may help readers of your code recognise KBDLLHOOKSTRUCT as a layout-sensitive interop type.)Both of these solutions work for me in an (admittedly simple) test. Of the two, I would recommend the second, because Win32 / C structures are conventionally declared as
struct
in P/Invoke scenarios -- and if nothing else a name that ends in STRUCT should probably be a struct rather than a class!Finally, let me mention an alternative approach.
Instead of declaring LowLevelKeyboardProc as receiving an IntPtr as its lParam, it is possible to declare it as receiving a
ref KBDLLHOOKSTRUCT
(where KBDLLHOOKSTRUCT is astruct
, not aclass
). This also requires changes to CallNextHookEx, but the net result is to simplify the use of the KBDLLHOOKSTRUCT info by avoiding the Marshal call altogether. The use of a ref parameter also means you can write to the struct (which I know from other questions is your goal) and not need to marshal it back after writing:(I should probably warn you, though, that when I tried modifying kbd.vkCode, it didn't actually affect what appeared in text boxes etc. I don't know enough about low-level keyboard hooks to know why not or what I would need to do to make this work; sorry.)