ddrawex: Implement a wrapper around IDirectDraw.
This allows us to cleanly implement the differences between ddraw.dll and
ddrawex.dll without private functions and if checks if an object was created
by ddraw or ddrawex. A similar wrapper will be added for IDirectDrawSurface.
diff --git a/dlls/ddrawex/ddraw.c b/dlls/ddrawex/ddraw.c
new file mode 100644
index 0000000..f3c5c41
--- /dev/null
+++ b/dlls/ddrawex/ddraw.c
@@ -0,0 +1,1484 @@
+/*
+ * Copyright 2008 Stefan Dösinger for CodeWeavers
+ *
+ * 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
+ */
+#include "wine/debug.h"
+
+#define COBJMACROS
+
+#include "winbase.h"
+#include "wingdi.h"
+
+#include "ddraw.h"
+#include "d3d.h"
+
+#include "ddrawex_private.h"
+
+WINE_DEFAULT_DEBUG_CHANNEL(ddrawex);
+
+/******************************************************************************
+ * Helper functions for COM management
+ ******************************************************************************/
+static IDirectDrawImpl *impl_from_dd1(IDirectDraw *iface)
+{
+ return (IDirectDrawImpl *)((char*)iface - FIELD_OFFSET(IDirectDrawImpl, IDirectDraw_Vtbl));
+}
+static IDirectDraw *dd1_from_impl(IDirectDrawImpl *This)
+{
+ return (IDirectDraw *) &This->IDirectDraw_Vtbl;
+}
+
+static IDirectDrawImpl *impl_from_dd2(IDirectDraw2 *iface)
+{
+ return (IDirectDrawImpl *)((char*)iface - FIELD_OFFSET(IDirectDrawImpl, IDirectDraw2_Vtbl));
+}
+static IDirectDraw2 *dd2_from_impl(IDirectDrawImpl *This)
+{
+ return (IDirectDraw2 *) &This->IDirectDraw2_Vtbl;
+}
+
+static IDirectDrawImpl *impl_from_dd3(IDirectDraw3 *iface)
+{
+ return (IDirectDrawImpl *)((char*)iface - FIELD_OFFSET(IDirectDrawImpl, IDirectDraw3_Vtbl));
+}
+static IDirectDraw3 *dd3_from_impl(IDirectDrawImpl *This)
+{
+ return (IDirectDraw3 *) &This->IDirectDraw3_Vtbl;
+}
+
+static IDirectDrawImpl *impl_from_dd4(IDirectDraw4 *iface)
+{
+ return (IDirectDrawImpl *)((char*)iface - FIELD_OFFSET(IDirectDrawImpl, IDirectDraw4_Vtbl));
+}
+static IDirectDraw4 *dd4_from_impl(IDirectDrawImpl *This)
+{
+ return (IDirectDraw4 *) &This->IDirectDraw4_Vtbl;
+}
+
+/******************************************************************************
+ * IDirectDraw4 -> ddraw.dll wrappers
+ ******************************************************************************/
+static HRESULT WINAPI
+IDirectDraw4Impl_QueryInterface(IDirectDraw4 *iface,
+ REFIID refiid,
+ void **obj)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+
+ TRACE("(%p)->(%s,%p)\n", This, debugstr_guid(refiid), obj);
+ *obj = NULL;
+
+ if(!refiid)
+ {
+ return DDERR_INVALIDPARAMS;
+ }
+
+ if (IsEqualGUID( &IID_IDirectDraw7, refiid ) )
+ {
+ WARN("IDirectDraw7 not allowed in ddrawex.dll\n");
+ return E_NOINTERFACE;
+ }
+ else if ( IsEqualGUID( &IID_IUnknown, refiid ) ||
+ IsEqualGUID( &IID_IDirectDraw4, refiid ) )
+ {
+ *obj = dd4_from_impl(This);
+ TRACE("(%p) Returning IDirectDraw4 interface at %p\n", This, *obj);
+ IDirectDraw4_AddRef((IDirectDraw4 *) *obj);
+ }
+ else if ( IsEqualGUID( &IID_IDirectDraw3, refiid ) )
+ {
+ *obj = dd3_from_impl(This);
+ TRACE("(%p) Returning IDirectDraw3 interface at %p\n", This, *obj);
+ IDirectDraw3_AddRef((IDirectDraw3 *) *obj);
+ }
+ else if ( IsEqualGUID( &IID_IDirectDraw2, refiid ) )
+ {
+ *obj = dd2_from_impl(This);
+ TRACE("(%p) Returning IDirectDraw2 interface at %p\n", This, *obj);
+ IDirectDraw2_AddRef((IDirectDraw2 *) *obj);
+ }
+ else if ( IsEqualGUID( &IID_IDirectDraw, refiid ) )
+ {
+ *obj = dd1_from_impl(This);
+ TRACE("(%p) Returning IDirectDraw interface at %p\n", This, *obj);
+ IDirectDraw_AddRef((IDirectDraw *) *obj);
+ }
+ else if ( IsEqualGUID( &IID_IDirect3D , refiid ) ||
+ IsEqualGUID( &IID_IDirect3D2 , refiid ) ||
+ IsEqualGUID( &IID_IDirect3D3 , refiid ) ||
+ IsEqualGUID( &IID_IDirect3D7 , refiid ) )
+ {
+ WARN("Direct3D not allowed in ddrawex.dll\n");
+ return E_NOINTERFACE;
+ }
+ /* Unknown interface */
+ else
+ {
+ ERR("(%p)->(%s, %p): No interface found\n", This, debugstr_guid(refiid), obj);
+ return E_NOINTERFACE;
+ }
+ TRACE("Returning S_OK\n");
+ return S_OK;
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_QueryInterface(IDirectDraw3 *iface,
+ REFIID refiid,
+ void **obj)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_QueryInterface(dd4_from_impl(This), refiid, obj);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_QueryInterface(IDirectDraw2 *iface,
+ REFIID refiid,
+ void **obj)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_QueryInterface(dd4_from_impl(This), refiid, obj);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_QueryInterface(IDirectDraw *iface,
+ REFIID refiid,
+ void **obj)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_QueryInterface(dd4_from_impl(This), refiid, obj);
+}
+
+static ULONG WINAPI
+IDirectDraw4Impl_AddRef(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ ULONG ref = InterlockedIncrement(&This->ref);
+
+ TRACE("(%p) : incrementing refcount from %u.\n", This, ref - 1);
+
+ return ref;
+}
+
+static ULONG WINAPI
+IDirectDraw3Impl_AddRef(IDirectDraw3 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_AddRef(dd4_from_impl(This));
+}
+
+static ULONG WINAPI
+IDirectDraw2Impl_AddRef(IDirectDraw2 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_AddRef(dd4_from_impl(This));
+}
+
+static ULONG WINAPI
+IDirectDrawImpl_AddRef(IDirectDraw *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_AddRef(dd4_from_impl(This));
+}
+
+static ULONG WINAPI
+IDirectDraw4Impl_Release(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ ULONG ref = InterlockedDecrement(&This->ref);
+
+ TRACE("(%p) : decrementing refcount to %u.\n", This, ref);
+
+ if(ref == 0)
+ {
+ TRACE("Destroying object\n");
+ IDirectDraw4_Release(This->parent);
+ HeapFree(GetProcessHeap(), 0, This);
+ }
+ return ref;
+}
+
+static ULONG WINAPI
+IDirectDraw3Impl_Release(IDirectDraw3 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_Release(dd4_from_impl(This));
+}
+
+static ULONG WINAPI
+IDirectDraw2Impl_Release(IDirectDraw2 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_Release(dd4_from_impl(This));
+}
+
+static ULONG WINAPI
+IDirectDrawImpl_Release(IDirectDraw *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_Release(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_Compact(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)\n", This);
+
+ return IDirectDraw4_Compact(This->parent);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_Compact(IDirectDraw3 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_Compact(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_Compact(IDirectDraw2 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_Compact(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_Compact(IDirectDraw *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_Compact(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_CreateClipper(IDirectDraw4 *iface,
+ DWORD Flags,
+ IDirectDrawClipper **clipper,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(0x%08x, %p, %p)\n", This, Flags, clipper, UnkOuter);
+
+ if(UnkOuter != NULL)
+ {
+ /* This may require a wrapper interface for clippers too which handles this */
+ FIXME("Test and implement Aggregation for ddrawex clippers\n");
+ }
+
+ return IDirectDraw4_CreateClipper(This->parent, Flags, clipper, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_CreateClipper(IDirectDraw3 *iface,
+ DWORD Flags,
+ IDirectDrawClipper **clipper,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_CreateClipper(dd4_from_impl(This), Flags, clipper, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_CreateClipper(IDirectDraw2 *iface,
+ DWORD Flags,
+ IDirectDrawClipper **clipper,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_CreateClipper(dd4_from_impl(This), Flags, clipper, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_CreateClipper(IDirectDraw *iface,
+ DWORD Flags,
+ IDirectDrawClipper **clipper,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_CreateClipper(dd4_from_impl(This), Flags, clipper, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_CreatePalette(IDirectDraw4 *iface,
+ DWORD Flags,
+ PALETTEENTRY *ColorTable,
+ IDirectDrawPalette **Palette,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)(0x%08x,%p,%p,%p)\n", This, Flags, ColorTable, Palette, UnkOuter);
+
+ if(UnkOuter != NULL)
+ {
+ /* This may require a wrapper interface for palettes too which handles this */
+ FIXME("Test and implement Aggregation for ddrawex palettes\n");
+ }
+
+ return IDirectDraw4_CreatePalette(This->parent, Flags, ColorTable, Palette, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_CreatePalette(IDirectDraw3 *iface,
+ DWORD Flags,
+ PALETTEENTRY *ColorTable,
+ IDirectDrawPalette **Palette,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_CreatePalette(dd4_from_impl(This), Flags, ColorTable, Palette, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_CreatePalette(IDirectDraw2 *iface,
+ DWORD Flags,
+ PALETTEENTRY *ColorTable,
+ IDirectDrawPalette **Palette,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_CreatePalette(dd4_from_impl(This), Flags, ColorTable, Palette, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_CreatePalette(IDirectDraw *iface,
+ DWORD Flags,
+ PALETTEENTRY *ColorTable,
+ IDirectDrawPalette **Palette,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw4\n");
+ return IDirectDraw4_CreatePalette(dd4_from_impl(This), Flags, ColorTable, Palette, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_CreateSurface(IDirectDraw4 *iface,
+ DDSURFACEDESC2 *DDSD,
+ IDirectDrawSurface4 **Surf,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)(%p, %p, %p)\n", This, DDSD, Surf, UnkOuter);
+
+ if(UnkOuter != NULL)
+ {
+ /* Handle this in this dll. Don't forward the UnkOuter to ddraw.dll */
+ FIXME("Implement aggregation for ddrawex surfaces\n");
+ }
+
+ return IDirectDraw4_CreateSurface(This->parent, DDSD, Surf, UnkOuter);
+}
+
+static void DDSD_to_DDSD2(const DDSURFACEDESC *in, DDSURFACEDESC2 *out)
+{
+ memset(out, 0, sizeof(*out));
+ out->dwSize = sizeof(*out);
+ out->dwFlags = in->dwFlags;
+ if(in->dwFlags & DDSD_WIDTH) out->dwWidth = in->dwWidth;
+ if(in->dwFlags & DDSD_HEIGHT) out->dwHeight = in->dwHeight;
+ if(in->dwFlags & DDSD_PIXELFORMAT) out->ddpfPixelFormat = in->ddpfPixelFormat;
+ if(in->dwFlags & DDSD_CAPS) out->ddsCaps.dwCaps = in->ddsCaps.dwCaps;
+ if(in->dwFlags & DDSD_PITCH) out->lPitch = in->lPitch;
+ if(in->dwFlags & DDSD_BACKBUFFERCOUNT) out->dwBackBufferCount = in->dwBackBufferCount;
+ if(in->dwFlags & DDSD_ZBUFFERBITDEPTH) out->dwMipMapCount = in->dwZBufferBitDepth; /* same union */
+ if(in->dwFlags & DDSD_ALPHABITDEPTH) out->dwAlphaBitDepth = in->dwAlphaBitDepth;
+ if(in->dwFlags & DDSD_LPSURFACE) out->lpSurface = in->lpSurface;
+ if(in->dwFlags & DDSD_CKDESTOVERLAY) out->ddckCKDestOverlay = in->ddckCKDestOverlay;
+ if(in->dwFlags & DDSD_CKDESTBLT) out->ddckCKDestBlt = in->ddckCKDestBlt;
+ if(in->dwFlags & DDSD_CKSRCOVERLAY) out->ddckCKSrcOverlay = in->ddckCKSrcOverlay;
+ if(in->dwFlags & DDSD_CKSRCBLT) out->ddckCKSrcBlt = in->ddckCKSrcBlt;
+ if(in->dwFlags & DDSD_MIPMAPCOUNT) out->dwMipMapCount = in->dwMipMapCount;
+ if(in->dwFlags & DDSD_REFRESHRATE) out->dwRefreshRate = in->dwRefreshRate;
+ if(in->dwFlags & DDSD_LINEARSIZE) out->dwLinearSize = in->dwLinearSize;
+ /* Does not exist in DDSURFACEDESC:
+ * DDSD_TEXTURESTAGE, DDSD_FVF, DDSD_SRCVBHANDLE,
+ */
+}
+
+static void DDSD2_to_DDSD(const DDSURFACEDESC2 *in, DDSURFACEDESC *out)
+{
+ memset(out, 0, sizeof(*out));
+ out->dwSize = sizeof(*out);
+ out->dwFlags = in->dwFlags;
+ if(in->dwFlags & DDSD_WIDTH) out->dwWidth = in->dwWidth;
+ if(in->dwFlags & DDSD_HEIGHT) out->dwHeight = in->dwHeight;
+ if(in->dwFlags & DDSD_PIXELFORMAT) out->ddpfPixelFormat = in->ddpfPixelFormat;
+ if(in->dwFlags & DDSD_CAPS) out->ddsCaps.dwCaps = in->ddsCaps.dwCaps;
+ if(in->dwFlags & DDSD_PITCH) out->lPitch = in->lPitch;
+ if(in->dwFlags & DDSD_BACKBUFFERCOUNT) out->dwBackBufferCount = in->dwBackBufferCount;
+ if(in->dwFlags & DDSD_ZBUFFERBITDEPTH) out->dwZBufferBitDepth = in->dwMipMapCount; /* same union */
+ if(in->dwFlags & DDSD_ALPHABITDEPTH) out->dwAlphaBitDepth = in->dwAlphaBitDepth;
+ if(in->dwFlags & DDSD_LPSURFACE) out->lpSurface = in->lpSurface;
+ if(in->dwFlags & DDSD_CKDESTOVERLAY) out->ddckCKDestOverlay = in->ddckCKDestOverlay;
+ if(in->dwFlags & DDSD_CKDESTBLT) out->ddckCKDestBlt = in->ddckCKDestBlt;
+ if(in->dwFlags & DDSD_CKSRCOVERLAY) out->ddckCKSrcOverlay = in->ddckCKSrcOverlay;
+ if(in->dwFlags & DDSD_CKSRCBLT) out->ddckCKSrcBlt = in->ddckCKSrcBlt;
+ if(in->dwFlags & DDSD_MIPMAPCOUNT) out->dwMipMapCount = in->dwMipMapCount;
+ if(in->dwFlags & DDSD_REFRESHRATE) out->dwRefreshRate = in->dwRefreshRate;
+ if(in->dwFlags & DDSD_LINEARSIZE) out->dwLinearSize = in->dwLinearSize;
+ /* Does not exist in DDSURFACEDESC:
+ * DDSD_TEXTURESTAGE, DDSD_FVF, DDSD_SRCVBHANDLE,
+ */
+ if(in->dwFlags & DDSD_TEXTURESTAGE) WARN("Does not exist in DDSURFACEDESC: DDSD_TEXTURESTAGE\n");
+ if(in->dwFlags & DDSD_FVF) WARN("Does not exist in DDSURFACEDESC: DDSD_FVF\n");
+ if(in->dwFlags & DDSD_SRCVBHANDLE) WARN("Does not exist in DDSURFACEDESC: DDSD_SRCVBHANDLE\n");
+ out->dwFlags &= ~(DDSD_TEXTURESTAGE | DDSD_FVF | DDSD_SRCVBHANDLE);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_CreateSurface(IDirectDraw3 *iface,
+ DDSURFACEDESC *DDSD,
+ IDirectDrawSurface **Surf,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ DDSURFACEDESC2 ddsd2;
+ IDirectDrawSurface4 *surf4 = NULL;
+ HRESULT hr;
+ TRACE("Thunking to IDirectDraw4\n");
+
+ DDSD_to_DDSD2(DDSD, &ddsd2);
+
+ hr = IDirectDraw4_CreateSurface(dd4_from_impl(This), &ddsd2, &surf4, UnkOuter);
+ if(FAILED(hr))
+ {
+ *Surf = NULL;
+ return hr;
+ }
+
+ IDirectDrawSurface4_QueryInterface(surf4, &IID_IDirectDrawSurface, (void **) Surf);
+ IDirectDrawSurface4_Release(surf4);
+ return hr;
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_CreateSurface(IDirectDraw2 *iface,
+ DDSURFACEDESC *DDSD,
+ IDirectDrawSurface **Surf,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw3\n");
+ return IDirectDraw3_CreateSurface(dd3_from_impl(This), DDSD, Surf, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_CreateSurface(IDirectDraw *iface,
+ DDSURFACEDESC *DDSD,
+ IDirectDrawSurface **Surf,
+ IUnknown *UnkOuter)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw3\n");
+ return IDirectDraw3_CreateSurface(dd3_from_impl(This), DDSD, Surf, UnkOuter);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_DuplicateSurface(IDirectDraw4 *iface,
+ IDirectDrawSurface4 *src,
+ IDirectDrawSurface4 **dst)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p,%p)\n", This, src, dst);
+
+ return IDirectDraw4_DuplicateSurface(This->parent, src, dst);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_DuplicateSurface(IDirectDraw3 *iface,
+ IDirectDrawSurface *src,
+ IDirectDrawSurface **dst)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ IDirectDrawSurface4 *src_4;
+ IDirectDrawSurface4 *dst_4;
+ HRESULT hr;
+
+ TRACE("Thunking to IDirectDraw4\n");
+ IDirectDrawSurface_QueryInterface(src, &IID_IDirectDrawSurface4, (void **) &src_4);
+ hr = IDirectDraw4_DuplicateSurface(dd4_from_impl(This), src_4, &dst_4);
+ IDirectDrawSurface4_Release(src_4);
+
+ if(FAILED(hr))
+ {
+ *dst = NULL;
+ return hr;
+ }
+ IDirectDrawSurface4_QueryInterface(dst_4, &IID_IDirectDrawSurface, (void **) dst);
+ IDirectDrawSurface4_Release(dst_4);
+ return hr;
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_DuplicateSurface(IDirectDraw2 *iface,
+ IDirectDrawSurface *src,
+ IDirectDrawSurface **dst)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("Thunking to IDirectDraw3\n");
+ return IDirectDraw3_DuplicateSurface(dd3_from_impl(This), src, dst);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_DuplicateSurface(IDirectDraw *iface,
+ IDirectDrawSurface *src,
+ IDirectDrawSurface **dst)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("Thunking to IDirectDraw3\n");
+ return IDirectDraw3_DuplicateSurface(dd3_from_impl(This), src, dst);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_EnumDisplayModes(IDirectDraw4 *iface,
+ DWORD Flags,
+ DDSURFACEDESC2 *DDSD,
+ void *Context,
+ LPDDENUMMODESCALLBACK2 cb)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(0x%08x,%p,%p,%p)\n", This, Flags, DDSD, Context, cb);
+
+ return IDirectDraw4_EnumDisplayModes(This->parent, Flags, DDSD, Context, cb);
+}
+
+struct enummodes_ctx
+{
+ LPDDENUMMODESCALLBACK orig_cb;
+ void *orig_ctx;
+};
+
+static HRESULT WINAPI
+enum_modes_cb2(DDSURFACEDESC2 *ddsd2, void *vctx)
+{
+ struct enummodes_ctx *ctx = (struct enummodes_ctx *) vctx;
+ DDSURFACEDESC ddsd;
+
+ DDSD2_to_DDSD(ddsd2, &ddsd);
+ return ctx->orig_cb(&ddsd, ctx->orig_ctx);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_EnumDisplayModes(IDirectDraw3 *iface,
+ DWORD Flags,
+ DDSURFACEDESC *DDSD,
+ void *Context,
+ LPDDENUMMODESCALLBACK cb)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ DDSURFACEDESC2 ddsd2;
+ struct enummodes_ctx ctx;
+ TRACE("(%p)->(0x%08x,%p,%p,%p): Thunking to IDirectDraw4\n", This, Flags, DDSD, Context, cb);
+
+ DDSD_to_DDSD2(DDSD, &ddsd2);
+ ctx.orig_cb = cb;
+ ctx.orig_ctx = Context;
+ return IDirectDraw4_EnumDisplayModes(dd4_from_impl(This), Flags, &ddsd2, &ctx, enum_modes_cb2);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_EnumDisplayModes(IDirectDraw2 *iface,
+ DWORD Flags,
+ DDSURFACEDESC *DDSD,
+ void *Context,
+ LPDDENUMMODESCALLBACK cb)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(0x%08x,%p,%p,%p): Thunking to IDirectDraw3\n", This, Flags, DDSD, Context, cb);
+ return IDirectDraw3_EnumDisplayModes(dd3_from_impl(This), Flags, DDSD, Context, cb);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_EnumDisplayModes(IDirectDraw *iface,
+ DWORD Flags,
+ DDSURFACEDESC *DDSD,
+ void *Context,
+ LPDDENUMMODESCALLBACK cb)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(0x%08x,%p,%p,%p): Thunking to IDirectDraw3\n", This, Flags, DDSD, Context, cb);
+ return IDirectDraw3_EnumDisplayModes(dd3_from_impl(This), Flags, DDSD, Context, cb);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_EnumSurfaces(IDirectDraw4 *iface,
+ DWORD Flags,
+ DDSURFACEDESC2 *DDSD,
+ void *Context,
+ LPDDENUMSURFACESCALLBACK2 Callback)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(0x%08x,%p,%p,%p)\n", This, Flags, DDSD, Context, Callback);
+
+ return IDirectDraw4Impl_EnumSurfaces(This->parent, Flags, DDSD, Context, Callback);
+}
+
+struct enumsurfaces_ctx
+{
+ LPDDENUMSURFACESCALLBACK orig_cb;
+ void *orig_ctx;
+};
+
+static HRESULT WINAPI
+enum_surfaces_cb2(IDirectDrawSurface4 *surf4, DDSURFACEDESC2 *ddsd2, void *vctx)
+{
+ struct enumsurfaces_ctx *ctx = (struct enumsurfaces_ctx *) vctx;
+ IDirectDrawSurface *surf1;
+ DDSURFACEDESC ddsd;
+
+ /* Keep the reference, it goes to the application */
+ IDirectDrawSurface4_QueryInterface(surf4, &IID_IDirectDrawSurface, (void **) &surf1);
+ /* Release the reference this function got */
+ IDirectDrawSurface4_Release(surf4);
+
+ DDSD2_to_DDSD(ddsd2, &ddsd);
+ return ctx->orig_cb(surf1, &ddsd, ctx->orig_ctx);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_EnumSurfaces(IDirectDraw3 *iface,
+ DWORD Flags,
+ DDSURFACEDESC *DDSD,
+ void *Context,
+ LPDDENUMSURFACESCALLBACK Callback)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ DDSURFACEDESC2 ddsd2;
+ struct enumsurfaces_ctx ctx;
+ TRACE("(%p)->(0x%08x,%p,%p,%p): Thunking to IDirectDraw4\n", This, Flags, DDSD, Context, Callback);
+
+ DDSD_to_DDSD2(DDSD, &ddsd2);
+ ctx.orig_cb = Callback;
+ ctx.orig_ctx = Context;
+ return IDirectDraw4_EnumSurfaces(dd4_from_impl(This), Flags, &ddsd2, &ctx, enum_surfaces_cb2);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_EnumSurfaces(IDirectDraw2 *iface,
+ DWORD Flags,
+ DDSURFACEDESC *DDSD,
+ void *Context,
+ LPDDENUMSURFACESCALLBACK Callback)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(0x%08x,%p,%p,%p): Thunking to IDirectDraw3\n", This, Flags, DDSD, Context, Callback);
+ return IDirectDraw3_EnumSurfaces(dd3_from_impl(This), Flags, DDSD, Context, Callback);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_EnumSurfaces(IDirectDraw *iface,
+ DWORD Flags,
+ DDSURFACEDESC *DDSD,
+ void *Context,
+ LPDDENUMSURFACESCALLBACK Callback)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(0x%08x,%p,%p,%p): Thunking to IDirectDraw3\n", This, Flags, DDSD, Context, Callback);
+ return IDirectDraw3_EnumSurfaces(dd3_from_impl(This), Flags, DDSD, Context, Callback);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_FlipToGDISurface(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)\n", This);
+
+ return IDirectDraw4_FlipToGDISurface(This->parent);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_FlipToGDISurface(IDirectDraw3 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p). Thunking to IDirectDraw4\n", This);
+ return IDirectDraw4_FlipToGDISurface(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_FlipToGDISurface(IDirectDraw2 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p). Thunking to IDirectDraw4\n", This);
+ return IDirectDraw4_FlipToGDISurface(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_FlipToGDISurface(IDirectDraw *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p). Thunking to IDirectDraw4\n", This);
+ return IDirectDraw4_FlipToGDISurface(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetCaps(IDirectDraw4 *iface,
+ DDCAPS *DriverCaps,
+ DDCAPS *HELCaps)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p,%p)\n", This, DriverCaps, HELCaps);
+ return IDirectDraw4_GetCaps(This->parent, DriverCaps, HELCaps);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetCaps(IDirectDraw3 *iface,
+ DDCAPS *DriverCaps,
+ DDCAPS *HELCaps)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%p,%p). Thunking to IDirectDraw4\n", This, DriverCaps, HELCaps);
+ return IDirectDraw4_GetCaps(dd4_from_impl(This), DriverCaps, HELCaps);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetCaps(IDirectDraw2 *iface,
+ DDCAPS *DriverCaps,
+ DDCAPS *HELCaps)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p,%p). Thunking to IDirectDraw4\n", This, DriverCaps, HELCaps);
+ return IDirectDraw4_GetCaps(dd4_from_impl(This), DriverCaps, HELCaps);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetCaps(IDirectDraw *iface,
+ DDCAPS *DriverCaps,
+ DDCAPS *HELCaps)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p,%p). Thunking to IDirectDraw4\n", This, DriverCaps, HELCaps);
+ return IDirectDraw4_GetCaps(dd4_from_impl(This), DriverCaps, HELCaps);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetDisplayMode(IDirectDraw4 *iface,
+ DDSURFACEDESC2 *DDSD)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p)\n", This, DDSD);
+ return IDirectDraw4_GetDisplayMode(This->parent, DDSD);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetDisplayMode(IDirectDraw3 *iface,
+ DDSURFACEDESC *DDSD)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ DDSURFACEDESC2 ddsd2;
+ HRESULT hr;
+
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, DDSD);
+ hr = IDirectDraw4_GetDisplayMode(dd4_from_impl(This), &ddsd2);
+ DDSD2_to_DDSD(&ddsd2, DDSD);
+ return hr;
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetDisplayMode(IDirectDraw2 *iface,
+ DDSURFACEDESC *DDSD)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw3\n", This, DDSD);
+ return IDirectDraw3_GetDisplayMode(dd3_from_impl(This), DDSD);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetDisplayMode(IDirectDraw *iface,
+ DDSURFACEDESC *DDSD)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw3\n", This, DDSD);
+ return IDirectDraw3_GetDisplayMode(dd3_from_impl(This), DDSD);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetFourCCCodes(IDirectDraw4 *iface,
+ DWORD *NumCodes,
+ DWORD *Codes)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p, %p):\n", This, NumCodes, Codes);
+ return IDirectDraw4_GetFourCCCodes(This->parent, NumCodes, Codes);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetFourCCCodes(IDirectDraw3 *iface,
+ DWORD *NumCodes,
+ DWORD *Codes)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%p, %p): Thunking to IDirectDraw4\n", This, NumCodes, Codes);
+ return IDirectDraw4_GetFourCCCodes(dd4_from_impl(This), NumCodes, Codes);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetFourCCCodes(IDirectDraw2 *iface,
+ DWORD *NumCodes,
+ DWORD *Codes)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p, %p): Thunking to IDirectDraw4\n", This, NumCodes, Codes);
+ return IDirectDraw4_GetFourCCCodes(dd4_from_impl(This), NumCodes, Codes);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetFourCCCodes(IDirectDraw *iface,
+ DWORD *NumCodes,
+ DWORD *Codes)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p, %p): Thunking to IDirectDraw4\n", This, NumCodes, Codes);
+ return IDirectDraw4_GetFourCCCodes(dd4_from_impl(This), NumCodes, Codes);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetGDISurface(IDirectDraw4 *iface,
+ IDirectDrawSurface4 **GDISurface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p)\n", This, GDISurface);
+ return IDirectDraw4_GetGDISurface(This->parent, GDISurface);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetGDISurface(IDirectDraw3 *iface,
+ IDirectDrawSurface **GDISurface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ IDirectDrawSurface4 *surf4;
+ HRESULT hr;
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, GDISurface);
+
+ hr = IDirectDraw4_GetGDISurface(dd4_from_impl(This), &surf4);
+ if(FAILED(hr)) {
+ *GDISurface = NULL;
+ return hr;
+ }
+
+ /* Release the reference we got from the DDraw4 call, and pass a reference to the caller */
+ IDirectDrawSurface4_QueryInterface(surf4, &IID_IDirectDrawSurface, (void **) GDISurface);
+ IDirectDrawSurface4_Release(surf4);
+ return hr;
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetGDISurface(IDirectDraw2 *iface,
+ IDirectDrawSurface **GDISurface)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw3\n", This, GDISurface);
+ return IDirectDraw3_GetGDISurface(dd3_from_impl(This), GDISurface);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetGDISurface(IDirectDraw *iface,
+ IDirectDrawSurface **GDISurface)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw3\n", This, GDISurface);
+ return IDirectDraw3_GetGDISurface(dd3_from_impl(This), GDISurface);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetMonitorFrequency(IDirectDraw4 *iface,
+ DWORD *Freq)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p)\n", This, Freq);
+ return IDirectDraw4_GetMonitorFrequency(This->parent, Freq);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetMonitorFrequency(IDirectDraw3 *iface,
+ DWORD *Freq)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, Freq);
+ return IDirectDraw4_GetMonitorFrequency(dd4_from_impl(This), Freq);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetMonitorFrequency(IDirectDraw2 *iface,
+ DWORD *Freq)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, Freq);
+ return IDirectDraw4_GetMonitorFrequency(dd4_from_impl(This), Freq);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetMonitorFrequency(IDirectDraw *iface,
+ DWORD *Freq)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, Freq);
+ return IDirectDraw4_GetMonitorFrequency(dd4_from_impl(This), Freq);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetScanLine(IDirectDraw4 *iface,
+ DWORD *Scanline)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p)\n", This, Scanline);
+ return IDirectDraw4_GetScanLine(This->parent, Scanline);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetScanLine(IDirectDraw3 *iface,
+ DWORD *Scanline)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, Scanline);
+ return IDirectDraw4_GetScanLine(dd4_from_impl(This), Scanline);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetScanLine(IDirectDraw2 *iface,
+ DWORD *Scanline)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, Scanline);
+ return IDirectDraw4_GetScanLine(dd4_from_impl(This), Scanline);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetScanLine(IDirectDraw *iface,
+ DWORD *Scanline)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, Scanline);
+ return IDirectDraw4_GetScanLine(dd4_from_impl(This), Scanline);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetVerticalBlankStatus(IDirectDraw4 *iface,
+ BOOL *status)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p)\n", This, status);
+ return IDirectDraw4_GetVerticalBlankStatus(This->parent, status);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetVerticalBlankStatus(IDirectDraw3 *iface,
+ BOOL *status)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, status);
+ return IDirectDraw4_GetVerticalBlankStatus(dd4_from_impl(This), status);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetVerticalBlankStatus(IDirectDraw2 *iface,
+ BOOL *status)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, status);
+ return IDirectDraw4_GetVerticalBlankStatus(dd4_from_impl(This), status);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_GetVerticalBlankStatus(IDirectDraw *iface,
+ BOOL *status)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p): Thunking to IDirectDraw4\n", This, status);
+ return IDirectDraw4_GetVerticalBlankStatus(dd4_from_impl(This), status);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_Initialize(IDirectDraw4 *iface,
+ GUID *Guid)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%s)\n", This, debugstr_guid(Guid));
+ return IDirectDraw4_Initialize(This->parent, Guid);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_Initialize(IDirectDraw3 *iface,
+ GUID *Guid)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%s): Thunking to IDirectDraw4\n", This, debugstr_guid(Guid));
+ return IDirectDraw4_Initialize(dd4_from_impl(This), Guid);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_Initialize(IDirectDraw2 *iface,
+ GUID *Guid)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%s): Thunking to IDirectDraw4\n", This, debugstr_guid(Guid));
+ return IDirectDraw4_Initialize(dd4_from_impl(This), Guid);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_Initialize(IDirectDraw *iface,
+ GUID *Guid)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%s): Thunking to IDirectDraw4\n", This, debugstr_guid(Guid));
+ return IDirectDraw4_Initialize(dd4_from_impl(This), Guid);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_RestoreDisplayMode(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)\n", This);
+ return IDirectDraw4_RestoreDisplayMode(This->parent);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_RestoreDisplayMode(IDirectDraw3 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p): Thunking to IDirectDraw4\n", This);
+ return IDirectDraw4_RestoreDisplayMode(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_RestoreDisplayMode(IDirectDraw2 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p): Thunking to IDirectDraw4\n", This);
+ return IDirectDraw4_RestoreDisplayMode(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_RestoreDisplayMode(IDirectDraw *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p): Thunking to IDirectDraw4\n", This);
+ return IDirectDraw4_RestoreDisplayMode(dd4_from_impl(This));
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_SetCooperativeLevel(IDirectDraw4 *iface,
+ HWND hwnd,
+ DWORD cooplevel)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p, 0x%08x)\n", This, hwnd, cooplevel);
+ return IDirectDraw4_SetCooperativeLevel(This->parent, hwnd, cooplevel);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_SetCooperativeLevel(IDirectDraw3 *iface,
+ HWND hwnd,
+ DWORD cooplevel)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%p, 0x%08x): Thunking to IDirectDraw4\n", This, hwnd, cooplevel);
+ return IDirectDraw4_SetCooperativeLevel(dd4_from_impl(This), hwnd, cooplevel);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_SetCooperativeLevel(IDirectDraw2 *iface,
+ HWND hwnd,
+ DWORD cooplevel)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%p, 0x%08x): Thunking to IDirectDraw4\n", This, hwnd, cooplevel);
+ return IDirectDraw4_SetCooperativeLevel(dd4_from_impl(This), hwnd, cooplevel);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_SetCooperativeLevel(IDirectDraw *iface,
+ HWND hwnd,
+ DWORD cooplevel)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%p, 0x%08x): Thunking to IDirectDraw4\n", This, hwnd, cooplevel);
+ return IDirectDraw4_SetCooperativeLevel(dd4_from_impl(This), hwnd, cooplevel);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_SetDisplayMode(IDirectDraw4 *iface,
+ DWORD Width,
+ DWORD Height,
+ DWORD BPP,
+ DWORD RefreshRate,
+ DWORD Flags)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%u, %u, %u, %u, 0x%08x)\n", This, Width, Height, BPP, RefreshRate, Flags);
+ return IDirectDraw4_SetDisplayMode(This->parent, Width, Height, BPP, RefreshRate, Flags);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_SetDisplayMode(IDirectDraw3 *iface,
+ DWORD Width,
+ DWORD Height,
+ DWORD BPP,
+ DWORD RefreshRate,
+ DWORD Flags)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(%u, %u, %u, %u, 0x%08x): Thunking to IDirectDraw4\n", This, Width, Height, BPP, RefreshRate, Flags);
+ return IDirectDraw3_SetDisplayMode(dd4_from_impl(This), Width, Height, BPP, RefreshRate, Flags);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_SetDisplayMode(IDirectDraw2 *iface,
+ DWORD Width,
+ DWORD Height,
+ DWORD BPP,
+ DWORD RefreshRate,
+ DWORD Flags)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(%u, %u, %u, %u, 0x%08x): Thunking to IDirectDraw4\n", This, Width, Height, BPP, RefreshRate, Flags);
+ return IDirectDraw3_SetDisplayMode(dd4_from_impl(This), Width, Height, BPP, RefreshRate, Flags);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_SetDisplayMode(IDirectDraw *iface,
+ DWORD Width,
+ DWORD Height,
+ DWORD BPP)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(%u, %u, %u): Thunking to IDirectDraw4\n", This, Width, Height, BPP);
+ return IDirectDraw3_SetDisplayMode(dd4_from_impl(This), Width, Height, BPP, 0, 0);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_WaitForVerticalBlank(IDirectDraw4 *iface,
+ DWORD Flags,
+ HANDLE h)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(0x%08x, %p)\n", This, Flags, h);
+ return IDirectDraw4_WaitForVerticalBlank(This->parent, Flags, h);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_WaitForVerticalBlank(IDirectDraw3 *iface,
+ DWORD Flags,
+ HANDLE h)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ TRACE("(%p)->(0x%08x, %p): Thunking to IDirectDraw4\n", This, Flags, h);
+ return IDirectDraw4_WaitForVerticalBlank(dd4_from_impl(This), Flags, h);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_WaitForVerticalBlank(IDirectDraw2 *iface,
+ DWORD Flags,
+ HANDLE h)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ TRACE("(%p)->(0x%08x, %p): Thunking to IDirectDraw4\n", This, Flags, h);
+ return IDirectDraw4_WaitForVerticalBlank(dd4_from_impl(This), Flags, h);
+}
+
+static HRESULT WINAPI
+IDirectDrawImpl_WaitForVerticalBlank(IDirectDraw *iface,
+ DWORD Flags,
+ HANDLE h)
+{
+ IDirectDrawImpl *This = impl_from_dd1(iface);
+ TRACE("(%p)->(0x%08x, %p): Thunking to IDirectDraw4\n", This, Flags, h);
+ return IDirectDraw4_WaitForVerticalBlank(dd4_from_impl(This), Flags, h);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetAvailableVidMem(IDirectDraw4 *iface,
+ DDSCAPS2 *Caps,
+ DWORD *total,
+ DWORD *free)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p, %p, %p)\n", This, Caps, total, free);
+ return IDirectDraw4_GetAvailableVidMem(This->parent, Caps, total, free);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetAvailableVidMem(IDirectDraw3 *iface,
+ DDSCAPS *Caps,
+ DWORD *total,
+ DWORD *free)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ DDSCAPS2 caps2;
+ TRACE("(%p)->(%p, %p, %p): Thunking to IDirectDraw4\n", This, Caps, total, free);
+ memset(&caps2, 0, sizeof(caps2));
+ caps2.dwCaps = Caps->dwCaps;
+ return IDirectDraw4_GetAvailableVidMem(dd4_from_impl(This), &caps2, total, free);
+}
+
+static HRESULT WINAPI
+IDirectDraw2Impl_GetAvailableVidMem(IDirectDraw2 *iface,
+ DDSCAPS *Caps,
+ DWORD *total,
+ DWORD *free)
+{
+ IDirectDrawImpl *This = impl_from_dd2(iface);
+ DDSCAPS2 caps2;
+ TRACE("(%p)->(%p, %p, %p): Thunking to IDirectDraw4\n", This, Caps, total, free);
+ memset(&caps2, 0, sizeof(caps2));
+ caps2.dwCaps = Caps->dwCaps;
+ return IDirectDraw4_GetAvailableVidMem(dd4_from_impl(This), &caps2, total, free);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetSurfaceFromDC(IDirectDraw4 *iface,
+ HDC hdc,
+ IDirectDrawSurface4 **Surface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p, %p)\n", This, hdc, Surface);
+ return IDirectDraw4_GetSurfaceFromDC(This->parent,hdc, Surface);
+}
+
+static HRESULT WINAPI
+IDirectDraw3Impl_GetSurfaceFromDC(IDirectDraw3 *iface,
+ HDC hdc,
+ IDirectDrawSurface **Surface)
+{
+ IDirectDrawImpl *This = impl_from_dd3(iface);
+ IDirectDrawSurface4 *surf4;
+ HRESULT hr;
+ TRACE("(%p)->(%p, %p): Thunking to IDirectDraw4\n", This, hdc, Surface);
+
+ hr = IDirectDraw4_GetSurfaceFromDC(dd4_from_impl(This), hdc, &surf4);
+ if(FAILED(hr))
+ {
+ *Surface = NULL;
+ return hr;
+ }
+ IDirectDrawSurface4_QueryInterface(surf4, &IID_IDirectDrawSurface, (void **) Surface);
+ IDirectDrawSurface4_Release(surf4);
+ return hr;
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_RestoreAllSurfaces(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)\n", This);
+ return IDirectDraw4_RestoreAllSurfaces(This->parent);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_TestCooperativeLevel(IDirectDraw4 *iface)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)\n", This);
+ return IDirectDraw4_TestCooperativeLevel(This->parent);
+}
+
+static HRESULT WINAPI
+IDirectDraw4Impl_GetDeviceIdentifier(IDirectDraw4 *iface,
+ DDDEVICEIDENTIFIER *DDDI,
+ DWORD Flags)
+{
+ IDirectDrawImpl *This = impl_from_dd4(iface);
+ TRACE("(%p)->(%p,0x%08x)\n", This, DDDI, Flags);
+ return IDirectDraw4_GetDeviceIdentifier(This->parent, DDDI, Flags);
+}
+
+static const IDirectDrawVtbl IDirectDraw1_Vtbl =
+{
+ IDirectDrawImpl_QueryInterface,
+ IDirectDrawImpl_AddRef,
+ IDirectDrawImpl_Release,
+ IDirectDrawImpl_Compact,
+ IDirectDrawImpl_CreateClipper,
+ IDirectDrawImpl_CreatePalette,
+ IDirectDrawImpl_CreateSurface,
+ IDirectDrawImpl_DuplicateSurface,
+ IDirectDrawImpl_EnumDisplayModes,
+ IDirectDrawImpl_EnumSurfaces,
+ IDirectDrawImpl_FlipToGDISurface,
+ IDirectDrawImpl_GetCaps,
+ IDirectDrawImpl_GetDisplayMode,
+ IDirectDrawImpl_GetFourCCCodes,
+ IDirectDrawImpl_GetGDISurface,
+ IDirectDrawImpl_GetMonitorFrequency,
+ IDirectDrawImpl_GetScanLine,
+ IDirectDrawImpl_GetVerticalBlankStatus,
+ IDirectDrawImpl_Initialize,
+ IDirectDrawImpl_RestoreDisplayMode,
+ IDirectDrawImpl_SetCooperativeLevel,
+ IDirectDrawImpl_SetDisplayMode,
+ IDirectDrawImpl_WaitForVerticalBlank,
+};
+
+static const IDirectDraw2Vtbl IDirectDraw2_Vtbl =
+{
+ IDirectDraw2Impl_QueryInterface,
+ IDirectDraw2Impl_AddRef,
+ IDirectDraw2Impl_Release,
+ IDirectDraw2Impl_Compact,
+ IDirectDraw2Impl_CreateClipper,
+ IDirectDraw2Impl_CreatePalette,
+ IDirectDraw2Impl_CreateSurface,
+ IDirectDraw2Impl_DuplicateSurface,
+ IDirectDraw2Impl_EnumDisplayModes,
+ IDirectDraw2Impl_EnumSurfaces,
+ IDirectDraw2Impl_FlipToGDISurface,
+ IDirectDraw2Impl_GetCaps,
+ IDirectDraw2Impl_GetDisplayMode,
+ IDirectDraw2Impl_GetFourCCCodes,
+ IDirectDraw2Impl_GetGDISurface,
+ IDirectDraw2Impl_GetMonitorFrequency,
+ IDirectDraw2Impl_GetScanLine,
+ IDirectDraw2Impl_GetVerticalBlankStatus,
+ IDirectDraw2Impl_Initialize,
+ IDirectDraw2Impl_RestoreDisplayMode,
+ IDirectDraw2Impl_SetCooperativeLevel,
+ IDirectDraw2Impl_SetDisplayMode,
+ IDirectDraw2Impl_WaitForVerticalBlank,
+ IDirectDraw2Impl_GetAvailableVidMem
+};
+
+static const IDirectDraw3Vtbl IDirectDraw3_Vtbl =
+{
+ IDirectDraw3Impl_QueryInterface,
+ IDirectDraw3Impl_AddRef,
+ IDirectDraw3Impl_Release,
+ IDirectDraw3Impl_Compact,
+ IDirectDraw3Impl_CreateClipper,
+ IDirectDraw3Impl_CreatePalette,
+ IDirectDraw3Impl_CreateSurface,
+ IDirectDraw3Impl_DuplicateSurface,
+ IDirectDraw3Impl_EnumDisplayModes,
+ IDirectDraw3Impl_EnumSurfaces,
+ IDirectDraw3Impl_FlipToGDISurface,
+ IDirectDraw3Impl_GetCaps,
+ IDirectDraw3Impl_GetDisplayMode,
+ IDirectDraw3Impl_GetFourCCCodes,
+ IDirectDraw3Impl_GetGDISurface,
+ IDirectDraw3Impl_GetMonitorFrequency,
+ IDirectDraw3Impl_GetScanLine,
+ IDirectDraw3Impl_GetVerticalBlankStatus,
+ IDirectDraw3Impl_Initialize,
+ IDirectDraw3Impl_RestoreDisplayMode,
+ IDirectDraw3Impl_SetCooperativeLevel,
+ IDirectDraw3Impl_SetDisplayMode,
+ IDirectDraw3Impl_WaitForVerticalBlank,
+ IDirectDraw3Impl_GetAvailableVidMem,
+ IDirectDraw3Impl_GetSurfaceFromDC,
+};
+
+static const IDirectDraw4Vtbl IDirectDraw4_Vtbl =
+{
+ IDirectDraw4Impl_QueryInterface,
+ IDirectDraw4Impl_AddRef,
+ IDirectDraw4Impl_Release,
+ IDirectDraw4Impl_Compact,
+ IDirectDraw4Impl_CreateClipper,
+ IDirectDraw4Impl_CreatePalette,
+ IDirectDraw4Impl_CreateSurface,
+ IDirectDraw4Impl_DuplicateSurface,
+ IDirectDraw4Impl_EnumDisplayModes,
+ IDirectDraw4Impl_EnumSurfaces,
+ IDirectDraw4Impl_FlipToGDISurface,
+ IDirectDraw4Impl_GetCaps,
+ IDirectDraw4Impl_GetDisplayMode,
+ IDirectDraw4Impl_GetFourCCCodes,
+ IDirectDraw4Impl_GetGDISurface,
+ IDirectDraw4Impl_GetMonitorFrequency,
+ IDirectDraw4Impl_GetScanLine,
+ IDirectDraw4Impl_GetVerticalBlankStatus,
+ IDirectDraw4Impl_Initialize,
+ IDirectDraw4Impl_RestoreDisplayMode,
+ IDirectDraw4Impl_SetCooperativeLevel,
+ IDirectDraw4Impl_SetDisplayMode,
+ IDirectDraw4Impl_WaitForVerticalBlank,
+ IDirectDraw4Impl_GetAvailableVidMem,
+ IDirectDraw4Impl_GetSurfaceFromDC,
+ IDirectDraw4Impl_RestoreAllSurfaces,
+ IDirectDraw4Impl_TestCooperativeLevel,
+ IDirectDraw4Impl_GetDeviceIdentifier
+};
+
+/*******************************************************************************
+ * IDirectDrawFactoryImpl_CreateDirectDraw
+ *******************************************************************************/
+HRESULT WINAPI
+IDirectDrawFactoryImpl_CreateDirectDraw(IDirectDrawFactory* iface,
+ GUID * pGUID,
+ HWND hWnd,
+ DWORD dwCoopLevelFlags,
+ DWORD dwReserved,
+ IUnknown *pUnkOuter,
+ IDirectDraw **ppDirectDraw)
+{
+ HRESULT hr;
+ IDirectDrawImpl *object = NULL;
+ IDirectDraw *parent = NULL;
+
+ TRACE("(%p)->(%s,%p,0x%08x,0x%08x,%p,%p)", iface, debugstr_guid(pGUID), hWnd, dwCoopLevelFlags,
+ dwReserved, pUnkOuter, ppDirectDraw);
+
+ if(pUnkOuter)
+ {
+ FIXME("Implement aggregation in ddrawex's IDirectDraw interface\n");
+ }
+
+ object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*object));
+ if(!object)
+ {
+ ERR("Out of memory\n");
+ hr = E_OUTOFMEMORY;
+ goto err;
+ }
+ object->ref = 1;
+ object->IDirectDraw_Vtbl = &IDirectDraw1_Vtbl;
+ object->IDirectDraw2_Vtbl = &IDirectDraw2_Vtbl;
+ object->IDirectDraw3_Vtbl = &IDirectDraw3_Vtbl;
+ object->IDirectDraw4_Vtbl = &IDirectDraw4_Vtbl;
+
+ hr = DirectDrawCreate(pGUID, &parent, NULL);
+ if (FAILED(hr)) goto err;
+
+ hr = IDirectDraw_QueryInterface(parent, &IID_IDirectDraw4, (void **) &object->parent);
+ if(FAILED(hr)) goto err;
+
+ hr = IDirectDraw_SetCooperativeLevel(dd1_from_impl(object), hWnd, dwCoopLevelFlags);
+ if (FAILED(hr)) goto err;
+
+ *ppDirectDraw = dd1_from_impl(object);
+ IDirectDraw_Release(parent);
+ return DD_OK;
+
+err:
+ if(object && object->parent) IDirectDraw4_Release(object->parent);
+ if(parent) IDirectDraw_Release(parent);
+ if(object) HeapFree(GetProcessHeap(), 0, object);
+ *ppDirectDraw = NULL;
+ return hr;
+}