| /* User-based primary surface driver |
| * |
| * Copyright 2000-2001 TransGaming Technologies Inc. |
| * |
| * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include "config.h" |
| |
| #include <assert.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| #include "winerror.h" |
| #include "wine/debug.h" |
| #include "ddraw_private.h" |
| #include "dsurface/main.h" |
| #include "dsurface/dib.h" |
| #include "dsurface/user.h" |
| #include "dsurface/wndproc.h" |
| |
| WINE_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, DWORD dwFlags) |
| { |
| if (!(dwFlags & DDLOCK_WRITEONLY)) |
| User_copy_from_screen(This, pRect); |
| |
| if (pRect) { |
| This->lastlockrect = *pRect; |
| } else { |
| This->lastlockrect.left = This->lastlockrect.right = 0; |
| } |
| } |
| |
| 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 |
| This->lastlockrect.left = This->lastlockrect.right = 0; |
| assert(This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE); |
| User_copy_to_screen(This,NULL); |
| #else |
| USER_PRIV_VAR(priv, This); |
| This->lastlockrect.left = This->lastlockrect.right = 0; |
| 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|DCX_CACHE); |
| 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|DCX_CACHE); |
| 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) |
| { |
| if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) |
| { |
| POINT offset; |
| HWND hDisplayWnd; |
| HDC hDisplayDC; |
| HDC hSurfaceDC; |
| RECT drawrect; |
| |
| if (FAILED(This->get_dc(This, &hSurfaceDC))) |
| return; |
| |
| hDisplayWnd = get_display_window(This, &offset); |
| hDisplayDC = GetDCEx(hDisplayWnd, 0, DCX_CLIPSIBLINGS|DCX_CACHE); |
| #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 |
| drawrect.left = 0; |
| drawrect.right = This->surface_desc.dwWidth; |
| drawrect.top = 0; |
| drawrect.bottom = This->surface_desc.dwHeight; |
| |
| if (This->clipper) { |
| RECT xrc; |
| HWND hwnd = This->clipper->hWnd; |
| if (hwnd && GetWindowRect(hwnd,&xrc)) { |
| /* Do not forget to honor the offset within the clip window. */ |
| /* translate the surface to 0.0 of the clip window */ |
| OffsetRect(&drawrect,offset.x,offset.y); |
| IntersectRect(&drawrect,&drawrect,&xrc); |
| /* translate it back to its original position */ |
| OffsetRect(&drawrect,-offset.x,-offset.y); |
| } |
| } |
| if (rc) |
| IntersectRect(&drawrect,&drawrect,rc); |
| else { |
| /* Only use this if the caller did not pass a rectangle, since |
| * due to double locking this could be the wrong one ... */ |
| if (This->lastlockrect.left != This->lastlockrect.right) |
| IntersectRect(&drawrect,&drawrect,&This->lastlockrect); |
| } |
| BitBlt(hDisplayDC, |
| drawrect.left+offset.x, drawrect.top+offset.y, |
| drawrect.right-drawrect.left, drawrect.bottom-drawrect.top, |
| hSurfaceDC, |
| drawrect.left, drawrect.top, |
| SRCCOPY |
| ); |
| ReleaseDC(hDisplayWnd, hDisplayDC); |
| } |
| } |
| |
| static void User_copy_from_screen(IDirectDrawSurfaceImpl* This, LPCRECT rc) |
| { |
| if (This->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE) |
| { |
| POINT offset; |
| HWND hDisplayWnd = get_display_window(This, &offset); |
| HDC hDisplayDC = GetDC(hDisplayWnd); |
| RECT drawrect; |
| |
| drawrect.left = 0; |
| drawrect.right = This->surface_desc.dwWidth; |
| drawrect.top = 0; |
| drawrect.bottom = This->surface_desc.dwHeight; |
| if (rc) |
| IntersectRect(&drawrect,&drawrect,rc); |
| |
| BitBlt(This->hDC, |
| drawrect.left, drawrect.top, |
| drawrect.right-drawrect.left, drawrect.bottom-drawrect.top, |
| hDisplayDC, |
| drawrect.left+offset.x, drawrect.top+offset.y, |
| 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 |
| }; |