|  | /* | 
|  | * Windows hook functions | 
|  | * | 
|  | * Copyright 2002 Alexandre Julliard | 
|  | * Copyright 2005 Dmitry Timoshkov | 
|  | * | 
|  | * This library is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU Lesser General Public | 
|  | * License as published by the Free Software Foundation; either | 
|  | * version 2.1 of the License, or (at your option) any later version. | 
|  | * | 
|  | * This library is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | * Lesser General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU Lesser General Public | 
|  | * License along with this library; if not, write to the Free Software | 
|  | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
|  | * | 
|  | * NOTES: | 
|  | *   Status of the various hooks: | 
|  | *     WH_MSGFILTER                 OK | 
|  | *     WH_JOURNALRECORD             Partially implemented | 
|  | *     WH_JOURNALPLAYBACK           Partially implemented | 
|  | *     WH_KEYBOARD                  OK | 
|  | *     WH_GETMESSAGE	            OK (FIXME: A/W mapping?) | 
|  | *     WH_CALLWNDPROC	            OK (FIXME: A/W mapping?) | 
|  | *     WH_CBT | 
|  | *       HCBT_MOVESIZE              OK | 
|  | *       HCBT_MINMAX                OK | 
|  | *       HCBT_QS                    OK | 
|  | *       HCBT_CREATEWND             OK | 
|  | *       HCBT_DESTROYWND            OK | 
|  | *       HCBT_ACTIVATE              OK | 
|  | *       HCBT_CLICKSKIPPED          OK | 
|  | *       HCBT_KEYSKIPPED            OK | 
|  | *       HCBT_SYSCOMMAND            OK | 
|  | *       HCBT_SETFOCUS              OK | 
|  | *     WH_SYSMSGFILTER	            OK | 
|  | *     WH_MOUSE	                    OK | 
|  | *     WH_HARDWARE	            Not supported in Win32 | 
|  | *     WH_DEBUG	                    Not implemented | 
|  | *     WH_SHELL | 
|  | *       HSHELL_WINDOWCREATED       OK | 
|  | *       HSHELL_WINDOWDESTROYED     OK | 
|  | *       HSHELL_ACTIVATESHELLWINDOW Not implemented | 
|  | *       HSHELL_WINDOWACTIVATED     Not implemented | 
|  | *       HSHELL_GETMINRECT          Not implemented | 
|  | *       HSHELL_REDRAW              Not implemented | 
|  | *       HSHELL_TASKMAN             Not implemented | 
|  | *       HSHELL_LANGUAGE            Not implemented | 
|  | *       HSHELL_SYSMENU             Not implemented | 
|  | *       HSHELL_ENDTASK             Not implemented | 
|  | *       HSHELL_ACCESSIBILITYSTATE  Not implemented | 
|  | *       HSHELL_APPCOMMAND          Not implemented | 
|  | *       HSHELL_WINDOWREPLACED      Not implemented | 
|  | *       HSHELL_WINDOWREPLACING     Not implemented | 
|  | *     WH_FOREGROUNDIDLE            Not implemented | 
|  | *     WH_CALLWNDPROCRET            OK (FIXME: A/W mapping?) | 
|  | *     WH_KEYBOARD_LL               Implemented but should use SendMessage instead | 
|  | *     WH_MOUSE_LL                  Implemented but should use SendMessage instead | 
|  | */ | 
|  |  | 
|  | #include "config.h" | 
|  | #include "wine/port.h" | 
|  |  | 
|  | #include <stdarg.h> | 
|  | #include <assert.h> | 
|  |  | 
|  | #include "windef.h" | 
|  | #include "winbase.h" | 
|  | #include "winuser.h" | 
|  | #include "winerror.h" | 
|  | #include "win.h" | 
|  | #include "user_private.h" | 
|  | #include "wine/server.h" | 
|  | #include "wine/unicode.h" | 
|  | #include "wine/debug.h" | 
|  | #include "winternl.h" | 
|  |  | 
|  | WINE_DEFAULT_DEBUG_CHANNEL(hook); | 
|  | WINE_DECLARE_DEBUG_CHANNEL(relay); | 
|  |  | 
|  | struct hook_info | 
|  | { | 
|  | INT id; | 
|  | FARPROC proc; | 
|  | void *handle; | 
|  | DWORD pid, tid; | 
|  | BOOL prev_unicode, next_unicode; | 
|  | WCHAR module[MAX_PATH]; | 
|  | }; | 
|  |  | 
|  | #define WH_WINEVENT (WH_MAXHOOK+1) | 
|  |  | 
|  | static const char * const hook_names[WH_WINEVENT - WH_MINHOOK + 1] = | 
|  | { | 
|  | "WH_MSGFILTER", | 
|  | "WH_JOURNALRECORD", | 
|  | "WH_JOURNALPLAYBACK", | 
|  | "WH_KEYBOARD", | 
|  | "WH_GETMESSAGE", | 
|  | "WH_CALLWNDPROC", | 
|  | "WH_CBT", | 
|  | "WH_SYSMSGFILTER", | 
|  | "WH_MOUSE", | 
|  | "WH_HARDWARE", | 
|  | "WH_DEBUG", | 
|  | "WH_SHELL", | 
|  | "WH_FOREGROUNDIDLE", | 
|  | "WH_CALLWNDPROCRET", | 
|  | "WH_KEYBOARD_LL", | 
|  | "WH_MOUSE_LL", | 
|  | "WH_WINEVENT" | 
|  | }; | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		get_ll_hook_timeout | 
|  | * | 
|  | */ | 
|  | static UINT get_ll_hook_timeout(void) | 
|  | { | 
|  | /* FIXME: should retrieve LowLevelHooksTimeout in HKEY_CURRENT_USER\Control Panel\Desktop */ | 
|  | return 2000; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		set_windows_hook | 
|  | * | 
|  | * Implementation of SetWindowsHookExA and SetWindowsHookExW. | 
|  | */ | 
|  | static HHOOK set_windows_hook( INT id, HOOKPROC proc, HINSTANCE inst, DWORD tid, BOOL unicode ) | 
|  | { | 
|  | HHOOK handle = 0; | 
|  | WCHAR module[MAX_PATH]; | 
|  | DWORD len; | 
|  |  | 
|  | if (!proc) | 
|  | { | 
|  | SetLastError( ERROR_INVALID_FILTER_PROC ); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* FIXME: what if the tid belongs to another process? */ | 
|  | if (tid)  /* thread-local hook */ | 
|  | { | 
|  | if (id == WH_JOURNALRECORD || | 
|  | id == WH_JOURNALPLAYBACK || | 
|  | id == WH_KEYBOARD_LL || | 
|  | id == WH_MOUSE_LL || | 
|  | id == WH_SYSMSGFILTER) | 
|  | { | 
|  | /* these can only be global */ | 
|  | SetLastError( ERROR_INVALID_PARAMETER ); | 
|  | return 0; | 
|  | } | 
|  | inst = 0; | 
|  | } | 
|  | else  /* system-global hook */ | 
|  | { | 
|  | if (id == WH_KEYBOARD_LL || id == WH_MOUSE_LL) inst = 0; | 
|  | else if (!inst || !(len = GetModuleFileNameW( inst, module, MAX_PATH )) || len >= MAX_PATH) | 
|  | { | 
|  | SetLastError( ERROR_HOOK_NEEDS_HMOD ); | 
|  | return 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | SERVER_START_REQ( set_hook ) | 
|  | { | 
|  | req->id        = id; | 
|  | req->pid       = 0; | 
|  | req->tid       = tid; | 
|  | req->event_min = EVENT_MIN; | 
|  | req->event_max = EVENT_MAX; | 
|  | req->flags     = WINEVENT_INCONTEXT; | 
|  | req->unicode   = unicode; | 
|  | if (inst) /* make proc relative to the module base */ | 
|  | { | 
|  | req->proc = (void *)((char *)proc - (char *)inst); | 
|  | wine_server_add_data( req, module, strlenW(module) * sizeof(WCHAR) ); | 
|  | } | 
|  | else req->proc = proc; | 
|  |  | 
|  | if (!wine_server_call_err( req )) | 
|  | { | 
|  | handle = reply->handle; | 
|  | get_user_thread_info()->active_hooks = reply->active_hooks; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  |  | 
|  | TRACE( "%s %p %x -> %p\n", hook_names[id-WH_MINHOOK], proc, tid, handle ); | 
|  | return handle; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		call_hook_AtoW | 
|  | */ | 
|  | static LRESULT call_hook_AtoW( HOOKPROC proc, INT id, INT code, WPARAM wparam, LPARAM lparam ) | 
|  | { | 
|  | LRESULT ret; | 
|  | UNICODE_STRING usBuffer; | 
|  | if (id != WH_CBT || code != HCBT_CREATEWND) ret = proc( code, wparam, lparam ); | 
|  | else | 
|  | { | 
|  | CBT_CREATEWNDA *cbtcwA = (CBT_CREATEWNDA *)lparam; | 
|  | CBT_CREATEWNDW cbtcwW; | 
|  | CREATESTRUCTW csW; | 
|  | LPWSTR nameW = NULL; | 
|  | LPWSTR classW = NULL; | 
|  |  | 
|  | cbtcwW.lpcs = &csW; | 
|  | cbtcwW.hwndInsertAfter = cbtcwA->hwndInsertAfter; | 
|  | csW = *(CREATESTRUCTW *)cbtcwA->lpcs; | 
|  |  | 
|  | if (HIWORD(cbtcwA->lpcs->lpszName)) | 
|  | { | 
|  | RtlCreateUnicodeStringFromAsciiz(&usBuffer,cbtcwA->lpcs->lpszName); | 
|  | csW.lpszName = nameW = usBuffer.Buffer; | 
|  | } | 
|  | if (HIWORD(cbtcwA->lpcs->lpszClass)) | 
|  | { | 
|  | RtlCreateUnicodeStringFromAsciiz(&usBuffer,cbtcwA->lpcs->lpszClass); | 
|  | csW.lpszClass = classW = usBuffer.Buffer; | 
|  | } | 
|  | ret = proc( code, wparam, (LPARAM)&cbtcwW ); | 
|  | cbtcwA->hwndInsertAfter = cbtcwW.hwndInsertAfter; | 
|  | HeapFree( GetProcessHeap(), 0, nameW ); | 
|  | HeapFree( GetProcessHeap(), 0, classW ); | 
|  | } | 
|  | return ret; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		call_hook_WtoA | 
|  | */ | 
|  | static LRESULT call_hook_WtoA( HOOKPROC proc, INT id, INT code, WPARAM wparam, LPARAM lparam ) | 
|  | { | 
|  | LRESULT ret; | 
|  |  | 
|  | if (id != WH_CBT || code != HCBT_CREATEWND) ret = proc( code, wparam, lparam ); | 
|  | else | 
|  | { | 
|  | CBT_CREATEWNDW *cbtcwW = (CBT_CREATEWNDW *)lparam; | 
|  | CBT_CREATEWNDA cbtcwA; | 
|  | CREATESTRUCTA csA; | 
|  | int len; | 
|  | LPSTR nameA = NULL; | 
|  | LPSTR classA = NULL; | 
|  |  | 
|  | cbtcwA.lpcs = &csA; | 
|  | cbtcwA.hwndInsertAfter = cbtcwW->hwndInsertAfter; | 
|  | csA = *(CREATESTRUCTA *)cbtcwW->lpcs; | 
|  |  | 
|  | if (HIWORD(cbtcwW->lpcs->lpszName)) { | 
|  | len = WideCharToMultiByte( CP_ACP, 0, cbtcwW->lpcs->lpszName, -1, NULL, 0, NULL, NULL ); | 
|  | nameA = HeapAlloc( GetProcessHeap(), 0, len*sizeof(CHAR) ); | 
|  | WideCharToMultiByte( CP_ACP, 0, cbtcwW->lpcs->lpszName, -1, nameA, len, NULL, NULL ); | 
|  | csA.lpszName = nameA; | 
|  | } | 
|  |  | 
|  | if (HIWORD(cbtcwW->lpcs->lpszClass)) { | 
|  | len = WideCharToMultiByte( CP_ACP, 0, cbtcwW->lpcs->lpszClass, -1, NULL, 0, NULL, NULL ); | 
|  | classA = HeapAlloc( GetProcessHeap(), 0, len*sizeof(CHAR) ); | 
|  | WideCharToMultiByte( CP_ACP, 0, cbtcwW->lpcs->lpszClass, -1, classA, len, NULL, NULL ); | 
|  | csA.lpszClass = classA; | 
|  | } | 
|  |  | 
|  | ret = proc( code, wparam, (LPARAM)&cbtcwA ); | 
|  | cbtcwW->hwndInsertAfter = cbtcwA.hwndInsertAfter; | 
|  | HeapFree( GetProcessHeap(), 0, nameA ); | 
|  | HeapFree( GetProcessHeap(), 0, classA ); | 
|  | } | 
|  | return ret; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		call_hook_proc | 
|  | */ | 
|  | static LRESULT call_hook_proc( HOOKPROC proc, INT id, INT code, WPARAM wparam, LPARAM lparam, | 
|  | BOOL prev_unicode, BOOL next_unicode ) | 
|  | { | 
|  | LRESULT ret; | 
|  |  | 
|  | if (TRACE_ON(relay)) | 
|  | DPRINTF( "%04x:Call hook proc %p (id=%s,code=%x,wp=%08x,lp=%08lx)\n", | 
|  | GetCurrentThreadId(), proc, hook_names[id-WH_MINHOOK], code, wparam, lparam ); | 
|  |  | 
|  | if (!prev_unicode == !next_unicode) ret = proc( code, wparam, lparam ); | 
|  | else if (prev_unicode) ret = call_hook_WtoA( proc, id, code, wparam, lparam ); | 
|  | else ret = call_hook_AtoW( proc, id, code, wparam, lparam ); | 
|  |  | 
|  | if (TRACE_ON(relay)) | 
|  | DPRINTF( "%04x:Ret  hook proc %p (id=%s,code=%x,wp=%08x,lp=%08lx) retval=%08lx\n", | 
|  | GetCurrentThreadId(), proc, hook_names[id-WH_MINHOOK], code, wparam, lparam, ret ); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		get_hook_proc | 
|  | * | 
|  | * Retrieve the hook procedure real value for a module-relative proc | 
|  | */ | 
|  | static void *get_hook_proc( void *proc, const WCHAR *module ) | 
|  | { | 
|  | HMODULE mod; | 
|  |  | 
|  | if (!(mod = GetModuleHandleW(module))) | 
|  | { | 
|  | TRACE( "loading %s\n", debugstr_w(module) ); | 
|  | /* FIXME: the library will never be freed */ | 
|  | if (!(mod = LoadLibraryW(module))) return NULL; | 
|  | } | 
|  | return (char *)mod + (ULONG_PTR)proc; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		call_hook | 
|  | * | 
|  | * Call hook either in current thread or send message to the destination | 
|  | * thread. | 
|  | */ | 
|  | static LRESULT call_hook( struct hook_info *info, INT code, WPARAM wparam, LPARAM lparam ) | 
|  | { | 
|  | DWORD_PTR ret = 0; | 
|  |  | 
|  | if (info->tid) | 
|  | { | 
|  | struct hook_extra_info h_extra; | 
|  | h_extra.handle = info->handle; | 
|  | h_extra.lparam = lparam; | 
|  |  | 
|  | TRACE( "calling hook in thread %04x %s code %x wp %x lp %lx\n", | 
|  | info->tid, hook_names[info->id-WH_MINHOOK], code, wparam, lparam ); | 
|  |  | 
|  | switch(info->id) | 
|  | { | 
|  | case WH_KEYBOARD_LL: | 
|  | MSG_SendInternalMessageTimeout( info->pid, info->tid, WM_WINE_KEYBOARD_LL_HOOK, | 
|  | wparam, (LPARAM)&h_extra, SMTO_ABORTIFHUNG, | 
|  | get_ll_hook_timeout(), &ret ); | 
|  | break; | 
|  | case WH_MOUSE_LL: | 
|  | MSG_SendInternalMessageTimeout( info->pid, info->tid, WM_WINE_MOUSE_LL_HOOK, | 
|  | wparam, (LPARAM)&h_extra, SMTO_ABORTIFHUNG, | 
|  | get_ll_hook_timeout(), &ret ); | 
|  | break; | 
|  | default: | 
|  | ERR("Unknown hook id %d\n", info->id); | 
|  | assert(0); | 
|  | break; | 
|  | } | 
|  | } | 
|  | else if (info->proc) | 
|  | { | 
|  | TRACE( "calling hook %p %s code %x wp %x lp %lx module %s\n", | 
|  | info->proc, hook_names[info->id-WH_MINHOOK], code, wparam, | 
|  | lparam, debugstr_w(info->module) ); | 
|  |  | 
|  | if (!info->module[0] || | 
|  | (info->proc = get_hook_proc( info->proc, info->module )) != NULL) | 
|  | { | 
|  | struct user_thread_info *thread_info = get_user_thread_info(); | 
|  | HHOOK prev = thread_info->hook; | 
|  | BOOL prev_unicode = thread_info->hook_unicode; | 
|  |  | 
|  | thread_info->hook = info->handle; | 
|  | thread_info->hook_unicode = info->next_unicode; | 
|  | ret = call_hook_proc( (HOOKPROC)info->proc, info->id, code, wparam, lparam, | 
|  | info->prev_unicode, info->next_unicode ); | 
|  | thread_info->hook = prev; | 
|  | thread_info->hook_unicode = prev_unicode; | 
|  | } | 
|  | } | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		HOOK_CallHooks | 
|  | */ | 
|  | LRESULT HOOK_CallHooks( INT id, INT code, WPARAM wparam, LPARAM lparam, BOOL unicode ) | 
|  | { | 
|  | struct user_thread_info *thread_info = get_user_thread_info(); | 
|  | struct hook_info info; | 
|  | DWORD_PTR ret = 0; | 
|  |  | 
|  | USER_CheckNotLock(); | 
|  |  | 
|  | if (!HOOK_IsHooked( id )) | 
|  | { | 
|  | TRACE( "skipping hook %s mask %x\n", hook_names[id-WH_MINHOOK], thread_info->active_hooks ); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | ZeroMemory( &info, sizeof(info) - sizeof(info.module) ); | 
|  | info.prev_unicode = unicode; | 
|  | info.id = id; | 
|  |  | 
|  | SERVER_START_REQ( start_hook_chain ) | 
|  | { | 
|  | req->id = info.id; | 
|  | req->event = EVENT_MIN; | 
|  | wine_server_set_reply( req, info.module, sizeof(info.module)-sizeof(WCHAR) ); | 
|  | if (!wine_server_call( req )) | 
|  | { | 
|  | info.module[wine_server_reply_size(req) / sizeof(WCHAR)] = 0; | 
|  | info.handle       = reply->handle; | 
|  | info.pid          = reply->pid; | 
|  | info.tid          = reply->tid; | 
|  | info.proc         = reply->proc; | 
|  | info.next_unicode = reply->unicode; | 
|  | thread_info->active_hooks = reply->active_hooks; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  |  | 
|  | if (!info.tid && !info.proc) return 0; | 
|  | ret = call_hook( &info, code, wparam, lparam ); | 
|  |  | 
|  | SERVER_START_REQ( finish_hook_chain ) | 
|  | { | 
|  | req->id = id; | 
|  | wine_server_call( req ); | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | return ret; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           HOOK_IsHooked | 
|  | */ | 
|  | BOOL HOOK_IsHooked( INT id ) | 
|  | { | 
|  | struct user_thread_info *thread_info = get_user_thread_info(); | 
|  |  | 
|  | if (!thread_info->active_hooks) return TRUE; | 
|  | return (thread_info->active_hooks & (1 << (id - WH_MINHOOK))) != 0; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		SetWindowsHookA (USER32.@) | 
|  | */ | 
|  | HHOOK WINAPI SetWindowsHookA( INT id, HOOKPROC proc ) | 
|  | { | 
|  | return SetWindowsHookExA( id, proc, 0, GetCurrentThreadId() ); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		SetWindowsHookW (USER32.@) | 
|  | */ | 
|  | HHOOK WINAPI SetWindowsHookW( INT id, HOOKPROC proc ) | 
|  | { | 
|  | return SetWindowsHookExW( id, proc, 0, GetCurrentThreadId() ); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		SetWindowsHookExA (USER32.@) | 
|  | */ | 
|  | HHOOK WINAPI SetWindowsHookExA( INT id, HOOKPROC proc, HINSTANCE inst, DWORD tid ) | 
|  | { | 
|  | return set_windows_hook( id, proc, inst, tid, FALSE ); | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		SetWindowsHookExW (USER32.@) | 
|  | */ | 
|  | HHOOK WINAPI SetWindowsHookExW( INT id, HOOKPROC proc, HINSTANCE inst, DWORD tid ) | 
|  | { | 
|  | return set_windows_hook( id, proc, inst, tid, TRUE ); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		UnhookWindowsHook (USER32.@) | 
|  | */ | 
|  | BOOL WINAPI UnhookWindowsHook( INT id, HOOKPROC proc ) | 
|  | { | 
|  | BOOL ret; | 
|  |  | 
|  | TRACE( "%s %p\n", hook_names[id-WH_MINHOOK], proc ); | 
|  |  | 
|  | SERVER_START_REQ( remove_hook ) | 
|  | { | 
|  | req->handle = 0; | 
|  | req->id   = id; | 
|  | req->proc = proc; | 
|  | ret = !wine_server_call_err( req ); | 
|  | if (ret) get_user_thread_info()->active_hooks = reply->active_hooks; | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | if (!ret && GetLastError() == ERROR_INVALID_HANDLE) SetLastError( ERROR_INVALID_HOOK_HANDLE ); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		UnhookWindowsHookEx (USER32.@) | 
|  | */ | 
|  | BOOL WINAPI UnhookWindowsHookEx( HHOOK hhook ) | 
|  | { | 
|  | BOOL ret; | 
|  |  | 
|  | TRACE( "%p\n", hhook ); | 
|  |  | 
|  | SERVER_START_REQ( remove_hook ) | 
|  | { | 
|  | req->handle = hhook; | 
|  | req->id     = 0; | 
|  | ret = !wine_server_call_err( req ); | 
|  | if (ret) get_user_thread_info()->active_hooks = reply->active_hooks; | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | if (!ret && GetLastError() == ERROR_INVALID_HANDLE) SetLastError( ERROR_INVALID_HOOK_HANDLE ); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		CallNextHookEx (USER32.@) | 
|  | */ | 
|  | LRESULT WINAPI CallNextHookEx( HHOOK hhook, INT code, WPARAM wparam, LPARAM lparam ) | 
|  | { | 
|  | struct user_thread_info *thread_info = get_user_thread_info(); | 
|  | struct hook_info info; | 
|  |  | 
|  | ZeroMemory( &info, sizeof(info) - sizeof(info.module) ); | 
|  |  | 
|  | SERVER_START_REQ( get_hook_info ) | 
|  | { | 
|  | req->handle = thread_info->hook; | 
|  | req->get_next = 1; | 
|  | req->event = EVENT_MIN; | 
|  | wine_server_set_reply( req, info.module, sizeof(info.module)-sizeof(WCHAR) ); | 
|  | if (!wine_server_call_err( req )) | 
|  | { | 
|  | info.module[wine_server_reply_size(req) / sizeof(WCHAR)] = 0; | 
|  | info.handle       = reply->handle; | 
|  | info.id           = reply->id; | 
|  | info.pid          = reply->pid; | 
|  | info.tid          = reply->tid; | 
|  | info.proc         = reply->proc; | 
|  | info.next_unicode = reply->unicode; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  |  | 
|  | info.prev_unicode = thread_info->hook_unicode; | 
|  | return call_hook( &info, code, wparam, lparam ); | 
|  | } | 
|  |  | 
|  |  | 
|  | LRESULT call_current_hook( HHOOK hhook, INT code, WPARAM wparam, LPARAM lparam ) | 
|  | { | 
|  | struct hook_info info; | 
|  |  | 
|  | ZeroMemory( &info, sizeof(info) - sizeof(info.module) ); | 
|  |  | 
|  | SERVER_START_REQ( get_hook_info ) | 
|  | { | 
|  | req->handle = hhook; | 
|  | req->get_next = 0; | 
|  | req->event = EVENT_MIN; | 
|  | wine_server_set_reply( req, info.module, sizeof(info.module)-sizeof(WCHAR) ); | 
|  | if (!wine_server_call_err( req )) | 
|  | { | 
|  | info.module[wine_server_reply_size(req) / sizeof(WCHAR)] = 0; | 
|  | info.handle       = reply->handle; | 
|  | info.id           = reply->id; | 
|  | info.pid          = reply->pid; | 
|  | info.tid          = reply->tid; | 
|  | info.proc         = reply->proc; | 
|  | info.next_unicode = reply->unicode; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  |  | 
|  | info.prev_unicode = TRUE;  /* assume Unicode for this function */ | 
|  | return call_hook( &info, code, wparam, lparam ); | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		CallMsgFilterA (USER32.@) | 
|  | */ | 
|  | BOOL WINAPI CallMsgFilterA( LPMSG msg, INT code ) | 
|  | { | 
|  | if (HOOK_CallHooks( WH_SYSMSGFILTER, code, 0, (LPARAM)msg, FALSE )) return TRUE; | 
|  | return HOOK_CallHooks( WH_MSGFILTER, code, 0, (LPARAM)msg, FALSE ); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *		CallMsgFilterW (USER32.@) | 
|  | */ | 
|  | BOOL WINAPI CallMsgFilterW( LPMSG msg, INT code ) | 
|  | { | 
|  | if (HOOK_CallHooks( WH_SYSMSGFILTER, code, 0, (LPARAM)msg, TRUE )) return TRUE; | 
|  | return HOOK_CallHooks( WH_MSGFILTER, code, 0, (LPARAM)msg, TRUE ); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           SetWinEventHook                            [USER32.@] | 
|  | * | 
|  | * Set up an event hook for a set of events. | 
|  | * | 
|  | * PARAMS | 
|  | *  event_min [I] Lowest event handled by pfnProc | 
|  | *  event_max [I] Highest event handled by pfnProc | 
|  | *  inst      [I] DLL containing pfnProc | 
|  | *  proc      [I] Callback event hook function | 
|  | *  pid       [I] Process to get events from, or 0 for all processes | 
|  | *  tid       [I] Thread to get events from, or 0 for all threads | 
|  | *  flags     [I] Flags indicating the status of pfnProc | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: A handle representing the hook. | 
|  | *  Failure: A NULL handle. | 
|  | */ | 
|  | HWINEVENTHOOK WINAPI SetWinEventHook(DWORD event_min, DWORD event_max, | 
|  | HMODULE inst, WINEVENTPROC proc, | 
|  | DWORD pid, DWORD tid, DWORD flags) | 
|  | { | 
|  | HWINEVENTHOOK handle = 0; | 
|  | WCHAR module[MAX_PATH]; | 
|  | DWORD len; | 
|  |  | 
|  | TRACE("%d,%d,%p,%p,%08x,%04x,%08x\n", event_min, event_max, inst, | 
|  | proc, pid, tid, flags); | 
|  |  | 
|  | if (inst) | 
|  | { | 
|  | if (!(len = GetModuleFileNameW(inst, module, MAX_PATH)) || len >= MAX_PATH) | 
|  | inst = 0; | 
|  | } | 
|  |  | 
|  | if ((flags & WINEVENT_INCONTEXT) && !inst) | 
|  | { | 
|  | SetLastError(ERROR_HOOK_NEEDS_HMOD); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (event_min > event_max) | 
|  | { | 
|  | SetLastError(ERROR_INVALID_HOOK_FILTER); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* FIXME: what if the tid or pid belongs to another process? */ | 
|  | if (tid)  /* thread-local hook */ | 
|  | inst = 0; | 
|  |  | 
|  | SERVER_START_REQ( set_hook ) | 
|  | { | 
|  | req->id        = WH_WINEVENT; | 
|  | req->pid       = pid; | 
|  | req->tid       = tid; | 
|  | req->event_min = event_min; | 
|  | req->event_max = event_max; | 
|  | req->flags     = flags; | 
|  | req->unicode   = 1; | 
|  | if (inst) /* make proc relative to the module base */ | 
|  | { | 
|  | req->proc = (void *)((char *)proc - (char *)inst); | 
|  | wine_server_add_data( req, module, strlenW(module) * sizeof(WCHAR) ); | 
|  | } | 
|  | else req->proc = proc; | 
|  |  | 
|  | if (!wine_server_call_err( req )) | 
|  | { | 
|  | handle = reply->handle; | 
|  | get_user_thread_info()->active_hooks = reply->active_hooks; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  |  | 
|  | TRACE("-> %p\n", handle); | 
|  | return handle; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           UnhookWinEvent                             [USER32.@] | 
|  | * | 
|  | * Remove an event hook for a set of events. | 
|  | * | 
|  | * PARAMS | 
|  | *  hEventHook [I] Event hook to remove | 
|  | * | 
|  | * RETURNS | 
|  | *  Success: TRUE. The event hook has been removed. | 
|  | *  Failure: FALSE, if hEventHook is invalid. | 
|  | */ | 
|  | BOOL WINAPI UnhookWinEvent(HWINEVENTHOOK hEventHook) | 
|  | { | 
|  | BOOL ret; | 
|  |  | 
|  | TRACE( "%p\n", hEventHook ); | 
|  |  | 
|  | SERVER_START_REQ( remove_hook ) | 
|  | { | 
|  | req->handle = hEventHook; | 
|  | req->id     = WH_WINEVENT; | 
|  | ret = !wine_server_call_err( req ); | 
|  | if (ret) get_user_thread_info()->active_hooks = reply->active_hooks; | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | inline static BOOL find_first_hook(DWORD id, DWORD event, HWND hwnd, LONG object_id, | 
|  | LONG child_id, struct hook_info *info) | 
|  | { | 
|  | struct user_thread_info *thread_info = get_user_thread_info(); | 
|  | BOOL ret; | 
|  |  | 
|  | if (!HOOK_IsHooked( id )) | 
|  | { | 
|  | TRACE( "skipping hook %s mask %x\n", hook_names[id-WH_MINHOOK], thread_info->active_hooks ); | 
|  | return FALSE; | 
|  | } | 
|  |  | 
|  | SERVER_START_REQ( start_hook_chain ) | 
|  | { | 
|  | req->id = id; | 
|  | req->event = event; | 
|  | req->window = hwnd; | 
|  | req->object_id = object_id; | 
|  | req->child_id = child_id; | 
|  | wine_server_set_reply( req, info->module, sizeof(info->module)-sizeof(WCHAR) ); | 
|  | ret = !wine_server_call( req ); | 
|  | if (ret) | 
|  | { | 
|  | info->module[wine_server_reply_size(req) / sizeof(WCHAR)] = 0; | 
|  | info->handle    = reply->handle; | 
|  | info->proc      = reply->proc; | 
|  | info->tid       = reply->tid; | 
|  | thread_info->active_hooks = reply->active_hooks; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | return ret && (info->tid || info->proc); | 
|  | } | 
|  |  | 
|  | inline static BOOL find_next_hook(DWORD event, HWND hwnd, LONG object_id, | 
|  | LONG child_id, struct hook_info *info) | 
|  | { | 
|  | BOOL ret; | 
|  |  | 
|  | SERVER_START_REQ( get_hook_info ) | 
|  | { | 
|  | req->handle = info->handle; | 
|  | req->get_next = 1; | 
|  | req->event = event; | 
|  | req->window = hwnd; | 
|  | req->object_id = object_id; | 
|  | req->child_id = child_id; | 
|  | wine_server_set_reply( req, info->module, sizeof(info->module)-sizeof(WCHAR) ); | 
|  | ret = !wine_server_call( req ); | 
|  | if (ret) | 
|  | { | 
|  | info->module[wine_server_reply_size(req) / sizeof(WCHAR)] = 0; | 
|  | info->handle    = reply->handle; | 
|  | info->proc      = reply->proc; | 
|  | info->tid       = reply->tid; | 
|  | } | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | inline static void find_hook_close(DWORD id) | 
|  | { | 
|  | SERVER_START_REQ( finish_hook_chain ) | 
|  | { | 
|  | req->id = id; | 
|  | wine_server_call( req ); | 
|  | } | 
|  | SERVER_END_REQ; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           NotifyWinEvent                             [USER32.@] | 
|  | * | 
|  | * Inform the OS that an event has occurred. | 
|  | * | 
|  | * PARAMS | 
|  | *  event     [I] Id of the event | 
|  | *  hwnd      [I] Window holding the object that created the event | 
|  | *  object_id [I] Type of object that created the event | 
|  | *  child_id  [I] Child object of nId, or CHILDID_SELF. | 
|  | * | 
|  | * RETURNS | 
|  | *  Nothing. | 
|  | */ | 
|  | void WINAPI NotifyWinEvent(DWORD event, HWND hwnd, LONG object_id, LONG child_id) | 
|  | { | 
|  | struct hook_info info; | 
|  |  | 
|  | TRACE("%04x,%p,%d,%d\n", event, hwnd, object_id, child_id); | 
|  |  | 
|  | if (!hwnd) | 
|  | { | 
|  | SetLastError(ERROR_INVALID_WINDOW_HANDLE); | 
|  | return; | 
|  | } | 
|  |  | 
|  | USER_CheckNotLock(); | 
|  |  | 
|  | #if 0 | 
|  | if (event & 0x80000000) | 
|  | { | 
|  | /* FIXME: on 64-bit platforms we need to invent some other way for | 
|  | * passing parameters, nId and nChildId can't hold full [W|L]PARAM. | 
|  | * struct call_hook *hook = (LRESULT *)hWnd; | 
|  | * wparam = hook->wparam; | 
|  | * lparam = hook->lparam; | 
|  | */ | 
|  | LRESULT *ret = (LRESULT *)hwnd; | 
|  | INT id, code, unicode; | 
|  |  | 
|  | id = (dwEvent & 0x7fff0000) >> 16; | 
|  | code = event & 0x7fff; | 
|  | unicode = event & 0x8000; | 
|  | *ret = HOOK_CallHooks(id, code, object_id, child_id, unicode); | 
|  | return; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | if (!find_first_hook(WH_WINEVENT, event, hwnd, object_id, child_id, &info)) return; | 
|  |  | 
|  | do | 
|  | { | 
|  | if (info.proc) | 
|  | { | 
|  | TRACE( "calling WH_WINEVENT hook %p event %x hwnd %p %x %x module %s\n", | 
|  | info.proc, event, hwnd, object_id, child_id, debugstr_w(info.module) ); | 
|  |  | 
|  | if (!info.module[0] || (info.proc = get_hook_proc( info.proc, info.module )) != NULL) | 
|  | { | 
|  | if (TRACE_ON(relay)) | 
|  | DPRINTF( "%04x:Call winevent hook proc %p (hhook=%p,event=%x,hwnd=%p,object_id=%x,child_id=%x,tid=%04x,time=%x)\n", | 
|  | GetCurrentThreadId(), info.proc, info.handle, event, hwnd, object_id, | 
|  | child_id, GetCurrentThreadId(), GetCurrentTime()); | 
|  |  | 
|  | info.proc(info.handle, event, hwnd, object_id, child_id, | 
|  | GetCurrentThreadId(), GetCurrentTime()); | 
|  |  | 
|  | if (TRACE_ON(relay)) | 
|  | DPRINTF( "%04x:Ret  winevent hook proc %p (hhook=%p,event=%x,hwnd=%p,object_id=%x,child_id=%x,tid=%04x,time=%x)\n", | 
|  | GetCurrentThreadId(), info.proc, info.handle, event, hwnd, object_id, | 
|  | child_id, GetCurrentThreadId(), GetCurrentTime()); | 
|  | } | 
|  | } | 
|  | else | 
|  | break; | 
|  | } | 
|  | while (find_next_hook(event, hwnd, object_id, child_id, &info)); | 
|  |  | 
|  | find_hook_close(WH_WINEVENT); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           IsWinEventHookInstalled                       [USER32.@] | 
|  | * | 
|  | * Determine if an event hook is installed for an event. | 
|  | * | 
|  | * PARAMS | 
|  | *  dwEvent  [I] Id of the event | 
|  | * | 
|  | * RETURNS | 
|  | *  TRUE,  If there are any hooks installed for the event. | 
|  | *  FALSE, Otherwise. | 
|  | * | 
|  | * BUGS | 
|  | *  Not implemented. | 
|  | */ | 
|  | BOOL WINAPI IsWinEventHookInstalled(DWORD dwEvent) | 
|  | { | 
|  | FIXME("(%d)-stub!\n", dwEvent); | 
|  | return TRUE; | 
|  | } |