|  | /* | 
|  | * This work is licensed under the terms of the GNU GPL, version 2 or | 
|  | * (at your option) any later version.  See the COPYING file in the | 
|  | * top-level directory. | 
|  | * | 
|  | * The win32 keyboard hooking code was imported from project spice-gtk. | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  | #include "sysemu/sysemu.h" | 
|  | #include "ui/win32-kbd-hook.h" | 
|  |  | 
|  | static Notifier win32_unhook_notifier; | 
|  | static HHOOK win32_keyboard_hook; | 
|  | static HWND win32_window; | 
|  | static DWORD win32_grab; | 
|  |  | 
|  | static LRESULT CALLBACK keyboard_hook_cb(int code, WPARAM wparam, LPARAM lparam) | 
|  | { | 
|  | if  (win32_window && code == HC_ACTION && win32_window == GetFocus()) { | 
|  | KBDLLHOOKSTRUCT *hooked = (KBDLLHOOKSTRUCT *)lparam; | 
|  |  | 
|  | if (wparam != WM_KEYUP) { | 
|  | DWORD dwmsg = (hooked->flags << 24) | | 
|  | ((hooked->scanCode & 0xff) << 16) | 1; | 
|  |  | 
|  | switch (hooked->vkCode) { | 
|  | case VK_CAPITAL: | 
|  | /* fall through */ | 
|  | case VK_SCROLL: | 
|  | /* fall through */ | 
|  | case VK_NUMLOCK: | 
|  | /* fall through */ | 
|  | case VK_LSHIFT: | 
|  | /* fall through */ | 
|  | case VK_RSHIFT: | 
|  | /* fall through */ | 
|  | case VK_RCONTROL: | 
|  | /* fall through */ | 
|  | case VK_LMENU: | 
|  | /* fall through */ | 
|  | case VK_RMENU: | 
|  | break; | 
|  |  | 
|  | case VK_LCONTROL: | 
|  | /* | 
|  | * When pressing AltGr, an extra VK_LCONTROL with a special | 
|  | * scancode with bit 9 set is sent. Let's ignore the extra | 
|  | * VK_LCONTROL, as that will make AltGr misbehave. | 
|  | */ | 
|  | if (hooked->scanCode & 0x200) { | 
|  | return 1; | 
|  | } | 
|  | break; | 
|  |  | 
|  | default: | 
|  | if (win32_grab) { | 
|  | SendMessage(win32_window, wparam, hooked->vkCode, dwmsg); | 
|  | return 1; | 
|  | } | 
|  | break; | 
|  | } | 
|  |  | 
|  | } else { | 
|  | switch (hooked->vkCode) { | 
|  | case VK_LCONTROL: | 
|  | if (hooked->scanCode & 0x200) { | 
|  | return 1; | 
|  | } | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return CallNextHookEx(NULL, code, wparam, lparam); | 
|  | } | 
|  |  | 
|  | static void keyboard_hook_unhook(Notifier *n, void *data) | 
|  | { | 
|  | UnhookWindowsHookEx(win32_keyboard_hook); | 
|  | win32_keyboard_hook = NULL; | 
|  | } | 
|  |  | 
|  | void win32_kbd_set_window(void *hwnd) | 
|  | { | 
|  | if (hwnd && !win32_keyboard_hook) { | 
|  | /* note: the installing thread must have a message loop */ | 
|  | win32_keyboard_hook = SetWindowsHookEx(WH_KEYBOARD_LL, keyboard_hook_cb, | 
|  | GetModuleHandle(NULL), 0); | 
|  | if (win32_keyboard_hook) { | 
|  | win32_unhook_notifier.notify = keyboard_hook_unhook; | 
|  | qemu_add_exit_notifier(&win32_unhook_notifier); | 
|  | } | 
|  | } | 
|  |  | 
|  | win32_window = hwnd; | 
|  | } | 
|  |  | 
|  | void win32_kbd_set_grab(bool grab) | 
|  | { | 
|  | win32_grab = grab; | 
|  | } |