/* User-based primary surface driver * * Copyright 2000-2001 TransGaming Technologies Inc. */ #include "config.h" #include #include #include #include "winerror.h" #include "debugtools.h" #include "ddraw_private.h" #include "dsurface/main.h" #include "dsurface/dib.h" #include "dsurface/user.h" #include "dsurface/wndproc.h" DEFAULT_DEBUG_CHANNEL(ddraw); /* if you use OWN_WINDOW, don't use SYNC_UPDATE, or you may get trouble */ /* #define SYNC_UPDATE */ #define OWN_WINDOW #ifdef OWN_WINDOW static void User_create_own_window(IDirectDrawSurfaceImpl* This); static void User_destroy_own_window(IDirectDrawSurfaceImpl* This); #endif #ifndef SYNC_UPDATE static DWORD CALLBACK User_update_thread(LPVOID); #endif static void User_copy_to_screen(IDirectDrawSurfaceImpl* This, LPCRECT rc); static void User_copy_from_screen(IDirectDrawSurfaceImpl* This, LPCRECT rc); static HWND get_display_window(IDirectDrawSurfaceImpl* This, LPPOINT pt); static ICOM_VTABLE(IDirectDrawSurface7) User_IDirectDrawSurface7_VTable; HRESULT User_DirectDrawSurface_Construct(IDirectDrawSurfaceImpl* This, IDirectDrawImpl* pDD, const DDSURFACEDESC2* pDDSD) { USER_PRIV_VAR(priv, This); HRESULT hr; TRACE("(%p,%p,%p)\n",This,pDD,pDDSD); hr = DIB_DirectDrawSurface_Construct(This, pDD, pDDSD); if (FAILED(hr)) return hr; ICOM_INIT_INTERFACE(This, IDirectDrawSurface7, User_IDirectDrawSurface7_VTable); This->final_release = User_DirectDrawSurface_final_release; This->duplicate_surface = User_DirectDrawSurface_duplicate_surface; This->lock_update = User_DirectDrawSurface_lock_update; This->unlock_update = User_DirectDrawSurface_unlock_update; This->flip_data = User_DirectDrawSurface_flip_data; This->flip_update = User_DirectDrawSurface_flip_update; This->get_dc = User_DirectDrawSurface_get_dc; This->release_dc = User_DirectDrawSurface_release_dc; This->set_palette = User_DirectDrawSurface_set_palette; This->update_palette = User_DirectDrawSurface_update_palette; This->get_gamma_ramp = User_DirectDrawSurface_get_gamma_ramp; This->set_gamma_ramp = User_DirectDrawSurface_set_gamma_ramp; This->get_display_window = User_DirectDrawSurface_get_display_window; if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) { #ifdef OWN_WINDOW DirectDrawSurface_RegisterClass(); #endif #ifndef SYNC_UPDATE priv->user.update_event = CreateEventA(NULL, FALSE, FALSE, NULL); priv->user.update_thread = CreateThread(NULL, 0, User_update_thread, This, 0, NULL); #ifdef OWN_WINDOW if (This->ddraw_owner->cooperative_level & DDSCL_FULLSCREEN) { /* wait for window creation (or update thread destruction) */ while (WaitForMultipleObjects(1, &priv->user.update_thread, FALSE, 10) == WAIT_TIMEOUT) if (This->more.lpDDRAWReserved) break; if (!This->more.lpDDRAWReserved) { ERR("window creation failed\n"); } } #endif #else #ifdef OWN_WINDOW User_create_own_window(This); #endif #endif if (!This->more.lpDDRAWReserved) This->more.lpDDRAWReserved = (LPVOID)pDD->window; } return DIB_DirectDrawSurface_alloc_dc(This, &priv->user.cached_dc); } HRESULT User_DirectDrawSurface_Create(IDirectDrawImpl *pDD, const DDSURFACEDESC2 *pDDSD, LPDIRECTDRAWSURFACE7 *ppSurf, IUnknown *pUnkOuter) { IDirectDrawSurfaceImpl* This; HRESULT hr; assert(pUnkOuter == NULL); This = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*This) + sizeof(User_DirectDrawSurfaceImpl)); if (This == NULL) return E_OUTOFMEMORY; This->private = (User_DirectDrawSurfaceImpl*)(This+1); hr = User_DirectDrawSurface_Construct(This, pDD, pDDSD); if (FAILED(hr)) HeapFree(GetProcessHeap(), 0, This); else *ppSurf = ICOM_INTERFACE(This, IDirectDrawSurface7); return hr; } void User_DirectDrawSurface_final_release(IDirectDrawSurfaceImpl* This) { USER_PRIV_VAR(priv, This); if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) { #ifndef SYNC_UPDATE HANDLE event = priv->user.update_event; priv->user.update_event = 0; SetEvent(event); TRACE("waiting for update thread to terminate...\n"); #ifdef OWN_WINDOW /* dispatch any waiting sendmessages */ { MSG msg; PeekMessageA(&msg, 0, 0, 0, PM_NOREMOVE); } /* to avoid deadlocks, allow SendMessages from update thread * through while we wait for it */ while (MsgWaitForMultipleObjects(1, &priv->user.update_thread, FALSE, INFINITE, QS_SENDMESSAGE) == WAIT_OBJECT_0+1) { MSG msg; PeekMessageA(&msg, 0, 0, 0, PM_NOREMOVE); } #else WaitForSingleObject(priv->user.update_thread,INFINITE); #endif TRACE("update thread terminated\n"); CloseHandle(priv->user.update_thread); #else #ifdef OWN_WINDOW User_destroy_own_window(This); #endif #endif This->more.lpDDRAWReserved = 0; #ifdef OWN_WINDOW DirectDrawSurface_UnregisterClass(); #endif } DIB_DirectDrawSurface_free_dc(This, priv->user.cached_dc); DIB_DirectDrawSurface_final_release(This); } void User_DirectDrawSurface_lock_update(IDirectDrawSurfaceImpl* This, LPCRECT pRect) { User_copy_from_screen(This, pRect); } void User_DirectDrawSurface_unlock_update(IDirectDrawSurfaceImpl* This, LPCRECT pRect) { #ifdef SYNC_UPDATE User_copy_to_screen(This, pRect); #else USER_PRIV_VAR(priv, This); SetEvent(priv->user.update_event); #endif } void User_DirectDrawSurface_set_palette(IDirectDrawSurfaceImpl* This, IDirectDrawPaletteImpl* pal) { USER_PRIV_VAR(priv, This); if (!pal) { FIXME("selecting null palette\n"); /* I don't think selecting GDI object 0 will work, we should select * the original palette, returned by the first SelectPalette */ SelectPalette(priv->user.cached_dc, 0, FALSE); return; } SelectPalette(priv->user.cached_dc, pal->hpal, FALSE); DIB_DirectDrawSurface_set_palette(This, pal); } void User_DirectDrawSurface_update_palette(IDirectDrawSurfaceImpl* This, IDirectDrawPaletteImpl* pal, DWORD dwStart, DWORD dwCount, LPPALETTEENTRY palent) { USER_PRIV_VAR(priv, This); DIB_DirectDrawSurface_update_palette(This, pal, dwStart, dwCount, palent); /* FIXME: realize palette on display window */ #ifndef SYNC_UPDATE /* with async updates, it's probably cool to force an update */ SetEvent(priv->user.update_event); #endif } HRESULT User_DirectDrawSurface_duplicate_surface(IDirectDrawSurfaceImpl* This, LPDIRECTDRAWSURFACE7* ppDup) { return User_DirectDrawSurface_Create(This->ddraw_owner, &This->surface_desc, ppDup, NULL); } BOOL User_DirectDrawSurface_flip_data(IDirectDrawSurfaceImpl* front, IDirectDrawSurfaceImpl* back, DWORD dwFlags) { USER_PRIV_VAR(front_priv, front); USER_PRIV_VAR(back_priv, back); { HDC tmp; tmp = front_priv->user.cached_dc; front_priv->user.cached_dc = back_priv->user.cached_dc; back_priv->user.cached_dc = tmp; } return DIB_DirectDrawSurface_flip_data(front, back, dwFlags); } void User_DirectDrawSurface_flip_update(IDirectDrawSurfaceImpl* This, DWORD dwFlags) { #ifdef SYNC_UPDATE assert(This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE); User_copy_to_screen(This,NULL); #else USER_PRIV_VAR(priv, This); assert(This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE); SetEvent(priv->user.update_event); #endif } HRESULT User_DirectDrawSurface_get_dc(IDirectDrawSurfaceImpl* This, HDC* phDC) { USER_PRIV_VAR(priv, This); *phDC = priv->user.cached_dc; return S_OK; } HRESULT User_DirectDrawSurface_release_dc(IDirectDrawSurfaceImpl* This, HDC hDC) { return S_OK; } HRESULT User_DirectDrawSurface_get_gamma_ramp(IDirectDrawSurfaceImpl* This, DWORD dwFlags, LPDDGAMMARAMP lpGammaRamp) { if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) { POINT offset; HWND hDisplayWnd; HDC hDisplayDC; HRESULT hr; hDisplayWnd = get_display_window(This, &offset); hDisplayDC = GetDCEx(hDisplayWnd, 0, DCX_CLIPSIBLINGS); hr = GetDeviceGammaRamp(hDisplayDC, lpGammaRamp) ? DD_OK : DDERR_UNSUPPORTED; ReleaseDC(hDisplayWnd, hDisplayDC); return hr; } return Main_DirectDrawSurface_get_gamma_ramp(This, dwFlags, lpGammaRamp); } HRESULT User_DirectDrawSurface_set_gamma_ramp(IDirectDrawSurfaceImpl* This, DWORD dwFlags, LPDDGAMMARAMP lpGammaRamp) { if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) { POINT offset; HWND hDisplayWnd; HDC hDisplayDC; HRESULT hr; hDisplayWnd = get_display_window(This, &offset); hDisplayDC = GetDCEx(hDisplayWnd, 0, DCX_CLIPSIBLINGS); hr = SetDeviceGammaRamp(hDisplayDC, lpGammaRamp) ? DD_OK : DDERR_UNSUPPORTED; ReleaseDC(hDisplayWnd, hDisplayDC); return hr; } return Main_DirectDrawSurface_set_gamma_ramp(This, dwFlags, lpGammaRamp); } /* Returns the window that hosts the display. * *pt is set to the upper left position of the window relative to the * upper left corner of the surface. */ static HWND get_display_window(IDirectDrawSurfaceImpl* This, LPPOINT pt) { memset(pt, 0, sizeof(*pt)); if (This->ddraw_owner->cooperative_level & DDSCL_FULLSCREEN) { #ifdef OWN_WINDOW USER_PRIV_VAR(priv, This); #if 1 SetWindowPos(priv->user.window, HWND_TOP, 0, 0, 0, 0, SWP_DEFERERASE|SWP_NOACTIVATE|SWP_NOCOPYBITS|SWP_NOMOVE| SWP_NOREDRAW|SWP_NOSENDCHANGING|SWP_NOSIZE); #endif return priv->user.window; #else return This->ddraw_owner->window; #endif } else { if (This->clipper != NULL) { /* looks silly, but since we don't have the clipper locked */ HWND hWnd = This->clipper->hWnd; if (hWnd != 0) { ClientToScreen(hWnd, pt); return hWnd; } else { static BOOL warn = 0; if (!warn++) FIXME("clipper clip lists not supported\n"); return GetDesktopWindow(); } } else { static BOOL warn = 0; if (!warn++) WARN("hosting on root\n"); return GetDesktopWindow(); } } } HWND User_DirectDrawSurface_get_display_window(IDirectDrawSurfaceImpl* This) { POINT offset; return get_display_window(This, &offset); } #ifdef OWN_WINDOW static void User_create_own_window(IDirectDrawSurfaceImpl* This) { USER_PRIV_VAR(priv, This); if (This->ddraw_owner->cooperative_level & DDSCL_FULLSCREEN) { priv->user.window = CreateWindowExA(WS_EX_TOPMOST | WS_EX_LAYERED | WS_EX_TRANSPARENT, "WINE_DDRAW", "DirectDraw", WS_POPUP, 0, 0, This->surface_desc.dwWidth, This->surface_desc.dwHeight, GetDesktopWindow(), 0, 0, This); This->more.lpDDRAWReserved = (LPVOID)priv->user.window; SetWindowPos(priv->user.window, HWND_TOP, 0, 0, 0, 0, SWP_DEFERERASE|SWP_NOACTIVATE|SWP_NOCOPYBITS|SWP_NOMOVE| SWP_NOREDRAW|SWP_NOSENDCHANGING|SWP_NOSIZE|SWP_SHOWWINDOW); UpdateWindow(priv->user.window); } } static void User_destroy_own_window(IDirectDrawSurfaceImpl* This) { USER_PRIV_VAR(priv, This); if (priv->user.window) { SetWindowPos(priv->user.window, 0, 0, 0, 0, 0, SWP_DEFERERASE|SWP_NOACTIVATE|SWP_NOCOPYBITS|SWP_NOMOVE|SWP_NOZORDER| SWP_NOREDRAW|SWP_NOSENDCHANGING|SWP_NOSIZE|SWP_HIDEWINDOW); This->more.lpDDRAWReserved = NULL; DestroyWindow(priv->user.window); priv->user.window = 0; } } #endif #ifndef SYNC_UPDATE static DWORD CALLBACK User_update_thread(LPVOID arg) { IDirectDrawSurfaceImpl *This = (IDirectDrawSurfaceImpl *)arg; USER_PRIV_VAR(priv, This); volatile DWORD *pActive = (volatile DWORD *)&priv->user.update_event; HANDLE event = *pActive; #ifdef OWN_WINDOW User_create_own_window(This); #endif /* the point of this is that many games lock the primary surface * multiple times per frame; this thread will then simply copy as * often as it can without keeping the main thread waiting for * each unlock, thus keeping the frame rate high */ do { #ifdef OWN_WINDOW DWORD ret = MsgWaitForMultipleObjects(1, &event, FALSE, INFINITE, QS_ALLINPUT); MSG msg; while (PeekMessageA(&msg, 0, 0, 0, PM_REMOVE)) { switch (msg.message) { case WM_PAINT: DispatchMessageA(&msg); break; default: /* since we risk getting keyboard messages posted to us, * repost posted messages to cooperative window */ PostMessageA(This->ddraw_owner->window, msg.message, msg.wParam, msg.lParam); } } #else DWORD ret = WaitForSingleObject(event, INFINITE); #endif if (ret == WAIT_OBJECT_0) { if (*pActive) User_copy_to_screen(This, NULL); else break; } else if (ret != WAIT_OBJECT_0+1) break; } while (TRUE); #ifdef OWN_WINDOW User_destroy_own_window(This); #endif return 0; } #endif static void User_copy_to_screen(IDirectDrawSurfaceImpl* This, LPCRECT rc) { /* rc is unused. We copy the whole thing. */ if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) { POINT offset; HWND hDisplayWnd; HDC hDisplayDC; HDC hSurfaceDC; if (FAILED(This->get_dc(This, &hSurfaceDC))) return; hDisplayWnd = get_display_window(This, &offset); hDisplayDC = GetDCEx(hDisplayWnd, 0, DCX_CLIPSIBLINGS); #if 0 /* FIXME: this doesn't work... if users really want to run * X in 8bpp, then we need to call directly into display.drv * (or Wine's equivalent), and force a private colormap * without default entries. */ if (This->palette) { SelectPalette(hDisplayDC, This->palette->hpal, FALSE); RealizePalette(hDisplayDC); /* sends messages => deadlocks */ } #endif BitBlt(hDisplayDC, 0, 0, This->surface_desc.dwWidth, This->surface_desc.dwHeight, hSurfaceDC, offset.x, offset.y, SRCCOPY); ReleaseDC(hDisplayWnd, hDisplayDC); } } static void User_copy_from_screen(IDirectDrawSurfaceImpl* This, LPCRECT rc) { /* rc is unused. We copy the whole thing. */ if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) { POINT offset; HWND hDisplayWnd = get_display_window(This, &offset); HDC hDisplayDC = GetDC(hDisplayWnd); BitBlt(This->hDC, offset.x, offset.y, This->surface_desc.dwWidth, This->surface_desc.dwHeight, hDisplayDC, 0, 0, SRCCOPY); ReleaseDC(hDisplayWnd, hDisplayDC); } } static ICOM_VTABLE(IDirectDrawSurface7) User_IDirectDrawSurface7_VTable = { Main_DirectDrawSurface_QueryInterface, Main_DirectDrawSurface_AddRef, Main_DirectDrawSurface_Release, Main_DirectDrawSurface_AddAttachedSurface, Main_DirectDrawSurface_AddOverlayDirtyRect, DIB_DirectDrawSurface_Blt, Main_DirectDrawSurface_BltBatch, DIB_DirectDrawSurface_BltFast, Main_DirectDrawSurface_DeleteAttachedSurface, Main_DirectDrawSurface_EnumAttachedSurfaces, Main_DirectDrawSurface_EnumOverlayZOrders, Main_DirectDrawSurface_Flip, Main_DirectDrawSurface_GetAttachedSurface, Main_DirectDrawSurface_GetBltStatus, Main_DirectDrawSurface_GetCaps, Main_DirectDrawSurface_GetClipper, Main_DirectDrawSurface_GetColorKey, Main_DirectDrawSurface_GetDC, Main_DirectDrawSurface_GetFlipStatus, Main_DirectDrawSurface_GetOverlayPosition, Main_DirectDrawSurface_GetPalette, Main_DirectDrawSurface_GetPixelFormat, Main_DirectDrawSurface_GetSurfaceDesc, Main_DirectDrawSurface_Initialize, Main_DirectDrawSurface_IsLost, Main_DirectDrawSurface_Lock, Main_DirectDrawSurface_ReleaseDC, DIB_DirectDrawSurface_Restore, Main_DirectDrawSurface_SetClipper, Main_DirectDrawSurface_SetColorKey, Main_DirectDrawSurface_SetOverlayPosition, Main_DirectDrawSurface_SetPalette, Main_DirectDrawSurface_Unlock, Main_DirectDrawSurface_UpdateOverlay, Main_DirectDrawSurface_UpdateOverlayDisplay, Main_DirectDrawSurface_UpdateOverlayZOrder, Main_DirectDrawSurface_GetDDInterface, Main_DirectDrawSurface_PageLock, Main_DirectDrawSurface_PageUnlock, DIB_DirectDrawSurface_SetSurfaceDesc, Main_DirectDrawSurface_SetPrivateData, Main_DirectDrawSurface_GetPrivateData, Main_DirectDrawSurface_FreePrivateData, Main_DirectDrawSurface_GetUniquenessValue, Main_DirectDrawSurface_ChangeUniquenessValue, Main_DirectDrawSurface_SetPriority, Main_DirectDrawSurface_GetPriority, Main_DirectDrawSurface_SetLOD, Main_DirectDrawSurface_GetLOD };