4 * Copyright 2000, 2005 Alexandre Julliard
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
27 #include "user_private.h"
28 #include "wine/debug.h"
30 WINE_DEFAULT_DEBUG_CHANNEL(user);
32 static const USER_DRIVER null_driver, lazy_load_driver;
34 const USER_DRIVER *USER_Driver = &lazy_load_driver;
36 /* load the graphics driver */
37 static const USER_DRIVER *load_driver(void)
39 char buffer[MAX_PATH], libname[32], *name, *next;
42 HMODULE graphics_driver;
43 USER_DRIVER *driver, *prev;
45 strcpy( buffer, "x11,tty" ); /* default value */
46 /* @@ Wine registry key: HKCU\Software\Wine\Drivers */
47 if (!RegOpenKeyA( HKEY_CURRENT_USER, "Software\\Wine\\Drivers", &hkey ))
49 DWORD type, count = sizeof(buffer);
50 RegQueryValueExA( hkey, "Graphics", 0, &type, (LPBYTE) buffer, &count );
57 next = strchr( name, ',' );
58 if (next) *next++ = 0;
60 snprintf( libname, sizeof(libname), "wine%s.drv", name );
61 if ((graphics_driver = LoadLibraryA( libname )) != 0) break;
66 MESSAGE( "wine: Could not load graphics driver '%s'.\n", buffer );
67 if (!strcasecmp( buffer, "x11" ))
68 MESSAGE( "Make sure that your X server is running and that $DISPLAY is set correctly.\n" );
72 driver = HeapAlloc( GetProcessHeap(), 0, sizeof(*driver) );
73 memcpy( driver, &null_driver, sizeof(*driver) );
75 #define GET_USER_FUNC(name) \
76 do { if ((ptr = GetProcAddress( graphics_driver, #name ))) driver->p##name = ptr; } while(0)
78 GET_USER_FUNC(ActivateKeyboardLayout);
80 GET_USER_FUNC(GetAsyncKeyState);
81 GET_USER_FUNC(GetKeyNameText);
82 GET_USER_FUNC(GetKeyboardLayout);
83 GET_USER_FUNC(GetKeyboardLayoutList);
84 GET_USER_FUNC(GetKeyboardLayoutName);
85 GET_USER_FUNC(LoadKeyboardLayout);
86 GET_USER_FUNC(MapVirtualKeyEx);
87 GET_USER_FUNC(SendInput);
88 GET_USER_FUNC(ToUnicodeEx);
89 GET_USER_FUNC(UnloadKeyboardLayout);
90 GET_USER_FUNC(VkKeyScanEx);
91 GET_USER_FUNC(SetCursor);
92 GET_USER_FUNC(GetCursorPos);
93 GET_USER_FUNC(SetCursorPos);
94 GET_USER_FUNC(GetScreenSaveActive);
95 GET_USER_FUNC(SetScreenSaveActive);
96 GET_USER_FUNC(AcquireClipboard);
97 GET_USER_FUNC(EmptyClipboard);
98 GET_USER_FUNC(SetClipboardData);
99 GET_USER_FUNC(GetClipboardData);
100 GET_USER_FUNC(CountClipboardFormats);
101 GET_USER_FUNC(EnumClipboardFormats);
102 GET_USER_FUNC(IsClipboardFormatAvailable);
103 GET_USER_FUNC(RegisterClipboardFormat);
104 GET_USER_FUNC(GetClipboardFormatName);
105 GET_USER_FUNC(EndClipboardUpdate);
106 GET_USER_FUNC(ResetSelectionOwner);
107 GET_USER_FUNC(ChangeDisplaySettingsEx);
108 GET_USER_FUNC(EnumDisplaySettingsEx);
109 GET_USER_FUNC(CreateDesktopWindow);
110 GET_USER_FUNC(CreateWindow);
111 GET_USER_FUNC(DestroyWindow);
112 GET_USER_FUNC(GetDCEx);
113 GET_USER_FUNC(MsgWaitForMultipleObjectsEx);
114 GET_USER_FUNC(ReleaseDC);
115 GET_USER_FUNC(ScrollDC);
116 GET_USER_FUNC(SetFocus);
117 GET_USER_FUNC(SetParent);
118 GET_USER_FUNC(SetWindowPos);
119 GET_USER_FUNC(SetWindowRgn);
120 GET_USER_FUNC(SetWindowIcon);
121 GET_USER_FUNC(SetWindowStyle);
122 GET_USER_FUNC(SetWindowText);
123 GET_USER_FUNC(ShowWindow);
124 GET_USER_FUNC(SysCommandSizeMove);
125 GET_USER_FUNC(WindowFromDC);
126 GET_USER_FUNC(WindowMessage);
129 prev = InterlockedCompareExchangePointer( (void **)&USER_Driver, driver, (void *)&lazy_load_driver );
130 if (prev != &lazy_load_driver)
132 /* another thread beat us to it */
133 HeapFree( GetProcessHeap(), 0, driver );
134 FreeLibrary( graphics_driver );
141 /**********************************************************************
144 * These are fallbacks for entry points that are not implemented in the real driver.
147 static HKL nulldrv_ActivateKeyboardLayout( HKL layout, UINT flags )
152 static void nulldrv_Beep(void)
156 static SHORT nulldrv_GetAsyncKeyState( INT key )
161 static INT nulldrv_GetKeyNameText( LONG lparam, LPWSTR buffer, INT size )
166 static HKL nulldrv_GetKeyboardLayout( DWORD layout )
171 static UINT nulldrv_GetKeyboardLayoutList( INT count, HKL *layouts )
176 static BOOL nulldrv_GetKeyboardLayoutName( LPWSTR name )
181 static HKL nulldrv_LoadKeyboardLayout( LPCWSTR name, UINT flags )
186 static UINT nulldrv_MapVirtualKeyEx( UINT code, UINT type, HKL layout )
191 static UINT nulldrv_SendInput( UINT count, LPINPUT inputs, int size )
196 static INT nulldrv_ToUnicodeEx( UINT virt, UINT scan, LPBYTE state, LPWSTR str,
197 int size, UINT flags, HKL layout )
202 static BOOL nulldrv_UnloadKeyboardLayout( HKL layout )
207 static SHORT nulldrv_VkKeyScanEx( WCHAR ch, HKL layout )
212 static void nulldrv_SetCursor( struct tagCURSORICONINFO *info )
216 static BOOL nulldrv_GetCursorPos( LPPOINT pt )
221 static BOOL nulldrv_SetCursorPos( INT x, INT y )
226 static BOOL nulldrv_GetScreenSaveActive(void)
231 static void nulldrv_SetScreenSaveActive( BOOL on )
235 static void nulldrv_AcquireClipboard( HWND hwnd )
239 static BOOL nulldrv_CountClipboardFormats(void)
244 static void nulldrv_EmptyClipboard( BOOL keepunowned )
248 static void nulldrv_EndClipboardUpdate(void)
252 static UINT nulldrv_EnumClipboardFormats( UINT format )
257 static BOOL nulldrv_GetClipboardData( UINT format, HANDLE16 *h16, HANDLE *h32 )
262 static INT nulldrv_GetClipboardFormatName( UINT format, LPWSTR buffer, UINT len )
267 static BOOL nulldrv_IsClipboardFormatAvailable( UINT format )
272 static UINT nulldrv_RegisterClipboardFormat( LPCWSTR name )
277 static void nulldrv_ResetSelectionOwner( HWND hwnd, BOOL flag )
281 static BOOL nulldrv_SetClipboardData( UINT format, HANDLE16 h16, HANDLE h32, BOOL owner )
286 static LONG nulldrv_ChangeDisplaySettingsEx( LPCWSTR name, LPDEVMODEW mode, HWND hwnd,
287 DWORD flags, LPVOID lparam )
289 return DISP_CHANGE_FAILED;
292 static BOOL nulldrv_EnumDisplaySettingsEx( LPCWSTR name, DWORD num, LPDEVMODEW mode, DWORD flags )
297 static BOOL nulldrv_CreateDesktopWindow( HWND hwnd )
302 static BOOL nulldrv_CreateWindow( HWND hwnd, CREATESTRUCTA *cs, BOOL unicode )
307 static void nulldrv_DestroyWindow( HWND hwnd )
311 static HDC nulldrv_GetDCEx( HWND hwnd, HRGN hrgn, DWORD flags )
316 static DWORD nulldrv_MsgWaitForMultipleObjectsEx( DWORD count, const HANDLE *handles, DWORD timeout,
317 DWORD mask, DWORD flags )
319 return WaitForMultipleObjectsEx( count, handles, flags & MWMO_WAITALL,
320 timeout, flags & MWMO_ALERTABLE );
323 static INT nulldrv_ReleaseDC( HWND hwnd, HDC hdc, BOOL end_paint )
328 static BOOL nulldrv_ScrollDC( HDC hdc, INT dx, INT dy, const RECT *scroll, const RECT *clip,
329 HRGN hrgn, LPRECT update )
334 static void nulldrv_SetFocus( HWND hwnd )
338 static HWND nulldrv_SetParent( HWND hwnd, HWND parent )
343 static BOOL nulldrv_SetWindowPos( WINDOWPOS *pos )
348 static int nulldrv_SetWindowRgn( HWND hwnd, HRGN hrgn, BOOL redraw )
353 static void nulldrv_SetWindowIcon( HWND hwnd, UINT type, HICON icon )
357 static void nulldrv_SetWindowStyle( HWND hwnd, DWORD old_style )
361 static void nulldrv_SetWindowText( HWND hwnd, LPCWSTR text )
365 static BOOL nulldrv_ShowWindow( HWND hwnd, INT cmd )
370 static void nulldrv_SysCommandSizeMove( HWND hwnd, WPARAM wparam )
374 static HWND nulldrv_WindowFromDC( HDC hdc )
379 static LRESULT nulldrv_WindowMessage( HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam )
384 static const USER_DRIVER null_driver =
386 /* keyboard functions */
387 nulldrv_ActivateKeyboardLayout,
389 nulldrv_GetAsyncKeyState,
390 nulldrv_GetKeyNameText,
391 nulldrv_GetKeyboardLayout,
392 nulldrv_GetKeyboardLayoutList,
393 nulldrv_GetKeyboardLayoutName,
394 nulldrv_LoadKeyboardLayout,
395 nulldrv_MapVirtualKeyEx,
398 nulldrv_UnloadKeyboardLayout,
400 /* mouse functions */
402 nulldrv_GetCursorPos,
403 nulldrv_SetCursorPos,
404 /* screen saver functions */
405 nulldrv_GetScreenSaveActive,
406 nulldrv_SetScreenSaveActive,
407 /* clipboard functions */
408 nulldrv_AcquireClipboard,
409 nulldrv_CountClipboardFormats,
410 nulldrv_EmptyClipboard,
411 nulldrv_EndClipboardUpdate,
412 nulldrv_EnumClipboardFormats,
413 nulldrv_GetClipboardData,
414 nulldrv_GetClipboardFormatName,
415 nulldrv_IsClipboardFormatAvailable,
416 nulldrv_RegisterClipboardFormat,
417 nulldrv_ResetSelectionOwner,
418 nulldrv_SetClipboardData,
420 nulldrv_ChangeDisplaySettingsEx,
421 nulldrv_EnumDisplaySettingsEx,
422 /* windowing functions */
423 nulldrv_CreateDesktopWindow,
424 nulldrv_CreateWindow,
425 nulldrv_DestroyWindow,
427 nulldrv_MsgWaitForMultipleObjectsEx,
432 nulldrv_SetWindowPos,
433 nulldrv_SetWindowRgn,
434 nulldrv_SetWindowIcon,
435 nulldrv_SetWindowStyle,
436 nulldrv_SetWindowText,
438 nulldrv_SysCommandSizeMove,
439 nulldrv_WindowFromDC,
440 nulldrv_WindowMessage
444 /**********************************************************************
445 * Lazy loading user driver
447 * Initial driver used before another driver is loaded.
448 * Each entry point simply loads the real driver and chains to it.
451 static HKL loaderdrv_ActivateKeyboardLayout( HKL layout, UINT flags )
453 return load_driver()->pActivateKeyboardLayout( layout, flags );
456 static void loaderdrv_Beep(void)
458 load_driver()->pBeep();
461 static SHORT loaderdrv_GetAsyncKeyState( INT key )
463 return load_driver()->pGetAsyncKeyState( key );
466 static INT loaderdrv_GetKeyNameText( LONG lparam, LPWSTR buffer, INT size )
468 return load_driver()->pGetKeyNameText( lparam, buffer, size );
471 static HKL loaderdrv_GetKeyboardLayout( DWORD layout )
473 return load_driver()->pGetKeyboardLayout( layout );
476 static UINT loaderdrv_GetKeyboardLayoutList( INT count, HKL *layouts )
478 return load_driver()->pGetKeyboardLayoutList( count, layouts );
481 static BOOL loaderdrv_GetKeyboardLayoutName( LPWSTR name )
483 return load_driver()->pGetKeyboardLayoutName( name );
486 static HKL loaderdrv_LoadKeyboardLayout( LPCWSTR name, UINT flags )
488 return load_driver()->pLoadKeyboardLayout( name, flags );
491 static UINT loaderdrv_MapVirtualKeyEx( UINT code, UINT type, HKL layout )
493 return load_driver()->pMapVirtualKeyEx( code, type, layout );
496 static UINT loaderdrv_SendInput( UINT count, LPINPUT inputs, int size )
498 return load_driver()->pSendInput( count, inputs, size );
501 static INT loaderdrv_ToUnicodeEx( UINT virt, UINT scan, LPBYTE state, LPWSTR str,
502 int size, UINT flags, HKL layout )
504 return load_driver()->pToUnicodeEx( virt, scan, state, str, size, flags, layout );
507 static BOOL loaderdrv_UnloadKeyboardLayout( HKL layout )
509 return load_driver()->pUnloadKeyboardLayout( layout );
512 static SHORT loaderdrv_VkKeyScanEx( WCHAR ch, HKL layout )
514 return load_driver()->pVkKeyScanEx( ch, layout );
517 static void loaderdrv_SetCursor( struct tagCURSORICONINFO *info )
519 load_driver()->pSetCursor( info );
522 static BOOL loaderdrv_GetCursorPos( LPPOINT pt )
524 return load_driver()->pGetCursorPos( pt );
527 static BOOL loaderdrv_SetCursorPos( INT x, INT y )
529 return load_driver()->pSetCursorPos( x, y );
532 static BOOL loaderdrv_GetScreenSaveActive(void)
534 return load_driver()->pGetScreenSaveActive();
537 static void loaderdrv_SetScreenSaveActive( BOOL on )
539 load_driver()->pSetScreenSaveActive( on );
542 static void loaderdrv_AcquireClipboard( HWND hwnd )
544 load_driver()->pAcquireClipboard( hwnd );
547 static BOOL loaderdrv_CountClipboardFormats(void)
549 return load_driver()->pCountClipboardFormats();
552 static void loaderdrv_EmptyClipboard( BOOL keepunowned )
554 load_driver()->pEmptyClipboard( keepunowned );
557 static void loaderdrv_EndClipboardUpdate(void)
559 load_driver()->pEndClipboardUpdate();
562 static UINT loaderdrv_EnumClipboardFormats( UINT format )
564 return load_driver()->pEnumClipboardFormats( format );
567 static BOOL loaderdrv_GetClipboardData( UINT format, HANDLE16 *h16, HANDLE *h32 )
569 return load_driver()->pGetClipboardData( format, h16, h32 );
572 static INT loaderdrv_GetClipboardFormatName( UINT format, LPWSTR buffer, UINT len )
574 return load_driver()->pGetClipboardFormatName( format, buffer, len );
577 static BOOL loaderdrv_IsClipboardFormatAvailable( UINT format )
579 return load_driver()->pIsClipboardFormatAvailable( format );
582 static UINT loaderdrv_RegisterClipboardFormat( LPCWSTR name )
584 return load_driver()->pRegisterClipboardFormat( name );
587 static void loaderdrv_ResetSelectionOwner( HWND hwnd, BOOL flag )
589 load_driver()->pResetSelectionOwner( hwnd, flag );
592 static BOOL loaderdrv_SetClipboardData( UINT format, HANDLE16 h16, HANDLE h32, BOOL owner )
594 return load_driver()->pSetClipboardData( format, h16, h32, owner );
597 static LONG loaderdrv_ChangeDisplaySettingsEx( LPCWSTR name, LPDEVMODEW mode, HWND hwnd,
598 DWORD flags, LPVOID lparam )
600 return load_driver()->pChangeDisplaySettingsEx( name, mode, hwnd, flags, lparam );
603 static BOOL loaderdrv_EnumDisplaySettingsEx( LPCWSTR name, DWORD num, LPDEVMODEW mode, DWORD flags )
605 return load_driver()->pEnumDisplaySettingsEx( name, num, mode, flags );
608 static BOOL loaderdrv_CreateDesktopWindow( HWND hwnd )
610 return load_driver()->pCreateDesktopWindow( hwnd );
613 static BOOL loaderdrv_CreateWindow( HWND hwnd, CREATESTRUCTA *cs, BOOL unicode )
615 return load_driver()->pCreateWindow( hwnd, cs, unicode );
618 static void loaderdrv_DestroyWindow( HWND hwnd )
620 load_driver()->pDestroyWindow( hwnd );
623 static HDC loaderdrv_GetDCEx( HWND hwnd, HRGN hrgn, DWORD flags )
625 return load_driver()->pGetDCEx( hwnd, hrgn, flags );
628 static DWORD loaderdrv_MsgWaitForMultipleObjectsEx( DWORD count, const HANDLE *handles, DWORD timeout,
629 DWORD mask, DWORD flags )
631 return load_driver()->pMsgWaitForMultipleObjectsEx( count, handles, timeout, mask, flags );
634 static INT loaderdrv_ReleaseDC( HWND hwnd, HDC hdc, BOOL end_paint )
636 return load_driver()->pReleaseDC( hwnd, hdc, end_paint );
639 static BOOL loaderdrv_ScrollDC( HDC hdc, INT dx, INT dy, const RECT *scroll, const RECT *clip,
640 HRGN hrgn, LPRECT update )
642 return load_driver()->pScrollDC( hdc, dx, dy, scroll, clip, hrgn, update );
645 static void loaderdrv_SetFocus( HWND hwnd )
647 load_driver()->pSetFocus( hwnd );
650 static HWND loaderdrv_SetParent( HWND hwnd, HWND parent )
652 return load_driver()->pSetParent( hwnd, parent );
655 static BOOL loaderdrv_SetWindowPos( WINDOWPOS *pos )
657 return load_driver()->pSetWindowPos( pos );
660 static int loaderdrv_SetWindowRgn( HWND hwnd, HRGN hrgn, BOOL redraw )
662 return load_driver()->pSetWindowRgn( hwnd, hrgn, redraw );
665 static void loaderdrv_SetWindowIcon( HWND hwnd, UINT type, HICON icon )
667 load_driver()->pSetWindowIcon( hwnd, type, icon );
670 static void loaderdrv_SetWindowStyle( HWND hwnd, DWORD old_style )
672 load_driver()->pSetWindowStyle( hwnd, old_style );
675 static void loaderdrv_SetWindowText( HWND hwnd, LPCWSTR text )
677 load_driver()->pSetWindowText( hwnd, text );
680 static BOOL loaderdrv_ShowWindow( HWND hwnd, INT cmd )
682 return load_driver()->pShowWindow( hwnd, cmd );
685 static void loaderdrv_SysCommandSizeMove( HWND hwnd, WPARAM wparam )
687 load_driver()->pSysCommandSizeMove( hwnd, wparam );
690 static HWND loaderdrv_WindowFromDC( HDC hdc )
692 return load_driver()->pWindowFromDC( hdc );
695 static LRESULT loaderdrv_WindowMessage( HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam )
697 return load_driver()->pWindowMessage( hwnd, msg, wparam, lparam );
700 static const USER_DRIVER lazy_load_driver =
702 /* keyboard functions */
703 loaderdrv_ActivateKeyboardLayout,
705 loaderdrv_GetAsyncKeyState,
706 loaderdrv_GetKeyNameText,
707 loaderdrv_GetKeyboardLayout,
708 loaderdrv_GetKeyboardLayoutList,
709 loaderdrv_GetKeyboardLayoutName,
710 loaderdrv_LoadKeyboardLayout,
711 loaderdrv_MapVirtualKeyEx,
713 loaderdrv_ToUnicodeEx,
714 loaderdrv_UnloadKeyboardLayout,
715 loaderdrv_VkKeyScanEx,
716 /* mouse functions */
718 loaderdrv_GetCursorPos,
719 loaderdrv_SetCursorPos,
720 /* screen saver functions */
721 loaderdrv_GetScreenSaveActive,
722 loaderdrv_SetScreenSaveActive,
723 /* clipboard functions */
724 loaderdrv_AcquireClipboard,
725 loaderdrv_CountClipboardFormats,
726 loaderdrv_EmptyClipboard,
727 loaderdrv_EndClipboardUpdate,
728 loaderdrv_EnumClipboardFormats,
729 loaderdrv_GetClipboardData,
730 loaderdrv_GetClipboardFormatName,
731 loaderdrv_IsClipboardFormatAvailable,
732 loaderdrv_RegisterClipboardFormat,
733 loaderdrv_ResetSelectionOwner,
734 loaderdrv_SetClipboardData,
736 loaderdrv_ChangeDisplaySettingsEx,
737 loaderdrv_EnumDisplaySettingsEx,
738 /* windowing functions */
739 loaderdrv_CreateDesktopWindow,
740 loaderdrv_CreateWindow,
741 loaderdrv_DestroyWindow,
743 loaderdrv_MsgWaitForMultipleObjectsEx,
748 loaderdrv_SetWindowPos,
749 loaderdrv_SetWindowRgn,
750 loaderdrv_SetWindowIcon,
751 loaderdrv_SetWindowStyle,
752 loaderdrv_SetWindowText,
753 loaderdrv_ShowWindow,
754 loaderdrv_SysCommandSizeMove,
755 loaderdrv_WindowFromDC,
756 loaderdrv_WindowMessage