ddraw: Separate IDirectDrawSurface2 reference count.
[wine/multimedia.git] / dlls / ddraw / surface.c
blob62047036e33d285ebc3e1a4a6e4fdd00bb0ddfae
1 /* DirectDraw Surface Implementation
3 * Copyright (c) 1997-2000 Marcus Meissner
4 * Copyright (c) 1998-2000 Lionel Ulmer
5 * Copyright (c) 2000-2001 TransGaming Technologies Inc.
6 * Copyright (c) 2006 Stefan Dösinger
7 * Copyright (c) 2011 Ričardas Barkauskas for CodeWeavers
9 * This file contains the (internal) driver registration functions,
10 * driver enumeration APIs and DirectDraw creation functions.
12 * This library is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU Lesser General Public
14 * License as published by the Free Software Foundation; either
15 * version 2.1 of the License, or (at your option) any later version.
17 * This library is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * Lesser General Public License for more details.
22 * You should have received a copy of the GNU Lesser General Public
23 * License along with this library; if not, write to the Free Software
24 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
27 #include "config.h"
28 #include "wine/port.h"
30 #include "ddraw_private.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(ddraw);
34 static IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface2(IDirectDrawSurface2 *iface);
35 static IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface3(IDirectDrawSurface3 *iface);
37 static inline IDirectDrawSurfaceImpl *impl_from_IDirectDrawGammaControl(IDirectDrawGammaControl *iface)
39 return CONTAINING_RECORD(iface, IDirectDrawSurfaceImpl, IDirectDrawGammaControl_iface);
42 /*****************************************************************************
43 * IUnknown parts follow
44 *****************************************************************************/
46 /*****************************************************************************
47 * IDirectDrawSurface7::QueryInterface
49 * A normal QueryInterface implementation. For QueryInterface rules
50 * see ddraw.c, IDirectDraw7::QueryInterface. This method
51 * can Query IDirectDrawSurface interfaces in all version, IDirect3DTexture
52 * in all versions, the IDirectDrawGammaControl interface and it can
53 * create an IDirect3DDevice. (Uses IDirect3D7::CreateDevice)
55 * Params:
56 * riid: The interface id queried for
57 * obj: Address to write the pointer to
59 * Returns:
60 * S_OK on success
61 * E_NOINTERFACE if the requested interface wasn't found
63 *****************************************************************************/
64 static HRESULT WINAPI ddraw_surface7_QueryInterface(IDirectDrawSurface7 *iface, REFIID riid, void **obj)
66 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
68 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), obj);
70 /* According to COM docs, if the QueryInterface fails, obj should be set to NULL */
71 *obj = NULL;
73 if(!riid)
74 return DDERR_INVALIDPARAMS;
76 if (IsEqualGUID(riid, &IID_IUnknown)
77 || IsEqualGUID(riid, &IID_IDirectDrawSurface7) )
79 IUnknown_AddRef(iface);
80 *obj = iface;
81 TRACE("(%p) returning IDirectDrawSurface7 interface at %p\n", This, *obj);
82 return S_OK;
84 else if (IsEqualGUID(riid, &IID_IDirectDrawSurface4))
86 IUnknown_AddRef(iface);
87 *obj = &This->IDirectDrawSurface4_iface;
88 TRACE("(%p) returning IDirectDrawSurface4 interface at %p\n", This, *obj);
89 return S_OK;
91 else if (IsEqualGUID(riid, &IID_IDirectDrawSurface3))
93 IUnknown_AddRef(iface);
94 *obj = &This->IDirectDrawSurface3_iface;
95 TRACE("(%p) returning IDirectDrawSurface3 interface at %p\n", This, *obj);
96 return S_OK;
98 else if (IsEqualGUID(riid, &IID_IDirectDrawSurface2))
100 IDirectDrawSurface2_AddRef(&This->IDirectDrawSurface2_iface);
101 *obj = &This->IDirectDrawSurface2_iface;
102 TRACE("(%p) returning IDirectDrawSurface2 interface at %p\n", This, *obj);
103 return S_OK;
105 else if (IsEqualGUID(riid, &IID_IDirectDrawSurface))
107 IUnknown_AddRef(iface);
108 *obj = &This->IDirectDrawSurface_iface;
109 TRACE("(%p) returning IDirectDrawSurface interface at %p\n", This, *obj);
110 return S_OK;
112 else if( IsEqualGUID(riid, &IID_IDirectDrawGammaControl) )
114 IUnknown_AddRef(iface);
115 *obj = &This->IDirectDrawGammaControl_iface;
116 TRACE("(%p) returning IDirectDrawGammaControl interface at %p\n", This, *obj);
117 return S_OK;
119 else if( IsEqualGUID(riid, &IID_D3DDEVICE_WineD3D) ||
120 IsEqualGUID(riid, &IID_IDirect3DHALDevice)||
121 IsEqualGUID(riid, &IID_IDirect3DRGBDevice) )
123 IDirect3DDevice7 *d3d;
125 /* Call into IDirect3D7 for creation */
126 IDirect3D7_CreateDevice(&This->ddraw->IDirect3D7_iface, riid, &This->IDirectDrawSurface7_iface,
127 &d3d);
129 if (d3d)
131 *obj = (IDirect3DDevice *)&((IDirect3DDeviceImpl *)d3d)->IDirect3DDevice_vtbl;
132 TRACE("(%p) Returning IDirect3DDevice interface at %p\n", This, *obj);
133 return S_OK;
136 WARN("Unable to create a IDirect3DDevice instance, returning E_NOINTERFACE\n");
137 return E_NOINTERFACE;
139 else if (IsEqualGUID( &IID_IDirect3DTexture, riid ) ||
140 IsEqualGUID( &IID_IDirect3DTexture2, riid ))
142 if (IsEqualGUID( &IID_IDirect3DTexture, riid ))
144 *obj = &This->IDirect3DTexture_vtbl;
145 TRACE(" returning Direct3DTexture interface at %p.\n", *obj);
147 else
149 *obj = &This->IDirect3DTexture2_vtbl;
150 TRACE(" returning Direct3DTexture2 interface at %p.\n", *obj);
152 IUnknown_AddRef( (IUnknown *) *obj);
153 return S_OK;
156 ERR("No interface\n");
157 return E_NOINTERFACE;
160 static HRESULT WINAPI ddraw_surface4_QueryInterface(IDirectDrawSurface4 *iface, REFIID riid, void **object)
162 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
163 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
165 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
168 static HRESULT WINAPI ddraw_surface3_QueryInterface(IDirectDrawSurface3 *iface, REFIID riid, void **object)
170 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
171 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
173 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
176 static HRESULT WINAPI ddraw_surface2_QueryInterface(IDirectDrawSurface2 *iface, REFIID riid, void **object)
178 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
179 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
181 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
184 static HRESULT WINAPI ddraw_surface1_QueryInterface(IDirectDrawSurface *iface, REFIID riid, void **object)
186 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
187 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
189 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
192 static HRESULT WINAPI ddraw_gamma_control_QueryInterface(IDirectDrawGammaControl *iface,
193 REFIID riid, void **object)
195 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawGammaControl(iface);
197 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
199 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
202 static HRESULT WINAPI d3d_texture2_QueryInterface(IDirect3DTexture2 *iface, REFIID riid, void **object)
204 IDirectDrawSurfaceImpl *This = surface_from_texture2(iface);
205 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
207 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
210 static HRESULT WINAPI d3d_texture1_QueryInterface(IDirect3DTexture *iface, REFIID riid, void **object)
212 IDirectDrawSurfaceImpl *This = surface_from_texture1(iface);
213 TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
215 return ddraw_surface7_QueryInterface(&This->IDirectDrawSurface7_iface, riid, object);
218 static void ddraw_surface_add_iface(IDirectDrawSurfaceImpl *This)
220 ULONG iface_count = InterlockedIncrement(&This->iface_count);
221 TRACE("%p increasing iface count to %u.\n", This, iface_count);
223 if (iface_count == 1)
225 EnterCriticalSection(&ddraw_cs);
226 if (This->wined3d_surface)
227 wined3d_surface_incref(This->wined3d_surface);
228 if (This->wined3d_texture)
229 wined3d_texture_incref(This->wined3d_texture);
230 LeaveCriticalSection(&ddraw_cs);
234 /*****************************************************************************
235 * IDirectDrawSurface7::AddRef
237 * A normal addref implementation
239 * Returns:
240 * The new refcount
242 *****************************************************************************/
243 static ULONG WINAPI ddraw_surface7_AddRef(IDirectDrawSurface7 *iface)
245 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
246 ULONG refcount = InterlockedIncrement(&This->ref);
248 TRACE("iface %p increasing refcount to %u.\n", iface, refcount);
250 if (refcount == 1)
252 ddraw_surface_add_iface(This);
255 return refcount;
258 static ULONG WINAPI ddraw_surface4_AddRef(IDirectDrawSurface4 *iface)
260 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
261 TRACE("iface %p.\n", iface);
263 return ddraw_surface7_AddRef(&This->IDirectDrawSurface7_iface);
266 static ULONG WINAPI ddraw_surface3_AddRef(IDirectDrawSurface3 *iface)
268 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
269 TRACE("iface %p.\n", iface);
271 return ddraw_surface7_AddRef(&This->IDirectDrawSurface7_iface);
274 static ULONG WINAPI ddraw_surface2_AddRef(IDirectDrawSurface2 *iface)
276 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
277 ULONG refcount = InterlockedIncrement(&This->ref2);
279 TRACE("iface %p increasing refcount to %u.\n", iface, refcount);
281 if (refcount == 1)
283 ddraw_surface_add_iface(This);
286 return refcount;
289 static ULONG WINAPI ddraw_surface1_AddRef(IDirectDrawSurface *iface)
291 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
292 TRACE("iface %p.\n", iface);
294 return ddraw_surface7_AddRef(&This->IDirectDrawSurface7_iface);
297 static ULONG WINAPI ddraw_gamma_control_AddRef(IDirectDrawGammaControl *iface)
299 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawGammaControl(iface);
301 TRACE("iface %p.\n", iface);
303 return ddraw_surface7_AddRef(&This->IDirectDrawSurface7_iface);
306 static ULONG WINAPI d3d_texture2_AddRef(IDirect3DTexture2 *iface)
308 IDirectDrawSurfaceImpl *This = surface_from_texture2(iface);
309 TRACE("iface %p.\n", iface);
311 return ddraw_surface7_AddRef(&This->IDirectDrawSurface7_iface);
314 static ULONG WINAPI d3d_texture1_AddRef(IDirect3DTexture *iface)
316 IDirectDrawSurfaceImpl *This = surface_from_texture1(iface);
317 TRACE("iface %p.\n", iface);
319 return ddraw_surface7_AddRef(&This->IDirectDrawSurface7_iface);
322 /*****************************************************************************
323 * ddraw_surface_destroy
325 * A helper function for IDirectDrawSurface7::Release
327 * Frees the surface, regardless of its refcount.
328 * See IDirectDrawSurface7::Release for more information
330 * Params:
331 * This: Surface to free
333 *****************************************************************************/
334 void ddraw_surface_destroy(IDirectDrawSurfaceImpl *This)
336 TRACE("surface %p.\n", This);
338 /* Check the iface count and give a warning */
339 if(This->iface_count > 1)
341 /* This can happen when a complex surface is destroyed,
342 * because the 2nd surface was addref()ed when the app
343 * called GetAttachedSurface
345 WARN("(%p): Destroying surface with refcounts %d 2: %d\n", This, This->ref, This->ref2);
348 if (This->wined3d_surface)
349 wined3d_surface_decref(This->wined3d_surface);
352 static void ddraw_surface_cleanup(IDirectDrawSurfaceImpl *surface)
354 IDirectDrawSurfaceImpl *surf;
355 IUnknown *ifaceToRelease;
356 UINT i;
358 TRACE("surface %p.\n", surface);
360 if (surface->wined3d_swapchain)
362 IDirectDrawImpl *ddraw = surface->ddraw;
364 /* If it's the render target, destroy the D3D device. */
365 if (ddraw->d3d_initialized && surface == ddraw->d3d_target)
367 TRACE("Destroying the render target, uninitializing D3D.\n");
369 for (i = 0; i < ddraw->numConvertedDecls; ++i)
371 wined3d_vertex_declaration_decref(ddraw->decls[i].decl);
373 HeapFree(GetProcessHeap(), 0, ddraw->decls);
374 ddraw->numConvertedDecls = 0;
376 if (FAILED(wined3d_device_uninit_3d(ddraw->wined3d_device)))
378 ERR("Failed to uninit 3D.\n");
380 else
382 /* Free the d3d window if one was created. */
383 if (ddraw->d3d_window && ddraw->d3d_window != ddraw->dest_window)
385 TRACE("Destroying the hidden render window %p.\n", ddraw->d3d_window);
386 DestroyWindow(ddraw->d3d_window);
387 ddraw->d3d_window = 0;
391 ddraw->d3d_initialized = FALSE;
392 ddraw->d3d_target = NULL;
394 else
396 wined3d_device_uninit_gdi(ddraw->wined3d_device);
399 surface->wined3d_swapchain = NULL;
401 /* Reset to the default surface implementation type. This is needed
402 * if applications use non render target surfaces and expect blits to
403 * work after destroying the render target.
405 * TODO: Recreate existing offscreen surfaces. */
406 ddraw->ImplType = DefaultSurfaceType;
408 TRACE("D3D unloaded.\n");
411 /* The refcount test shows that the palette is detached when the surface
412 * is destroyed. */
413 IDirectDrawSurface7_SetPalette(&surface->IDirectDrawSurface7_iface, NULL);
415 /* Loop through all complex attached surfaces and destroy them.
417 * Yet again, only the root can have more than one complexly attached
418 * surface, all the others have a total of one. */
419 for (i = 0; i < MAX_COMPLEX_ATTACHED; ++i)
421 if (!surface->complex_array[i])
422 break;
424 surf = surface->complex_array[i];
425 surface->complex_array[i] = NULL;
426 while (surf)
428 IDirectDrawSurfaceImpl *destroy = surf;
429 surf = surf->complex_array[0]; /* Iterate through the "tree" */
430 ddraw_surface_destroy(destroy); /* Destroy it */
434 ifaceToRelease = surface->ifaceToRelease;
436 /* Destroy the root surface. */
437 ddraw_surface_destroy(surface);
439 /* Reduce the ddraw refcount */
440 if (ifaceToRelease)
441 IUnknown_Release(ifaceToRelease);
444 static void ddraw_surface_release_iface(IDirectDrawSurfaceImpl *This)
446 ULONG iface_count = InterlockedDecrement(&This->iface_count);
447 TRACE("%p decreasing iface count to %u.\n", This, iface_count);
449 if (iface_count == 0)
451 /* Complex attached surfaces are destroyed implicitly when the root is released */
452 EnterCriticalSection(&ddraw_cs);
453 if(!This->is_complex_root)
455 WARN("(%p) Attempt to destroy a surface that is not a complex root\n", This);
456 LeaveCriticalSection(&ddraw_cs);
457 return;
459 if (This->wined3d_texture) /* If it's a texture, destroy the wined3d texture. */
460 wined3d_texture_decref(This->wined3d_texture);
461 else
462 ddraw_surface_cleanup(This);
463 LeaveCriticalSection(&ddraw_cs);
466 return;
469 /*****************************************************************************
470 * IDirectDrawSurface7::Release
472 * Reduces the surface's refcount by 1. If the refcount falls to 0, the
473 * surface is destroyed.
475 * Destroying the surface is a bit tricky. For the connection between
476 * WineD3DSurfaces and DirectDrawSurfaces see IDirectDraw7::CreateSurface
477 * It has a nice graph explaining the connection.
479 * What happens here is basically this:
480 * When a surface is destroyed, its WineD3DSurface is released,
481 * and the refcount of the DirectDraw interface is reduced by 1. If it has
482 * complex surfaces attached to it, then these surfaces are destroyed too,
483 * regardless of their refcount. If any surface being destroyed has another
484 * surface attached to it (with a "soft" attachment, not complex), then
485 * this surface is detached with DeleteAttachedSurface.
487 * When the surface is a texture, the WineD3DTexture is released.
488 * If the surface is the Direct3D render target, then the D3D
489 * capabilities of the WineD3DDevice are uninitialized, which causes the
490 * swapchain to be released.
492 * When a complex sublevel falls to ref zero, then this is ignored.
494 * Returns:
495 * The new refcount
497 *****************************************************************************/
498 static ULONG WINAPI ddraw_surface7_Release(IDirectDrawSurface7 *iface)
500 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
501 ULONG refcount = InterlockedDecrement(&This->ref);
503 TRACE("iface %p decreasing refcount to %u.\n", iface, refcount);
505 if (refcount == 0)
507 ddraw_surface_release_iface(This);
510 return refcount;
513 static ULONG WINAPI ddraw_surface4_Release(IDirectDrawSurface4 *iface)
515 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
516 TRACE("iface %p.\n", iface);
518 return ddraw_surface7_Release(&This->IDirectDrawSurface7_iface);
521 static ULONG WINAPI ddraw_surface3_Release(IDirectDrawSurface3 *iface)
523 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
524 TRACE("iface %p.\n", iface);
526 return ddraw_surface7_Release(&This->IDirectDrawSurface7_iface);
529 static ULONG WINAPI ddraw_surface2_Release(IDirectDrawSurface2 *iface)
531 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
532 ULONG refcount = InterlockedDecrement(&This->ref2);
534 TRACE("iface %p decreasing refcount to %u.\n", iface, refcount);
536 if (refcount == 0)
538 ddraw_surface_release_iface(This);
541 return refcount;
544 static ULONG WINAPI ddraw_surface1_Release(IDirectDrawSurface *iface)
546 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
547 TRACE("iface %p.\n", iface);
549 return ddraw_surface7_Release(&This->IDirectDrawSurface7_iface);
552 static ULONG WINAPI ddraw_gamma_control_Release(IDirectDrawGammaControl *iface)
554 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawGammaControl(iface);
556 TRACE("iface %p.\n", iface);
558 return ddraw_surface7_Release(&This->IDirectDrawSurface7_iface);
561 static ULONG WINAPI d3d_texture2_Release(IDirect3DTexture2 *iface)
563 IDirectDrawSurfaceImpl *This = surface_from_texture2(iface);
564 TRACE("iface %p.\n", iface);
566 return ddraw_surface7_Release(&This->IDirectDrawSurface7_iface);
569 static ULONG WINAPI d3d_texture1_Release(IDirect3DTexture *iface)
571 IDirectDrawSurfaceImpl *This = surface_from_texture1(iface);
572 TRACE("iface %p.\n", iface);
574 return ddraw_surface7_Release(&This->IDirectDrawSurface7_iface);
577 /*****************************************************************************
578 * IDirectDrawSurface7::GetAttachedSurface
580 * Returns an attached surface with the requested caps. Surface attachment
581 * and complex surfaces are not clearly described by the MSDN or sdk,
582 * so this method is tricky and likely to contain problems.
583 * This implementation searches the complex list first, then the
584 * attachment chain.
586 * The chains are searched from This down to the last surface in the chain,
587 * not from the first element in the chain. The first surface found is
588 * returned. The MSDN says that this method fails if more than one surface
589 * matches the caps, but it is not sure if that is right. The attachment
590 * structure may not even allow two matching surfaces.
592 * The found surface is AddRef-ed before it is returned.
594 * Params:
595 * Caps: Pointer to a DDCAPS2 structure describing the caps asked for
596 * Surface: Address to store the found surface
598 * Returns:
599 * DD_OK on success
600 * DDERR_INVALIDPARAMS if Caps or Surface is NULL
601 * DDERR_NOTFOUND if no surface was found
603 *****************************************************************************/
604 static HRESULT WINAPI ddraw_surface7_GetAttachedSurface(IDirectDrawSurface7 *iface,
605 DDSCAPS2 *Caps, IDirectDrawSurface7 **Surface)
607 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
608 IDirectDrawSurfaceImpl *surf;
609 DDSCAPS2 our_caps;
610 int i;
612 TRACE("iface %p, caps %p, attachment %p.\n", iface, Caps, Surface);
614 EnterCriticalSection(&ddraw_cs);
616 if(This->version < 7)
618 /* Earlier dx apps put garbage into these members, clear them */
619 our_caps.dwCaps = Caps->dwCaps;
620 our_caps.dwCaps2 = 0;
621 our_caps.dwCaps3 = 0;
622 our_caps.dwCaps4 = 0;
624 else
626 our_caps = *Caps;
629 TRACE("(%p): Looking for caps: %x,%x,%x,%x\n", This, our_caps.dwCaps, our_caps.dwCaps2, our_caps.dwCaps3, our_caps.dwCaps4); /* FIXME: Better debugging */
631 for(i = 0; i < MAX_COMPLEX_ATTACHED; i++)
633 surf = This->complex_array[i];
634 if(!surf) break;
636 if (TRACE_ON(ddraw))
638 TRACE("Surface: (%p) caps: %x,%x,%x,%x\n", surf,
639 surf->surface_desc.ddsCaps.dwCaps,
640 surf->surface_desc.ddsCaps.dwCaps2,
641 surf->surface_desc.ddsCaps.dwCaps3,
642 surf->surface_desc.ddsCaps.dwCaps4);
645 if (((surf->surface_desc.ddsCaps.dwCaps & our_caps.dwCaps) == our_caps.dwCaps) &&
646 ((surf->surface_desc.ddsCaps.dwCaps2 & our_caps.dwCaps2) == our_caps.dwCaps2)) {
648 /* MSDN: "This method fails if more than one surface is attached
649 * that matches the capabilities requested."
651 * Not sure how to test this.
654 TRACE("(%p): Returning surface %p\n", This, surf);
655 TRACE("(%p): mipmapcount=%d\n", This, surf->mipmap_level);
656 *Surface = &surf->IDirectDrawSurface7_iface;
657 ddraw_surface7_AddRef(*Surface);
658 LeaveCriticalSection(&ddraw_cs);
659 return DD_OK;
663 /* Next, look at the attachment chain */
664 surf = This;
666 while( (surf = surf->next_attached) )
668 if (TRACE_ON(ddraw))
670 TRACE("Surface: (%p) caps: %x,%x,%x,%x\n", surf,
671 surf->surface_desc.ddsCaps.dwCaps,
672 surf->surface_desc.ddsCaps.dwCaps2,
673 surf->surface_desc.ddsCaps.dwCaps3,
674 surf->surface_desc.ddsCaps.dwCaps4);
677 if (((surf->surface_desc.ddsCaps.dwCaps & our_caps.dwCaps) == our_caps.dwCaps) &&
678 ((surf->surface_desc.ddsCaps.dwCaps2 & our_caps.dwCaps2) == our_caps.dwCaps2)) {
680 TRACE("(%p): Returning surface %p\n", This, surf);
681 *Surface = &surf->IDirectDrawSurface7_iface;
682 ddraw_surface7_AddRef(*Surface);
683 LeaveCriticalSection(&ddraw_cs);
684 return DD_OK;
688 TRACE("(%p) Didn't find a valid surface\n", This);
689 LeaveCriticalSection(&ddraw_cs);
691 *Surface = NULL;
692 return DDERR_NOTFOUND;
695 static HRESULT WINAPI ddraw_surface4_GetAttachedSurface(IDirectDrawSurface4 *iface,
696 DDSCAPS2 *caps, IDirectDrawSurface4 **attachment)
698 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
699 IDirectDrawSurface7 *attachment7;
700 IDirectDrawSurfaceImpl *attachment_impl;
701 HRESULT hr;
703 TRACE("iface %p, caps %p, attachment %p.\n", iface, caps, attachment);
705 hr = ddraw_surface7_GetAttachedSurface(&This->IDirectDrawSurface7_iface,
706 caps, &attachment7);
707 if (FAILED(hr))
709 *attachment = NULL;
710 return hr;
712 attachment_impl = impl_from_IDirectDrawSurface7(attachment7);
713 *attachment = &attachment_impl->IDirectDrawSurface4_iface;
715 return hr;
718 static HRESULT WINAPI ddraw_surface3_GetAttachedSurface(IDirectDrawSurface3 *iface,
719 DDSCAPS *caps, IDirectDrawSurface3 **attachment)
721 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
722 IDirectDrawSurface7 *attachment7;
723 IDirectDrawSurfaceImpl *attachment_impl;
724 DDSCAPS2 caps2;
725 HRESULT hr;
727 TRACE("iface %p, caps %p, attachment %p.\n", iface, caps, attachment);
729 caps2.dwCaps = caps->dwCaps;
730 caps2.dwCaps2 = 0;
731 caps2.dwCaps3 = 0;
732 caps2.dwCaps4 = 0;
734 hr = ddraw_surface7_GetAttachedSurface(&This->IDirectDrawSurface7_iface,
735 &caps2, &attachment7);
736 if (FAILED(hr))
738 *attachment = NULL;
739 return hr;
741 attachment_impl = impl_from_IDirectDrawSurface7(attachment7);
742 *attachment = &attachment_impl->IDirectDrawSurface3_iface;
744 return hr;
747 static HRESULT WINAPI ddraw_surface2_GetAttachedSurface(IDirectDrawSurface2 *iface,
748 DDSCAPS *caps, IDirectDrawSurface2 **attachment)
750 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
751 IDirectDrawSurface7 *attachment7;
752 IDirectDrawSurfaceImpl *attachment_impl;
753 DDSCAPS2 caps2;
754 HRESULT hr;
756 TRACE("iface %p, caps %p, attachment %p.\n", iface, caps, attachment);
758 caps2.dwCaps = caps->dwCaps;
759 caps2.dwCaps2 = 0;
760 caps2.dwCaps3 = 0;
761 caps2.dwCaps4 = 0;
763 hr = ddraw_surface7_GetAttachedSurface(&This->IDirectDrawSurface7_iface,
764 &caps2, &attachment7);
765 if (FAILED(hr))
767 *attachment = NULL;
768 return hr;
770 attachment_impl = impl_from_IDirectDrawSurface7(attachment7);
771 *attachment = &attachment_impl->IDirectDrawSurface2_iface;
772 ddraw_surface2_AddRef(*attachment);
773 ddraw_surface7_Release(attachment7);
775 return hr;
778 static HRESULT WINAPI ddraw_surface1_GetAttachedSurface(IDirectDrawSurface *iface,
779 DDSCAPS *caps, IDirectDrawSurface **attachment)
781 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
782 IDirectDrawSurface7 *attachment7;
783 IDirectDrawSurfaceImpl *attachment_impl;
784 DDSCAPS2 caps2;
785 HRESULT hr;
787 TRACE("iface %p, caps %p, attachment %p.\n", iface, caps, attachment);
789 caps2.dwCaps = caps->dwCaps;
790 caps2.dwCaps2 = 0;
791 caps2.dwCaps3 = 0;
792 caps2.dwCaps4 = 0;
794 hr = ddraw_surface7_GetAttachedSurface(&This->IDirectDrawSurface7_iface,
795 &caps2, &attachment7);
796 if (FAILED(hr))
798 *attachment = NULL;
799 return hr;
801 attachment_impl = impl_from_IDirectDrawSurface7(attachment7);
802 *attachment = &attachment_impl->IDirectDrawSurface_iface;
804 return hr;
807 /*****************************************************************************
808 * IDirectDrawSurface7::Lock
810 * Locks the surface and returns a pointer to the surface's memory
812 * Params:
813 * Rect: Rectangle to lock. If NULL, the whole surface is locked
814 * DDSD: Pointer to a DDSURFACEDESC2 which shall receive the surface's desc.
815 * Flags: Locking flags, e.g Read only or write only
816 * h: An event handle that's not used and must be NULL
818 * Returns:
819 * DD_OK on success
820 * DDERR_INVALIDPARAMS if DDSD is NULL
821 * For more details, see IWineD3DSurface::LockRect
823 *****************************************************************************/
824 static HRESULT WINAPI ddraw_surface7_Lock(IDirectDrawSurface7 *iface,
825 RECT *Rect, DDSURFACEDESC2 *DDSD, DWORD Flags, HANDLE h)
827 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
828 WINED3DLOCKED_RECT LockedRect;
829 HRESULT hr;
831 TRACE("iface %p, rect %s, surface_desc %p, flags %#x, h %p.\n",
832 iface, wine_dbgstr_rect(Rect), DDSD, Flags, h);
834 if(!DDSD)
835 return DDERR_INVALIDPARAMS;
837 /* This->surface_desc.dwWidth and dwHeight are changeable, thus lock */
838 EnterCriticalSection(&ddraw_cs);
840 /* Should I check for the handle to be NULL?
842 * The DDLOCK flags and the D3DLOCK flags are equal
843 * for the supported values. The others are ignored by WineD3D
846 if(DDSD->dwSize != sizeof(DDSURFACEDESC) &&
847 DDSD->dwSize != sizeof(DDSURFACEDESC2))
849 WARN("Invalid structure size %d, returning DDERR_INVALIDPARAMS\n", DDERR_INVALIDPARAMS);
850 LeaveCriticalSection(&ddraw_cs);
851 return DDERR_INVALIDPARAMS;
854 /* Windows zeroes this if the rect is invalid */
855 DDSD->lpSurface = 0;
857 if (Rect)
859 if ((Rect->left < 0)
860 || (Rect->top < 0)
861 || (Rect->left > Rect->right)
862 || (Rect->top > Rect->bottom)
863 || (Rect->right > This->surface_desc.dwWidth)
864 || (Rect->bottom > This->surface_desc.dwHeight))
866 WARN("Trying to lock an invalid rectangle, returning DDERR_INVALIDPARAMS\n");
867 LeaveCriticalSection(&ddraw_cs);
868 return DDERR_INVALIDPARAMS;
872 hr = wined3d_surface_map(This->wined3d_surface, &LockedRect, Rect, Flags);
873 if (FAILED(hr))
875 LeaveCriticalSection(&ddraw_cs);
876 switch(hr)
878 /* D3D8 and D3D9 return the general D3DERR_INVALIDCALL error, but ddraw has a more
879 * specific error. But since IWineD3DSurface::LockRect returns that error in this
880 * only occasion, keep d3d8 and d3d9 free from the return value override. There are
881 * many different places where d3d8/9 would have to catch the DDERR_SURFACEBUSY, it
882 * is much easier to do it in one place in ddraw
884 case WINED3DERR_INVALIDCALL: return DDERR_SURFACEBUSY;
885 default: return hr;
889 /* Override the memory area. The pitch should be set already. Strangely windows
890 * does not set the LPSURFACE flag on locked surfaces !?!.
891 * DDSD->dwFlags |= DDSD_LPSURFACE;
893 This->surface_desc.lpSurface = LockedRect.pBits;
894 DD_STRUCT_COPY_BYSIZE(DDSD,&(This->surface_desc));
896 TRACE("locked surface returning description :\n");
897 if (TRACE_ON(ddraw)) DDRAW_dump_surface_desc(DDSD);
899 LeaveCriticalSection(&ddraw_cs);
900 return DD_OK;
903 static HRESULT WINAPI ddraw_surface4_Lock(IDirectDrawSurface4 *iface, RECT *rect,
904 DDSURFACEDESC2 *surface_desc, DWORD flags, HANDLE h)
906 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
907 TRACE("iface %p, rect %s, surface_desc %p, flags %#x, h %p.\n",
908 iface, wine_dbgstr_rect(rect), surface_desc, flags, h);
910 return ddraw_surface7_Lock(&This->IDirectDrawSurface7_iface,
911 rect, surface_desc, flags, h);
914 static HRESULT WINAPI ddraw_surface3_Lock(IDirectDrawSurface3 *iface, RECT *rect,
915 DDSURFACEDESC *surface_desc, DWORD flags, HANDLE h)
917 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
918 TRACE("iface %p, rect %s, surface_desc %p, flags %#x, h %p.\n",
919 iface, wine_dbgstr_rect(rect), surface_desc, flags, h);
921 return ddraw_surface7_Lock(&This->IDirectDrawSurface7_iface,
922 rect, (DDSURFACEDESC2 *)surface_desc, flags, h);
925 static HRESULT WINAPI ddraw_surface2_Lock(IDirectDrawSurface2 *iface, RECT *rect,
926 DDSURFACEDESC *surface_desc, DWORD flags, HANDLE h)
928 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
929 TRACE("iface %p, rect %s, surface_desc %p, flags %#x, h %p.\n",
930 iface, wine_dbgstr_rect(rect), surface_desc, flags, h);
932 return ddraw_surface7_Lock(&This->IDirectDrawSurface7_iface,
933 rect, (DDSURFACEDESC2 *)surface_desc, flags, h);
936 static HRESULT WINAPI ddraw_surface1_Lock(IDirectDrawSurface *iface, RECT *rect,
937 DDSURFACEDESC *surface_desc, DWORD flags, HANDLE h)
939 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
940 TRACE("iface %p, rect %s, surface_desc %p, flags %#x, h %p.\n",
941 iface, wine_dbgstr_rect(rect), surface_desc, flags, h);
943 return ddraw_surface7_Lock(&This->IDirectDrawSurface7_iface,
944 rect, (DDSURFACEDESC2 *)surface_desc, flags, h);
947 /*****************************************************************************
948 * IDirectDrawSurface7::Unlock
950 * Unlocks an locked surface
952 * Params:
953 * Rect: Not used by this implementation
955 * Returns:
956 * D3D_OK on success
957 * For more details, see IWineD3DSurface::UnlockRect
959 *****************************************************************************/
960 static HRESULT WINAPI ddraw_surface7_Unlock(IDirectDrawSurface7 *iface, RECT *pRect)
962 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
963 HRESULT hr;
965 TRACE("iface %p, rect %s.\n", iface, wine_dbgstr_rect(pRect));
967 EnterCriticalSection(&ddraw_cs);
968 hr = wined3d_surface_unmap(This->wined3d_surface);
969 if (SUCCEEDED(hr))
971 This->surface_desc.lpSurface = NULL;
973 LeaveCriticalSection(&ddraw_cs);
974 return hr;
977 static HRESULT WINAPI ddraw_surface4_Unlock(IDirectDrawSurface4 *iface, RECT *pRect)
979 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
980 TRACE("iface %p, rect %p.\n", iface, pRect);
982 return ddraw_surface7_Unlock(&This->IDirectDrawSurface7_iface, pRect);
985 static HRESULT WINAPI ddraw_surface3_Unlock(IDirectDrawSurface3 *iface, void *data)
987 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
988 TRACE("iface %p, data %p.\n", iface, data);
990 /* data might not be the LPRECT of later versions, so drop it. */
991 return ddraw_surface7_Unlock(&This->IDirectDrawSurface7_iface, NULL);
994 static HRESULT WINAPI ddraw_surface2_Unlock(IDirectDrawSurface2 *iface, void *data)
996 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
997 TRACE("iface %p, data %p.\n", iface, data);
999 /* data might not be the LPRECT of later versions, so drop it. */
1000 return ddraw_surface7_Unlock(&This->IDirectDrawSurface7_iface, NULL);
1003 static HRESULT WINAPI ddraw_surface1_Unlock(IDirectDrawSurface *iface, void *data)
1005 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1006 TRACE("iface %p, data %p.\n", iface, data);
1008 /* data might not be the LPRECT of later versions, so drop it. */
1009 return ddraw_surface7_Unlock(&This->IDirectDrawSurface7_iface, NULL);
1012 /*****************************************************************************
1013 * IDirectDrawSurface7::Flip
1015 * Flips a surface with the DDSCAPS_FLIP flag. The flip is relayed to
1016 * IWineD3DSurface::Flip. Because WineD3D doesn't handle attached surfaces,
1017 * the flip target is passed to WineD3D, even if the app didn't specify one
1019 * Params:
1020 * DestOverride: Specifies the surface that will become the new front
1021 * buffer. If NULL, the current back buffer is used
1022 * Flags: some DirectDraw flags, see include/ddraw.h
1024 * Returns:
1025 * DD_OK on success
1026 * DDERR_NOTFLIPPABLE if no flip target could be found
1027 * DDERR_INVALIDOBJECT if the surface isn't a front buffer
1028 * For more details, see IWineD3DSurface::Flip
1030 *****************************************************************************/
1031 static HRESULT WINAPI ddraw_surface7_Flip(IDirectDrawSurface7 *iface, IDirectDrawSurface7 *DestOverride, DWORD Flags)
1033 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1034 IDirectDrawSurfaceImpl *Override = unsafe_impl_from_IDirectDrawSurface7(DestOverride);
1035 IDirectDrawSurface7 *Override7;
1036 HRESULT hr;
1038 TRACE("iface %p, dst %p, flags %#x.\n", iface, DestOverride, Flags);
1040 /* Flip has to be called from a front buffer
1041 * What about overlay surfaces, AFAIK they can flip too?
1043 if( !(This->surface_desc.ddsCaps.dwCaps & (DDSCAPS_FRONTBUFFER | DDSCAPS_OVERLAY)) )
1044 return DDERR_INVALIDOBJECT; /* Unchecked */
1046 EnterCriticalSection(&ddraw_cs);
1048 /* WineD3D doesn't keep track of attached surface, so find the target */
1049 if(!Override)
1051 DDSCAPS2 Caps;
1053 memset(&Caps, 0, sizeof(Caps));
1054 Caps.dwCaps |= DDSCAPS_BACKBUFFER;
1055 hr = ddraw_surface7_GetAttachedSurface(iface, &Caps, &Override7);
1056 if(hr != DD_OK)
1058 ERR("Can't find a flip target\n");
1059 LeaveCriticalSection(&ddraw_cs);
1060 return DDERR_NOTFLIPPABLE; /* Unchecked */
1062 Override = impl_from_IDirectDrawSurface7(Override7);
1064 /* For the GetAttachedSurface */
1065 ddraw_surface7_Release(Override7);
1068 hr = wined3d_surface_flip(This->wined3d_surface, Override->wined3d_surface, Flags);
1069 LeaveCriticalSection(&ddraw_cs);
1070 return hr;
1073 static HRESULT WINAPI ddraw_surface4_Flip(IDirectDrawSurface4 *iface, IDirectDrawSurface4 *dst, DWORD flags)
1075 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1076 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface4(dst);
1077 TRACE("iface %p, dst %p, flags %#x.\n", iface, dst, flags);
1079 return ddraw_surface7_Flip(&This->IDirectDrawSurface7_iface,
1080 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, flags);
1083 static HRESULT WINAPI ddraw_surface3_Flip(IDirectDrawSurface3 *iface, IDirectDrawSurface3 *dst, DWORD flags)
1085 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1086 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface3(dst);
1087 TRACE("iface %p, dst %p, flags %#x.\n", iface, dst, flags);
1089 return ddraw_surface7_Flip(&This->IDirectDrawSurface7_iface,
1090 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, flags);
1093 static HRESULT WINAPI ddraw_surface2_Flip(IDirectDrawSurface2 *iface, IDirectDrawSurface2 *dst, DWORD flags)
1095 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1096 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface2(dst);
1097 TRACE("iface %p, dst %p, flags %#x.\n", iface, dst, flags);
1099 return ddraw_surface7_Flip(&This->IDirectDrawSurface7_iface,
1100 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, flags);
1103 static HRESULT WINAPI ddraw_surface1_Flip(IDirectDrawSurface *iface, IDirectDrawSurface *dst, DWORD flags)
1105 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1106 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface(dst);
1107 TRACE("iface %p, dst %p, flags %#x.\n", iface, dst, flags);
1109 return ddraw_surface7_Flip(&This->IDirectDrawSurface7_iface,
1110 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, flags);
1113 /*****************************************************************************
1114 * IDirectDrawSurface7::Blt
1116 * Performs a blit on the surface
1118 * Params:
1119 * DestRect: Destination rectangle, can be NULL
1120 * SrcSurface: Source surface, can be NULL
1121 * SrcRect: Source rectangle, can be NULL
1122 * Flags: Blt flags
1123 * DDBltFx: Some extended blt parameters, connected to the flags
1125 * Returns:
1126 * D3D_OK on success
1127 * See IWineD3DSurface::Blt for more details
1129 *****************************************************************************/
1130 static HRESULT WINAPI ddraw_surface7_Blt(IDirectDrawSurface7 *iface, RECT *DestRect,
1131 IDirectDrawSurface7 *SrcSurface, RECT *SrcRect, DWORD Flags, DDBLTFX *DDBltFx)
1133 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1134 IDirectDrawSurfaceImpl *Src = unsafe_impl_from_IDirectDrawSurface7(SrcSurface);
1135 HRESULT hr;
1137 TRACE("iface %p, dst_rect %s, src_surface %p, src_rect %s, flags %#x, fx %p.\n",
1138 iface, wine_dbgstr_rect(DestRect), SrcSurface, wine_dbgstr_rect(SrcRect), Flags, DDBltFx);
1140 /* Check for validity of the flags here. WineD3D Has the software-opengl selection path and would have
1141 * to check at 2 places, and sometimes do double checks. This also saves the call to wined3d :-)
1143 if((Flags & DDBLT_KEYSRCOVERRIDE) && (!DDBltFx || Flags & DDBLT_KEYSRC)) {
1144 WARN("Invalid source color key parameters, returning DDERR_INVALIDPARAMS\n");
1145 return DDERR_INVALIDPARAMS;
1148 if((Flags & DDBLT_KEYDESTOVERRIDE) && (!DDBltFx || Flags & DDBLT_KEYDEST)) {
1149 WARN("Invalid destination color key parameters, returning DDERR_INVALIDPARAMS\n");
1150 return DDERR_INVALIDPARAMS;
1153 /* Sizes can change, therefore hold the lock when testing the rectangles */
1154 EnterCriticalSection(&ddraw_cs);
1155 if(DestRect)
1157 if(DestRect->top >= DestRect->bottom || DestRect->left >= DestRect->right ||
1158 DestRect->right > This->surface_desc.dwWidth ||
1159 DestRect->bottom > This->surface_desc.dwHeight)
1161 WARN("Destination rectangle is invalid, returning DDERR_INVALIDRECT\n");
1162 LeaveCriticalSection(&ddraw_cs);
1163 return DDERR_INVALIDRECT;
1166 if(Src && SrcRect)
1168 if(SrcRect->top >= SrcRect->bottom || SrcRect->left >=SrcRect->right ||
1169 SrcRect->right > Src->surface_desc.dwWidth ||
1170 SrcRect->bottom > Src->surface_desc.dwHeight)
1172 WARN("Source rectangle is invalid, returning DDERR_INVALIDRECT\n");
1173 LeaveCriticalSection(&ddraw_cs);
1174 return DDERR_INVALIDRECT;
1178 if(Flags & DDBLT_KEYSRC && (!Src || !(Src->surface_desc.dwFlags & DDSD_CKSRCBLT))) {
1179 WARN("DDBLT_KEYDEST blit without color key in surface, returning DDERR_INVALIDPARAMS\n");
1180 LeaveCriticalSection(&ddraw_cs);
1181 return DDERR_INVALIDPARAMS;
1184 /* TODO: Check if the DDBltFx contains any ddraw surface pointers. If it
1185 * does, copy the struct, and replace the ddraw surfaces with the wined3d
1186 * surfaces. So far no blitting operations using surfaces in the bltfx
1187 * struct are supported anyway. */
1188 hr = wined3d_surface_blt(This->wined3d_surface, DestRect, Src ? Src->wined3d_surface : NULL,
1189 SrcRect, Flags, (WINEDDBLTFX *)DDBltFx, WINED3DTEXF_LINEAR);
1191 LeaveCriticalSection(&ddraw_cs);
1192 switch(hr)
1194 case WINED3DERR_NOTAVAILABLE: return DDERR_UNSUPPORTED;
1195 case WINED3DERR_WRONGTEXTUREFORMAT: return DDERR_INVALIDPIXELFORMAT;
1196 default: return hr;
1200 static HRESULT WINAPI ddraw_surface4_Blt(IDirectDrawSurface4 *iface, RECT *dst_rect,
1201 IDirectDrawSurface4 *src_surface, RECT *src_rect, DWORD flags, DDBLTFX *fx)
1203 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1204 IDirectDrawSurfaceImpl *src = unsafe_impl_from_IDirectDrawSurface4(src_surface);
1205 TRACE("iface %p, dst_rect %s, src_surface %p, src_rect %s, flags %#x, fx %p.\n",
1206 iface, wine_dbgstr_rect(dst_rect), src_surface, wine_dbgstr_rect(src_rect), flags, fx);
1208 return ddraw_surface7_Blt(&This->IDirectDrawSurface7_iface, dst_rect,
1209 src ? &src->IDirectDrawSurface7_iface : NULL, src_rect, flags, fx);
1212 static HRESULT WINAPI ddraw_surface3_Blt(IDirectDrawSurface3 *iface, RECT *dst_rect,
1213 IDirectDrawSurface3 *src_surface, RECT *src_rect, DWORD flags, DDBLTFX *fx)
1215 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1216 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface3(src_surface);
1217 TRACE("iface %p, dst_rect %s, src_surface %p, src_rect %s, flags %#x, fx %p.\n",
1218 iface, wine_dbgstr_rect(dst_rect), src_surface, wine_dbgstr_rect(src_rect), flags, fx);
1220 return ddraw_surface7_Blt(&This->IDirectDrawSurface7_iface, dst_rect,
1221 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags, fx);
1224 static HRESULT WINAPI ddraw_surface2_Blt(IDirectDrawSurface2 *iface, RECT *dst_rect,
1225 IDirectDrawSurface2 *src_surface, RECT *src_rect, DWORD flags, DDBLTFX *fx)
1227 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1228 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface2(src_surface);
1229 TRACE("iface %p, dst_rect %s, src_surface %p, src_rect %s, flags %#x, fx %p.\n",
1230 iface, wine_dbgstr_rect(dst_rect), src_surface, wine_dbgstr_rect(src_rect), flags, fx);
1232 return ddraw_surface7_Blt(&This->IDirectDrawSurface7_iface, dst_rect,
1233 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags, fx);
1236 static HRESULT WINAPI ddraw_surface1_Blt(IDirectDrawSurface *iface, RECT *dst_rect,
1237 IDirectDrawSurface *src_surface, RECT *src_rect, DWORD flags, DDBLTFX *fx)
1239 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1240 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface(src_surface);
1241 TRACE("iface %p, dst_rect %s, src_surface %p, src_rect %s, flags %#x, fx %p.\n",
1242 iface, wine_dbgstr_rect(dst_rect), src_surface, wine_dbgstr_rect(src_rect), flags, fx);
1244 return ddraw_surface7_Blt(&This->IDirectDrawSurface7_iface, dst_rect,
1245 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags, fx);
1248 /*****************************************************************************
1249 * IDirectDrawSurface7::AddAttachedSurface
1251 * Attaches a surface to another surface. How the surface attachments work
1252 * is not totally understood yet, and this method is prone to problems.
1253 * he surface that is attached is AddRef-ed.
1255 * Tests with complex surfaces suggest that the surface attachments form a
1256 * tree, but no method to test this has been found yet.
1258 * The attachment list consists of a first surface (first_attached) and
1259 * for each surface a pointer to the next attached surface (next_attached).
1260 * For the first surface, and a surface that has no attachments
1261 * first_attached points to the surface itself. A surface that has
1262 * no successors in the chain has next_attached set to NULL.
1264 * Newly attached surfaces are attached right after the root surface.
1265 * If a surface is attached to a complex surface compound, it's attached to
1266 * the surface that the app requested, not the complex root. See
1267 * GetAttachedSurface for a description how surfaces are found.
1269 * This is how the current implementation works, and it was coded by looking
1270 * at the needs of the applications.
1272 * So far only Z-Buffer attachments are tested, and they are activated in
1273 * WineD3D. Mipmaps could be tricky to activate in WineD3D.
1274 * Back buffers should work in 2D mode, but they are not tested(They can be
1275 * attached in older iface versions). Rendering to the front buffer and
1276 * switching between that and double buffering is not yet implemented in
1277 * WineD3D, so for 3D it might have unexpected results.
1279 * ddraw_surface_attach_surface is the real thing,
1280 * ddraw_surface7_AddAttachedSurface is a wrapper around it that
1281 * performs additional checks. Version 7 of this interface is much more restrictive
1282 * than its predecessors.
1284 * Params:
1285 * Attach: Surface to attach to iface
1287 * Returns:
1288 * DD_OK on success
1289 * DDERR_CANNOTATTACHSURFACE if the surface can't be attached for some reason
1291 *****************************************************************************/
1292 static HRESULT ddraw_surface_attach_surface(IDirectDrawSurfaceImpl *This, IDirectDrawSurfaceImpl *Surf)
1294 TRACE("surface %p, attachment %p.\n", This, Surf);
1296 if(Surf == This)
1297 return DDERR_CANNOTATTACHSURFACE; /* unchecked */
1299 EnterCriticalSection(&ddraw_cs);
1301 /* Check if the surface is already attached somewhere */
1302 if (Surf->next_attached || Surf->first_attached != Surf)
1304 /* TODO: Test for the structure of the manual attachment. Is it a
1305 * chain or a list? What happens if one surface is attached to 2
1306 * different surfaces? */
1307 WARN("Surface %p is already attached somewhere. next_attached %p, first_attached %p.\n",
1308 Surf, Surf->next_attached, Surf->first_attached);
1310 LeaveCriticalSection(&ddraw_cs);
1311 return DDERR_SURFACEALREADYATTACHED;
1314 /* This inserts the new surface at the 2nd position in the chain, right after the root surface */
1315 Surf->next_attached = This->next_attached;
1316 Surf->first_attached = This->first_attached;
1317 This->next_attached = Surf;
1319 /* Check if the WineD3D depth stencil needs updating */
1320 if(This->ddraw->d3ddevice)
1322 IDirect3DDeviceImpl_UpdateDepthStencil(This->ddraw->d3ddevice);
1325 LeaveCriticalSection(&ddraw_cs);
1326 return DD_OK;
1329 static HRESULT WINAPI ddraw_surface7_AddAttachedSurface(IDirectDrawSurface7 *iface, IDirectDrawSurface7 *Attach)
1331 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1332 IDirectDrawSurfaceImpl *Surf = unsafe_impl_from_IDirectDrawSurface7(Attach);
1333 HRESULT hr;
1335 TRACE("iface %p, attachment %p.\n", iface, Attach);
1337 /* Version 7 of this interface seems to refuse everything except z buffers, as per msdn */
1338 if(!(Surf->surface_desc.ddsCaps.dwCaps & DDSCAPS_ZBUFFER))
1341 WARN("Application tries to attach a non Z buffer surface. caps %08x\n",
1342 Surf->surface_desc.ddsCaps.dwCaps);
1343 return DDERR_CANNOTATTACHSURFACE;
1346 hr = ddraw_surface_attach_surface(This, Surf);
1347 if (FAILED(hr))
1349 return hr;
1351 ddraw_surface7_AddRef(Attach);
1352 return hr;
1355 static HRESULT WINAPI ddraw_surface4_AddAttachedSurface(IDirectDrawSurface4 *iface, IDirectDrawSurface4 *attachment)
1357 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1358 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface4(attachment);
1359 HRESULT hr;
1361 TRACE("iface %p, attachment %p.\n", iface, attachment);
1363 hr = ddraw_surface7_AddAttachedSurface(&This->IDirectDrawSurface7_iface,
1364 attachment_impl ? &attachment_impl->IDirectDrawSurface7_iface : NULL);
1365 if (FAILED(hr))
1367 return hr;
1369 ddraw_surface4_AddRef(attachment);
1370 ddraw_surface7_Release(&attachment_impl->IDirectDrawSurface7_iface);
1371 return hr;
1373 static HRESULT WINAPI ddraw_surface3_AddAttachedSurface(IDirectDrawSurface3 *iface, IDirectDrawSurface3 *attachment)
1375 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1376 IDirectDrawSurfaceImpl *attach_impl = unsafe_impl_from_IDirectDrawSurface3(attachment);
1377 HRESULT hr;
1379 TRACE("iface %p, attachment %p.\n", iface, attachment);
1381 /* Tests suggest that
1382 * -> offscreen plain surfaces can be attached to other offscreen plain surfaces
1383 * -> offscreen plain surfaces can be attached to primaries
1384 * -> primaries can be attached to offscreen plain surfaces
1385 * -> z buffers can be attached to primaries */
1386 if (This->surface_desc.ddsCaps.dwCaps & (DDSCAPS_PRIMARYSURFACE | DDSCAPS_OFFSCREENPLAIN)
1387 && attach_impl->surface_desc.ddsCaps.dwCaps & (DDSCAPS_PRIMARYSURFACE | DDSCAPS_OFFSCREENPLAIN))
1389 /* Sizes have to match */
1390 if (attach_impl->surface_desc.dwWidth != This->surface_desc.dwWidth
1391 || attach_impl->surface_desc.dwHeight != This->surface_desc.dwHeight)
1393 WARN("Surface sizes do not match.\n");
1394 return DDERR_CANNOTATTACHSURFACE;
1396 /* OK */
1398 else if (This->surface_desc.ddsCaps.dwCaps & (DDSCAPS_PRIMARYSURFACE | DDSCAPS_3DDEVICE)
1399 && attach_impl->surface_desc.ddsCaps.dwCaps & (DDSCAPS_ZBUFFER))
1401 /* OK */
1403 else
1405 WARN("Invalid attachment combination.\n");
1406 return DDERR_CANNOTATTACHSURFACE;
1409 hr = ddraw_surface_attach_surface(This, attach_impl);
1410 if (FAILED(hr))
1412 return hr;
1414 ddraw_surface3_AddRef(attachment);
1415 return hr;
1418 static HRESULT WINAPI ddraw_surface2_AddAttachedSurface(IDirectDrawSurface2 *iface, IDirectDrawSurface2 *attachment)
1420 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1421 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface2(attachment);
1422 HRESULT hr;
1424 TRACE("iface %p, attachment %p.\n", iface, attachment);
1426 hr = ddraw_surface3_AddAttachedSurface(&This->IDirectDrawSurface3_iface,
1427 attachment_impl ? &attachment_impl->IDirectDrawSurface3_iface : NULL);
1428 if (FAILED(hr))
1430 return hr;
1432 ddraw_surface2_AddRef(attachment);
1433 ddraw_surface3_Release(&attachment_impl->IDirectDrawSurface3_iface);
1434 return hr;
1437 static HRESULT WINAPI ddraw_surface1_AddAttachedSurface(IDirectDrawSurface *iface, IDirectDrawSurface *attachment)
1439 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1440 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface(attachment);
1441 HRESULT hr;
1443 TRACE("iface %p, attachment %p.\n", iface, attachment);
1445 hr = ddraw_surface3_AddAttachedSurface(&This->IDirectDrawSurface3_iface,
1446 attachment_impl ? &attachment_impl->IDirectDrawSurface3_iface : NULL);
1447 if (FAILED(hr))
1449 return hr;
1451 ddraw_surface1_AddRef(attachment);
1452 ddraw_surface3_Release(&attachment_impl->IDirectDrawSurface3_iface);
1453 return hr;
1456 /*****************************************************************************
1457 * IDirectDrawSurface7::DeleteAttachedSurface
1459 * Removes a surface from the attachment chain. The surface's refcount
1460 * is decreased by one after it has been removed
1462 * Params:
1463 * Flags: Some flags, not used by this implementation
1464 * Attach: Surface to detach
1466 * Returns:
1467 * DD_OK on success
1468 * DDERR_SURFACENOTATTACHED if the surface isn't attached to
1470 *****************************************************************************/
1471 static HRESULT ddraw_surface_delete_attached_surface(IDirectDrawSurfaceImpl *This,
1472 IDirectDrawSurfaceImpl *Surf)
1474 IDirectDrawSurfaceImpl *Prev = This;
1476 TRACE("surface %p, attachment %p.\n", This, Surf);
1478 EnterCriticalSection(&ddraw_cs);
1479 if (!Surf || (Surf->first_attached != This) || (Surf == This) )
1481 LeaveCriticalSection(&ddraw_cs);
1482 return DDERR_CANNOTDETACHSURFACE;
1485 /* Remove MIPMAPSUBLEVEL if this seemed to be one */
1486 if (This->surface_desc.ddsCaps.dwCaps &
1487 Surf->surface_desc.ddsCaps.dwCaps & DDSCAPS_MIPMAP)
1489 Surf->surface_desc.ddsCaps.dwCaps2 &= ~DDSCAPS2_MIPMAPSUBLEVEL;
1490 /* FIXME: we should probably also subtract from dwMipMapCount of this
1491 * and all parent surfaces */
1494 /* Find the predecessor of the detached surface */
1495 while(Prev)
1497 if(Prev->next_attached == Surf) break;
1498 Prev = Prev->next_attached;
1501 /* There must be a surface, otherwise there's a bug */
1502 assert(Prev != NULL);
1504 /* Unchain the surface */
1505 Prev->next_attached = Surf->next_attached;
1506 Surf->next_attached = NULL;
1507 Surf->first_attached = Surf;
1509 /* Check if the WineD3D depth stencil needs updating */
1510 if(This->ddraw->d3ddevice)
1512 IDirect3DDeviceImpl_UpdateDepthStencil(This->ddraw->d3ddevice);
1514 LeaveCriticalSection(&ddraw_cs);
1515 return DD_OK;
1518 static HRESULT WINAPI ddraw_surface7_DeleteAttachedSurface(IDirectDrawSurface7 *iface,
1519 DWORD flags, IDirectDrawSurface7 *attachment)
1521 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1522 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface7(attachment);
1523 HRESULT hr;
1525 TRACE("iface %p, flags %#x, attachment %p.\n", iface, flags, attachment);
1527 hr = ddraw_surface_delete_attached_surface(This, attachment_impl);
1528 if (FAILED(hr))
1530 return hr;
1532 ddraw_surface7_Release(attachment);
1533 return hr;
1536 static HRESULT WINAPI ddraw_surface4_DeleteAttachedSurface(IDirectDrawSurface4 *iface,
1537 DWORD flags, IDirectDrawSurface4 *attachment)
1539 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1540 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface4(attachment);
1541 HRESULT hr;
1543 TRACE("iface %p, flags %#x, attachment %p.\n", iface, flags, attachment);
1545 hr = ddraw_surface_delete_attached_surface(This, attachment_impl);
1546 if (FAILED(hr))
1548 return hr;
1550 ddraw_surface4_Release(attachment);
1551 return hr;
1554 static HRESULT WINAPI ddraw_surface3_DeleteAttachedSurface(IDirectDrawSurface3 *iface,
1555 DWORD flags, IDirectDrawSurface3 *attachment)
1557 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1558 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface3(attachment);
1559 HRESULT hr;
1560 TRACE("iface %p, flags %#x, attachment %p.\n", iface, flags, attachment);
1562 hr = ddraw_surface_delete_attached_surface(This, attachment_impl);
1563 if (FAILED(hr))
1565 return hr;
1567 ddraw_surface3_Release(attachment);
1568 return hr;
1571 static HRESULT WINAPI ddraw_surface2_DeleteAttachedSurface(IDirectDrawSurface2 *iface,
1572 DWORD flags, IDirectDrawSurface2 *attachment)
1574 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1575 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface2(attachment);
1576 HRESULT hr;
1577 TRACE("iface %p, flags %#x, attachment %p.\n", iface, flags, attachment);
1579 hr = ddraw_surface_delete_attached_surface(This, attachment_impl);
1580 if (FAILED(hr))
1582 return hr;
1584 ddraw_surface2_Release(attachment);
1585 return hr;
1588 static HRESULT WINAPI ddraw_surface1_DeleteAttachedSurface(IDirectDrawSurface *iface,
1589 DWORD flags, IDirectDrawSurface *attachment)
1591 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1592 IDirectDrawSurfaceImpl *attachment_impl = unsafe_impl_from_IDirectDrawSurface(attachment);
1593 HRESULT hr;
1594 TRACE("iface %p, flags %#x, attachment %p.\n", iface, flags, attachment);
1596 hr = ddraw_surface_delete_attached_surface(This, attachment_impl);
1597 if (FAILED(hr))
1599 return hr;
1601 ddraw_surface1_Release(attachment);
1602 return hr;
1605 /*****************************************************************************
1606 * IDirectDrawSurface7::AddOverlayDirtyRect
1608 * "This method is not currently implemented"
1610 * Params:
1611 * Rect: ?
1613 * Returns:
1614 * DDERR_UNSUPPORTED
1616 *****************************************************************************/
1617 static HRESULT WINAPI ddraw_surface7_AddOverlayDirtyRect(IDirectDrawSurface7 *iface, RECT *Rect)
1619 TRACE("iface %p, rect %s.\n", iface, wine_dbgstr_rect(Rect));
1621 return DDERR_UNSUPPORTED; /* unchecked */
1624 static HRESULT WINAPI ddraw_surface4_AddOverlayDirtyRect(IDirectDrawSurface4 *iface, RECT *rect)
1626 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1627 TRACE("iface %p, rect %s.\n", iface, wine_dbgstr_rect(rect));
1629 return ddraw_surface7_AddOverlayDirtyRect(&This->IDirectDrawSurface7_iface, rect);
1632 static HRESULT WINAPI ddraw_surface3_AddOverlayDirtyRect(IDirectDrawSurface3 *iface, RECT *rect)
1634 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1635 TRACE("iface %p, rect %s.\n", iface, wine_dbgstr_rect(rect));
1637 return ddraw_surface7_AddOverlayDirtyRect(&This->IDirectDrawSurface7_iface, rect);
1640 static HRESULT WINAPI ddraw_surface2_AddOverlayDirtyRect(IDirectDrawSurface2 *iface, RECT *rect)
1642 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1643 TRACE("iface %p, rect %s.\n", iface, wine_dbgstr_rect(rect));
1645 return ddraw_surface7_AddOverlayDirtyRect(&This->IDirectDrawSurface7_iface, rect);
1648 static HRESULT WINAPI ddraw_surface1_AddOverlayDirtyRect(IDirectDrawSurface *iface, RECT *rect)
1650 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1651 TRACE("iface %p, rect %s.\n", iface, wine_dbgstr_rect(rect));
1653 return ddraw_surface7_AddOverlayDirtyRect(&This->IDirectDrawSurface7_iface, rect);
1656 /*****************************************************************************
1657 * IDirectDrawSurface7::GetDC
1659 * Returns a GDI device context for the surface
1661 * Params:
1662 * hdc: Address of a HDC variable to store the dc to
1664 * Returns:
1665 * DD_OK on success
1666 * DDERR_INVALIDPARAMS if hdc is NULL
1667 * For details, see IWineD3DSurface::GetDC
1669 *****************************************************************************/
1670 static HRESULT WINAPI ddraw_surface7_GetDC(IDirectDrawSurface7 *iface, HDC *hdc)
1672 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1673 HRESULT hr;
1675 TRACE("iface %p, dc %p.\n", iface, hdc);
1677 if(!hdc)
1678 return DDERR_INVALIDPARAMS;
1680 EnterCriticalSection(&ddraw_cs);
1681 hr = wined3d_surface_getdc(This->wined3d_surface, hdc);
1682 LeaveCriticalSection(&ddraw_cs);
1683 switch(hr)
1685 /* Some, but not all errors set *hdc to NULL. E.g. DCALREADYCREATED does not
1686 * touch *hdc
1688 case WINED3DERR_INVALIDCALL:
1689 if(hdc) *hdc = NULL;
1690 return DDERR_INVALIDPARAMS;
1692 default: return hr;
1696 static HRESULT WINAPI ddraw_surface4_GetDC(IDirectDrawSurface4 *iface, HDC *dc)
1698 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1699 TRACE("iface %p, dc %p.\n", iface, dc);
1701 return ddraw_surface7_GetDC(&This->IDirectDrawSurface7_iface, dc);
1704 static HRESULT WINAPI ddraw_surface3_GetDC(IDirectDrawSurface3 *iface, HDC *dc)
1706 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1707 TRACE("iface %p, dc %p.\n", iface, dc);
1709 return ddraw_surface7_GetDC(&This->IDirectDrawSurface7_iface, dc);
1712 static HRESULT WINAPI ddraw_surface2_GetDC(IDirectDrawSurface2 *iface, HDC *dc)
1714 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1715 TRACE("iface %p, dc %p.\n", iface, dc);
1717 return ddraw_surface7_GetDC(&This->IDirectDrawSurface7_iface, dc);
1720 static HRESULT WINAPI ddraw_surface1_GetDC(IDirectDrawSurface *iface, HDC *dc)
1722 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1723 TRACE("iface %p, dc %p.\n", iface, dc);
1725 return ddraw_surface7_GetDC(&This->IDirectDrawSurface7_iface, dc);
1728 /*****************************************************************************
1729 * IDirectDrawSurface7::ReleaseDC
1731 * Releases the DC that was constructed with GetDC
1733 * Params:
1734 * hdc: HDC to release
1736 * Returns:
1737 * DD_OK on success
1738 * For more details, see IWineD3DSurface::ReleaseDC
1740 *****************************************************************************/
1741 static HRESULT WINAPI ddraw_surface7_ReleaseDC(IDirectDrawSurface7 *iface, HDC hdc)
1743 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1744 HRESULT hr;
1746 TRACE("iface %p, dc %p.\n", iface, hdc);
1748 EnterCriticalSection(&ddraw_cs);
1749 hr = wined3d_surface_releasedc(This->wined3d_surface, hdc);
1750 LeaveCriticalSection(&ddraw_cs);
1751 return hr;
1754 static HRESULT WINAPI ddraw_surface4_ReleaseDC(IDirectDrawSurface4 *iface, HDC dc)
1756 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1757 TRACE("iface %p, dc %p.\n", iface, dc);
1759 return ddraw_surface7_ReleaseDC(&This->IDirectDrawSurface7_iface, dc);
1762 static HRESULT WINAPI ddraw_surface3_ReleaseDC(IDirectDrawSurface3 *iface, HDC dc)
1764 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1765 TRACE("iface %p, dc %p.\n", iface, dc);
1767 return ddraw_surface7_ReleaseDC(&This->IDirectDrawSurface7_iface, dc);
1770 static HRESULT WINAPI ddraw_surface2_ReleaseDC(IDirectDrawSurface2 *iface, HDC dc)
1772 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1773 TRACE("iface %p, dc %p.\n", iface, dc);
1775 return ddraw_surface7_ReleaseDC(&This->IDirectDrawSurface7_iface, dc);
1778 static HRESULT WINAPI ddraw_surface1_ReleaseDC(IDirectDrawSurface *iface, HDC dc)
1780 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1781 TRACE("iface %p, dc %p.\n", iface, dc);
1783 return ddraw_surface7_ReleaseDC(&This->IDirectDrawSurface7_iface, dc);
1786 /*****************************************************************************
1787 * IDirectDrawSurface7::GetCaps
1789 * Returns the surface's caps
1791 * Params:
1792 * Caps: Address to write the caps to
1794 * Returns:
1795 * DD_OK on success
1796 * DDERR_INVALIDPARAMS if Caps is NULL
1798 *****************************************************************************/
1799 static HRESULT WINAPI ddraw_surface7_GetCaps(IDirectDrawSurface7 *iface, DDSCAPS2 *Caps)
1801 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1803 TRACE("iface %p, caps %p.\n", iface, Caps);
1805 if(!Caps)
1806 return DDERR_INVALIDPARAMS;
1808 *Caps = This->surface_desc.ddsCaps;
1809 return DD_OK;
1812 static HRESULT WINAPI ddraw_surface4_GetCaps(IDirectDrawSurface4 *iface, DDSCAPS2 *caps)
1814 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1815 TRACE("iface %p, caps %p.\n", iface, caps);
1817 return ddraw_surface7_GetCaps(&This->IDirectDrawSurface7_iface, caps);
1820 static HRESULT WINAPI ddraw_surface3_GetCaps(IDirectDrawSurface3 *iface, DDSCAPS *caps)
1822 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
1823 DDSCAPS2 caps2;
1824 HRESULT hr;
1826 TRACE("iface %p, caps %p.\n", iface, caps);
1828 hr = ddraw_surface7_GetCaps(&This->IDirectDrawSurface7_iface, &caps2);
1829 if (FAILED(hr)) return hr;
1831 caps->dwCaps = caps2.dwCaps;
1832 return hr;
1835 static HRESULT WINAPI ddraw_surface2_GetCaps(IDirectDrawSurface2 *iface, DDSCAPS *caps)
1837 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
1838 DDSCAPS2 caps2;
1839 HRESULT hr;
1841 TRACE("iface %p, caps %p.\n", iface, caps);
1843 hr = ddraw_surface7_GetCaps(&This->IDirectDrawSurface7_iface, &caps2);
1844 if (FAILED(hr)) return hr;
1846 caps->dwCaps = caps2.dwCaps;
1847 return hr;
1850 static HRESULT WINAPI ddraw_surface1_GetCaps(IDirectDrawSurface *iface, DDSCAPS *caps)
1852 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
1853 DDSCAPS2 caps2;
1854 HRESULT hr;
1856 TRACE("iface %p, caps %p.\n", iface, caps);
1858 hr = ddraw_surface7_GetCaps(&This->IDirectDrawSurface7_iface, &caps2);
1859 if (FAILED(hr)) return hr;
1861 caps->dwCaps = caps2.dwCaps;
1862 return hr;
1865 /*****************************************************************************
1866 * IDirectDrawSurface7::SetPriority
1868 * Sets a texture priority for managed textures.
1870 * Params:
1871 * Priority: The new priority
1873 * Returns:
1874 * DD_OK on success
1875 * For more details, see IWineD3DSurface::SetPriority
1877 *****************************************************************************/
1878 static HRESULT WINAPI ddraw_surface7_SetPriority(IDirectDrawSurface7 *iface, DWORD Priority)
1880 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1881 HRESULT hr;
1883 TRACE("iface %p, priority %u.\n", iface, Priority);
1885 EnterCriticalSection(&ddraw_cs);
1886 hr = wined3d_surface_set_priority(This->wined3d_surface, Priority);
1887 LeaveCriticalSection(&ddraw_cs);
1888 return hr;
1891 /*****************************************************************************
1892 * IDirectDrawSurface7::GetPriority
1894 * Returns the surface's priority
1896 * Params:
1897 * Priority: Address of a variable to write the priority to
1899 * Returns:
1900 * D3D_OK on success
1901 * DDERR_INVALIDPARAMS if Priority == NULL
1902 * For more details, see IWineD3DSurface::GetPriority
1904 *****************************************************************************/
1905 static HRESULT WINAPI ddraw_surface7_GetPriority(IDirectDrawSurface7 *iface, DWORD *Priority)
1907 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1909 TRACE("iface %p, priority %p.\n", iface, Priority);
1911 if(!Priority)
1913 return DDERR_INVALIDPARAMS;
1916 EnterCriticalSection(&ddraw_cs);
1917 *Priority = wined3d_surface_get_priority(This->wined3d_surface);
1918 LeaveCriticalSection(&ddraw_cs);
1919 return DD_OK;
1922 /*****************************************************************************
1923 * IDirectDrawSurface7::SetPrivateData
1925 * Stores some data in the surface that is intended for the application's
1926 * use.
1928 * Params:
1929 * tag: GUID that identifies the data
1930 * Data: Pointer to the private data
1931 * Size: Size of the private data
1932 * Flags: Some flags
1934 * Returns:
1935 * D3D_OK on success
1936 * For more details, see IWineD3DSurface::SetPrivateData
1938 *****************************************************************************/
1939 static HRESULT WINAPI ddraw_surface7_SetPrivateData(IDirectDrawSurface7 *iface,
1940 REFGUID tag, void *Data, DWORD Size, DWORD Flags)
1942 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1943 HRESULT hr;
1945 TRACE("iface %p, tag %s, data %p, data_size %u, flags %#x.\n",
1946 iface, debugstr_guid(tag), Data, Size, Flags);
1948 EnterCriticalSection(&ddraw_cs);
1949 hr = wined3d_surface_set_private_data(This->wined3d_surface, tag, Data, Size, Flags);
1950 LeaveCriticalSection(&ddraw_cs);
1951 switch(hr)
1953 case WINED3DERR_INVALIDCALL: return DDERR_INVALIDPARAMS;
1954 default: return hr;
1958 static HRESULT WINAPI ddraw_surface4_SetPrivateData(IDirectDrawSurface4 *iface,
1959 REFGUID tag, void *data, DWORD size, DWORD flags)
1961 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
1962 TRACE("iface %p, tag %s, data %p, data_size %u, flags %#x.\n",
1963 iface, debugstr_guid(tag), data, size, flags);
1965 return ddraw_surface7_SetPrivateData(&This->IDirectDrawSurface7_iface, tag, data, size, flags);
1968 /*****************************************************************************
1969 * IDirectDrawSurface7::GetPrivateData
1971 * Returns the private data set with IDirectDrawSurface7::SetPrivateData
1973 * Params:
1974 * tag: GUID of the data to return
1975 * Data: Address where to write the data to
1976 * Size: Size of the buffer at Data
1978 * Returns:
1979 * DD_OK on success
1980 * DDERR_INVALIDPARAMS if Data is NULL
1981 * For more details, see IWineD3DSurface::GetPrivateData
1983 *****************************************************************************/
1984 static HRESULT WINAPI ddraw_surface7_GetPrivateData(IDirectDrawSurface7 *iface, REFGUID tag, void *Data, DWORD *Size)
1986 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
1987 HRESULT hr;
1989 TRACE("iface %p, tag %s, data %p, data_size %p.\n",
1990 iface, debugstr_guid(tag), Data, Size);
1992 if(!Data)
1993 return DDERR_INVALIDPARAMS;
1995 EnterCriticalSection(&ddraw_cs);
1996 hr = wined3d_surface_get_private_data(This->wined3d_surface, tag, Data, Size);
1997 LeaveCriticalSection(&ddraw_cs);
1998 return hr;
2001 static HRESULT WINAPI ddraw_surface4_GetPrivateData(IDirectDrawSurface4 *iface, REFGUID tag, void *data, DWORD *size)
2003 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2004 TRACE("iface %p, tag %s, data %p, data_size %p.\n",
2005 iface, debugstr_guid(tag), data, size);
2007 return ddraw_surface7_GetPrivateData(&This->IDirectDrawSurface7_iface, tag, data, size);
2010 /*****************************************************************************
2011 * IDirectDrawSurface7::FreePrivateData
2013 * Frees private data stored in the surface
2015 * Params:
2016 * tag: Tag of the data to free
2018 * Returns:
2019 * D3D_OK on success
2020 * For more details, see IWineD3DSurface::FreePrivateData
2022 *****************************************************************************/
2023 static HRESULT WINAPI ddraw_surface7_FreePrivateData(IDirectDrawSurface7 *iface, REFGUID tag)
2025 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2026 HRESULT hr;
2028 TRACE("iface %p, tag %s.\n", iface, debugstr_guid(tag));
2030 EnterCriticalSection(&ddraw_cs);
2031 hr = wined3d_surface_free_private_data(This->wined3d_surface, tag);
2032 LeaveCriticalSection(&ddraw_cs);
2033 return hr;
2036 static HRESULT WINAPI ddraw_surface4_FreePrivateData(IDirectDrawSurface4 *iface, REFGUID tag)
2038 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2039 TRACE("iface %p, tag %s.\n", iface, debugstr_guid(tag));
2041 return ddraw_surface7_FreePrivateData(&This->IDirectDrawSurface7_iface, tag);
2044 /*****************************************************************************
2045 * IDirectDrawSurface7::PageLock
2047 * Prevents a sysmem surface from being paged out
2049 * Params:
2050 * Flags: Not used, must be 0(unchecked)
2052 * Returns:
2053 * DD_OK, because it's a stub
2055 *****************************************************************************/
2056 static HRESULT WINAPI ddraw_surface7_PageLock(IDirectDrawSurface7 *iface, DWORD Flags)
2058 TRACE("iface %p, flags %#x.\n", iface, Flags);
2060 /* This is Windows memory management related - we don't need this */
2061 return DD_OK;
2064 static HRESULT WINAPI ddraw_surface4_PageLock(IDirectDrawSurface4 *iface, DWORD flags)
2066 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2067 TRACE("iface %p, flags %#x.\n", iface, flags);
2069 return ddraw_surface7_PageLock(&This->IDirectDrawSurface7_iface, flags);
2072 static HRESULT WINAPI ddraw_surface3_PageLock(IDirectDrawSurface3 *iface, DWORD flags)
2074 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2075 TRACE("iface %p, flags %#x.\n", iface, flags);
2077 return ddraw_surface7_PageLock(&This->IDirectDrawSurface7_iface, flags);
2080 static HRESULT WINAPI ddraw_surface2_PageLock(IDirectDrawSurface2 *iface, DWORD flags)
2082 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2083 TRACE("iface %p, flags %#x.\n", iface, flags);
2085 return ddraw_surface7_PageLock(&This->IDirectDrawSurface7_iface, flags);
2088 /*****************************************************************************
2089 * IDirectDrawSurface7::PageUnlock
2091 * Allows a sysmem surface to be paged out
2093 * Params:
2094 * Flags: Not used, must be 0(unchecked)
2096 * Returns:
2097 * DD_OK, because it's a stub
2099 *****************************************************************************/
2100 static HRESULT WINAPI ddraw_surface7_PageUnlock(IDirectDrawSurface7 *iface, DWORD Flags)
2102 TRACE("iface %p, flags %#x.\n", iface, Flags);
2104 return DD_OK;
2107 static HRESULT WINAPI ddraw_surface4_PageUnlock(IDirectDrawSurface4 *iface, DWORD flags)
2109 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2110 TRACE("iface %p, flags %#x.\n", iface, flags);
2112 return ddraw_surface7_PageUnlock(&This->IDirectDrawSurface7_iface, flags);
2115 static HRESULT WINAPI ddraw_surface3_PageUnlock(IDirectDrawSurface3 *iface, DWORD flags)
2117 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2118 TRACE("iface %p, flags %#x.\n", iface, flags);
2120 return ddraw_surface7_PageUnlock(&This->IDirectDrawSurface7_iface, flags);
2123 static HRESULT WINAPI ddraw_surface2_PageUnlock(IDirectDrawSurface2 *iface, DWORD flags)
2125 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2126 TRACE("iface %p, flags %#x.\n", iface, flags);
2128 return ddraw_surface7_PageUnlock(&This->IDirectDrawSurface7_iface, flags);
2131 /*****************************************************************************
2132 * IDirectDrawSurface7::BltBatch
2134 * An unimplemented function
2136 * Params:
2139 * Returns:
2140 * DDERR_UNSUPPORTED
2142 *****************************************************************************/
2143 static HRESULT WINAPI ddraw_surface7_BltBatch(IDirectDrawSurface7 *iface, DDBLTBATCH *Batch, DWORD Count, DWORD Flags)
2145 TRACE("iface %p, batch %p, count %u, flags %#x.\n", iface, Batch, Count, Flags);
2147 /* MSDN: "not currently implemented" */
2148 return DDERR_UNSUPPORTED;
2151 static HRESULT WINAPI ddraw_surface4_BltBatch(IDirectDrawSurface4 *iface, DDBLTBATCH *batch, DWORD count, DWORD flags)
2153 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2154 TRACE("iface %p, batch %p, count %u, flags %#x.\n", iface, batch, count, flags);
2156 return ddraw_surface7_BltBatch(&This->IDirectDrawSurface7_iface, batch, count, flags);
2159 static HRESULT WINAPI ddraw_surface3_BltBatch(IDirectDrawSurface3 *iface, DDBLTBATCH *batch, DWORD count, DWORD flags)
2161 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2162 TRACE("iface %p, batch %p, count %u, flags %#x.\n", iface, batch, count, flags);
2164 return ddraw_surface7_BltBatch(&This->IDirectDrawSurface7_iface, batch, count, flags);
2167 static HRESULT WINAPI ddraw_surface2_BltBatch(IDirectDrawSurface2 *iface, DDBLTBATCH *batch, DWORD count, DWORD flags)
2169 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2170 TRACE("iface %p, batch %p, count %u, flags %#x.\n", iface, batch, count, flags);
2172 return ddraw_surface7_BltBatch(&This->IDirectDrawSurface7_iface, batch, count, flags);
2175 static HRESULT WINAPI ddraw_surface1_BltBatch(IDirectDrawSurface *iface, DDBLTBATCH *batch, DWORD count, DWORD flags)
2177 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2178 TRACE("iface %p, batch %p, count %u, flags %#x.\n", iface, batch, count, flags);
2180 return ddraw_surface7_BltBatch(&This->IDirectDrawSurface7_iface, batch, count, flags);
2183 /*****************************************************************************
2184 * IDirectDrawSurface7::EnumAttachedSurfaces
2186 * Enumerates all surfaces attached to this surface
2188 * Params:
2189 * context: Pointer to pass unmodified to the callback
2190 * cb: Callback function to call for each surface
2192 * Returns:
2193 * DD_OK on success
2194 * DDERR_INVALIDPARAMS if cb is NULL
2196 *****************************************************************************/
2197 static HRESULT WINAPI ddraw_surface7_EnumAttachedSurfaces(IDirectDrawSurface7 *iface,
2198 void *context, LPDDENUMSURFACESCALLBACK7 cb)
2200 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2201 IDirectDrawSurfaceImpl *surf;
2202 DDSURFACEDESC2 desc;
2203 int i;
2205 /* Attached surfaces aren't handled in WineD3D */
2206 TRACE("iface %p, context %p, callback %p.\n", iface, context, cb);
2208 if(!cb)
2209 return DDERR_INVALIDPARAMS;
2211 EnterCriticalSection(&ddraw_cs);
2212 for(i = 0; i < MAX_COMPLEX_ATTACHED; i++)
2214 surf = This->complex_array[i];
2215 if(!surf) break;
2217 ddraw_surface7_AddRef(&surf->IDirectDrawSurface7_iface);
2218 desc = surf->surface_desc;
2219 /* check: != DDENUMRET_OK or == DDENUMRET_CANCEL? */
2220 if (cb(&surf->IDirectDrawSurface7_iface, &desc, context) == DDENUMRET_CANCEL)
2222 LeaveCriticalSection(&ddraw_cs);
2223 return DD_OK;
2227 for (surf = This->next_attached; surf != NULL; surf = surf->next_attached)
2229 ddraw_surface7_AddRef(&surf->IDirectDrawSurface7_iface);
2230 desc = surf->surface_desc;
2231 /* check: != DDENUMRET_OK or == DDENUMRET_CANCEL? */
2232 if (cb(&surf->IDirectDrawSurface7_iface, &desc, context) == DDENUMRET_CANCEL)
2234 LeaveCriticalSection(&ddraw_cs);
2235 return DD_OK;
2239 TRACE(" end of enumeration.\n");
2241 LeaveCriticalSection(&ddraw_cs);
2242 return DD_OK;
2245 struct callback_info2
2247 LPDDENUMSURFACESCALLBACK2 callback;
2248 void *context;
2251 struct callback_info
2253 LPDDENUMSURFACESCALLBACK callback;
2254 void *context;
2257 static HRESULT CALLBACK EnumCallback2(IDirectDrawSurface7 *surface, DDSURFACEDESC2 *surface_desc, void *context)
2259 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(surface);
2260 const struct callback_info2 *info = context;
2262 return info->callback(&This->IDirectDrawSurface4_iface, surface_desc, info->context);
2265 static HRESULT CALLBACK EnumCallback(IDirectDrawSurface7 *surface, DDSURFACEDESC2 *surface_desc, void *context)
2267 IDirectDrawSurfaceImpl *surface_impl = impl_from_IDirectDrawSurface7(surface);
2268 const struct callback_info *info = context;
2270 return info->callback(&surface_impl->IDirectDrawSurface_iface,
2271 (DDSURFACEDESC *)surface_desc, info->context);
2274 static HRESULT WINAPI ddraw_surface4_EnumAttachedSurfaces(IDirectDrawSurface4 *iface,
2275 void *context, LPDDENUMSURFACESCALLBACK2 callback)
2277 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2278 struct callback_info2 info;
2280 TRACE("iface %p, context %p, callback %p.\n", iface, context, callback);
2282 info.callback = callback;
2283 info.context = context;
2285 return ddraw_surface7_EnumAttachedSurfaces(&This->IDirectDrawSurface7_iface,
2286 &info, EnumCallback2);
2289 static HRESULT WINAPI ddraw_surface3_EnumAttachedSurfaces(IDirectDrawSurface3 *iface,
2290 void *context, LPDDENUMSURFACESCALLBACK callback)
2292 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2293 struct callback_info info;
2295 TRACE("iface %p, context %p, callback %p.\n", iface, context, callback);
2297 info.callback = callback;
2298 info.context = context;
2300 return ddraw_surface7_EnumAttachedSurfaces(&This->IDirectDrawSurface7_iface,
2301 &info, EnumCallback);
2304 static HRESULT WINAPI ddraw_surface2_EnumAttachedSurfaces(IDirectDrawSurface2 *iface,
2305 void *context, LPDDENUMSURFACESCALLBACK callback)
2307 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2308 struct callback_info info;
2310 TRACE("iface %p, context %p, callback %p.\n", iface, context, callback);
2312 info.callback = callback;
2313 info.context = context;
2315 return ddraw_surface7_EnumAttachedSurfaces(&This->IDirectDrawSurface7_iface,
2316 &info, EnumCallback);
2319 static HRESULT WINAPI ddraw_surface1_EnumAttachedSurfaces(IDirectDrawSurface *iface,
2320 void *context, LPDDENUMSURFACESCALLBACK callback)
2322 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2323 struct callback_info info;
2325 TRACE("iface %p, context %p, callback %p.\n", iface, context, callback);
2327 info.callback = callback;
2328 info.context = context;
2330 return ddraw_surface7_EnumAttachedSurfaces(&This->IDirectDrawSurface7_iface,
2331 &info, EnumCallback);
2334 /*****************************************************************************
2335 * IDirectDrawSurface7::EnumOverlayZOrders
2337 * "Enumerates the overlay surfaces on the specified destination"
2339 * Params:
2340 * Flags: DDENUMOVERLAYZ_BACKTOFRONT or DDENUMOVERLAYZ_FRONTTOBACK
2341 * context: context to pass back to the callback
2342 * cb: callback function to call for each enumerated surface
2344 * Returns:
2345 * DD_OK, because it's a stub
2347 *****************************************************************************/
2348 static HRESULT WINAPI ddraw_surface7_EnumOverlayZOrders(IDirectDrawSurface7 *iface,
2349 DWORD Flags, void *context, LPDDENUMSURFACESCALLBACK7 cb)
2351 FIXME("iface %p, flags %#x, context %p, callback %p stub!\n", iface, Flags, context, cb);
2353 return DD_OK;
2356 static HRESULT WINAPI ddraw_surface4_EnumOverlayZOrders(IDirectDrawSurface4 *iface,
2357 DWORD flags, void *context, LPDDENUMSURFACESCALLBACK2 callback)
2359 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2360 struct callback_info2 info;
2362 TRACE("iface %p, flags %#x, context %p, callback %p.\n", iface, flags, context, callback);
2364 info.callback = callback;
2365 info.context = context;
2367 return ddraw_surface7_EnumOverlayZOrders(&This->IDirectDrawSurface7_iface,
2368 flags, &info, EnumCallback2);
2371 static HRESULT WINAPI ddraw_surface3_EnumOverlayZOrders(IDirectDrawSurface3 *iface,
2372 DWORD flags, void *context, LPDDENUMSURFACESCALLBACK callback)
2374 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2375 struct callback_info info;
2377 TRACE("iface %p, flags %#x, context %p, callback %p.\n", iface, flags, context, callback);
2379 info.callback = callback;
2380 info.context = context;
2382 return ddraw_surface7_EnumOverlayZOrders(&This->IDirectDrawSurface7_iface,
2383 flags, &info, EnumCallback);
2386 static HRESULT WINAPI ddraw_surface2_EnumOverlayZOrders(IDirectDrawSurface2 *iface,
2387 DWORD flags, void *context, LPDDENUMSURFACESCALLBACK callback)
2389 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2390 struct callback_info info;
2392 TRACE("iface %p, flags %#x, context %p, callback %p.\n", iface, flags, context, callback);
2394 info.callback = callback;
2395 info.context = context;
2397 return ddraw_surface7_EnumOverlayZOrders(&This->IDirectDrawSurface7_iface,
2398 flags, &info, EnumCallback);
2401 static HRESULT WINAPI ddraw_surface1_EnumOverlayZOrders(IDirectDrawSurface *iface,
2402 DWORD flags, void *context, LPDDENUMSURFACESCALLBACK callback)
2404 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2405 struct callback_info info;
2407 TRACE("iface %p, flags %#x, context %p, callback %p.\n", iface, flags, context, callback);
2409 info.callback = callback;
2410 info.context = context;
2412 return ddraw_surface7_EnumOverlayZOrders(&This->IDirectDrawSurface7_iface,
2413 flags, &info, EnumCallback);
2416 /*****************************************************************************
2417 * IDirectDrawSurface7::GetBltStatus
2419 * Returns the blitting status
2421 * Params:
2422 * Flags: DDGBS_CANBLT or DDGBS_ISBLTDONE
2424 * Returns:
2425 * See IWineD3DSurface::Blt
2427 *****************************************************************************/
2428 static HRESULT WINAPI ddraw_surface7_GetBltStatus(IDirectDrawSurface7 *iface, DWORD Flags)
2430 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2431 HRESULT hr;
2433 TRACE("iface %p, flags %#x.\n", iface, Flags);
2435 EnterCriticalSection(&ddraw_cs);
2436 hr = wined3d_surface_get_blt_status(This->wined3d_surface, Flags);
2437 LeaveCriticalSection(&ddraw_cs);
2438 switch(hr)
2440 case WINED3DERR_INVALIDCALL: return DDERR_INVALIDPARAMS;
2441 default: return hr;
2445 static HRESULT WINAPI ddraw_surface4_GetBltStatus(IDirectDrawSurface4 *iface, DWORD flags)
2447 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2448 TRACE("iface %p, flags %#x.\n", iface, flags);
2450 return ddraw_surface7_GetBltStatus(&This->IDirectDrawSurface7_iface, flags);
2453 static HRESULT WINAPI ddraw_surface3_GetBltStatus(IDirectDrawSurface3 *iface, DWORD flags)
2455 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2456 TRACE("iface %p, flags %#x.\n", iface, flags);
2458 return ddraw_surface7_GetBltStatus(&This->IDirectDrawSurface7_iface, flags);
2461 static HRESULT WINAPI ddraw_surface2_GetBltStatus(IDirectDrawSurface2 *iface, DWORD flags)
2463 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2464 TRACE("iface %p, flags %#x.\n", iface, flags);
2466 return ddraw_surface7_GetBltStatus(&This->IDirectDrawSurface7_iface, flags);
2469 static HRESULT WINAPI ddraw_surface1_GetBltStatus(IDirectDrawSurface *iface, DWORD flags)
2471 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2472 TRACE("iface %p, flags %#x.\n", iface, flags);
2474 return ddraw_surface7_GetBltStatus(&This->IDirectDrawSurface7_iface, flags);
2477 /*****************************************************************************
2478 * IDirectDrawSurface7::GetColorKey
2480 * Returns the color key assigned to the surface
2482 * Params:
2483 * Flags: Some flags
2484 * CKey: Address to store the key to
2486 * Returns:
2487 * DD_OK on success
2488 * DDERR_INVALIDPARAMS if CKey is NULL
2490 *****************************************************************************/
2491 static HRESULT WINAPI ddraw_surface7_GetColorKey(IDirectDrawSurface7 *iface, DWORD Flags, DDCOLORKEY *CKey)
2493 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2495 TRACE("iface %p, flags %#x, color_key %p.\n", iface, Flags, CKey);
2497 if(!CKey)
2498 return DDERR_INVALIDPARAMS;
2500 EnterCriticalSection(&ddraw_cs);
2502 switch (Flags)
2504 case DDCKEY_DESTBLT:
2505 if (!(This->surface_desc.dwFlags & DDSD_CKDESTBLT))
2507 LeaveCriticalSection(&ddraw_cs);
2508 return DDERR_NOCOLORKEY;
2510 *CKey = This->surface_desc.ddckCKDestBlt;
2511 break;
2513 case DDCKEY_DESTOVERLAY:
2514 if (!(This->surface_desc.dwFlags & DDSD_CKDESTOVERLAY))
2516 LeaveCriticalSection(&ddraw_cs);
2517 return DDERR_NOCOLORKEY;
2519 *CKey = This->surface_desc.u3.ddckCKDestOverlay;
2520 break;
2522 case DDCKEY_SRCBLT:
2523 if (!(This->surface_desc.dwFlags & DDSD_CKSRCBLT))
2525 LeaveCriticalSection(&ddraw_cs);
2526 return DDERR_NOCOLORKEY;
2528 *CKey = This->surface_desc.ddckCKSrcBlt;
2529 break;
2531 case DDCKEY_SRCOVERLAY:
2532 if (!(This->surface_desc.dwFlags & DDSD_CKSRCOVERLAY))
2534 LeaveCriticalSection(&ddraw_cs);
2535 return DDERR_NOCOLORKEY;
2537 *CKey = This->surface_desc.ddckCKSrcOverlay;
2538 break;
2540 default:
2541 LeaveCriticalSection(&ddraw_cs);
2542 return DDERR_INVALIDPARAMS;
2545 LeaveCriticalSection(&ddraw_cs);
2546 return DD_OK;
2549 static HRESULT WINAPI ddraw_surface4_GetColorKey(IDirectDrawSurface4 *iface, DWORD flags, DDCOLORKEY *color_key)
2551 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2552 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
2554 return ddraw_surface7_GetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
2557 static HRESULT WINAPI ddraw_surface3_GetColorKey(IDirectDrawSurface3 *iface, DWORD flags, DDCOLORKEY *color_key)
2559 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2560 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
2562 return ddraw_surface7_GetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
2565 static HRESULT WINAPI ddraw_surface2_GetColorKey(IDirectDrawSurface2 *iface, DWORD flags, DDCOLORKEY *color_key)
2567 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2568 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
2570 return ddraw_surface7_GetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
2573 static HRESULT WINAPI ddraw_surface1_GetColorKey(IDirectDrawSurface *iface, DWORD flags, DDCOLORKEY *color_key)
2575 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2576 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
2578 return ddraw_surface7_GetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
2581 /*****************************************************************************
2582 * IDirectDrawSurface7::GetFlipStatus
2584 * Returns the flipping status of the surface
2586 * Params:
2587 * Flags: DDGFS_CANFLIP of DDGFS_ISFLIPDONE
2589 * Returns:
2590 * See IWineD3DSurface::GetFlipStatus
2592 *****************************************************************************/
2593 static HRESULT WINAPI ddraw_surface7_GetFlipStatus(IDirectDrawSurface7 *iface, DWORD Flags)
2595 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2596 HRESULT hr;
2598 TRACE("iface %p, flags %#x.\n", iface, Flags);
2600 EnterCriticalSection(&ddraw_cs);
2601 hr = wined3d_surface_get_flip_status(This->wined3d_surface, Flags);
2602 LeaveCriticalSection(&ddraw_cs);
2603 switch(hr)
2605 case WINED3DERR_INVALIDCALL: return DDERR_INVALIDPARAMS;
2606 default: return hr;
2610 static HRESULT WINAPI ddraw_surface4_GetFlipStatus(IDirectDrawSurface4 *iface, DWORD flags)
2612 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2613 TRACE("iface %p, flags %#x.\n", iface, flags);
2615 return ddraw_surface7_GetFlipStatus(&This->IDirectDrawSurface7_iface, flags);
2618 static HRESULT WINAPI ddraw_surface3_GetFlipStatus(IDirectDrawSurface3 *iface, DWORD flags)
2620 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2621 TRACE("iface %p, flags %#x.\n", iface, flags);
2623 return ddraw_surface7_GetFlipStatus(&This->IDirectDrawSurface7_iface, flags);
2626 static HRESULT WINAPI ddraw_surface2_GetFlipStatus(IDirectDrawSurface2 *iface, DWORD flags)
2628 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2629 TRACE("iface %p, flags %#x.\n", iface, flags);
2631 return ddraw_surface7_GetFlipStatus(&This->IDirectDrawSurface7_iface, flags);
2634 static HRESULT WINAPI ddraw_surface1_GetFlipStatus(IDirectDrawSurface *iface, DWORD flags)
2636 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2637 TRACE("iface %p, flags %#x.\n", iface, flags);
2639 return ddraw_surface7_GetFlipStatus(&This->IDirectDrawSurface7_iface, flags);
2642 /*****************************************************************************
2643 * IDirectDrawSurface7::GetOverlayPosition
2645 * Returns the display coordinates of a visible and active overlay surface
2647 * Params:
2651 * Returns:
2652 * DDERR_NOTAOVERLAYSURFACE, because it's a stub
2653 *****************************************************************************/
2654 static HRESULT WINAPI ddraw_surface7_GetOverlayPosition(IDirectDrawSurface7 *iface, LONG *X, LONG *Y)
2656 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2657 HRESULT hr;
2659 TRACE("iface %p, x %p, y %p.\n", iface, X, Y);
2661 EnterCriticalSection(&ddraw_cs);
2662 hr = wined3d_surface_get_overlay_position(This->wined3d_surface, X, Y);
2663 LeaveCriticalSection(&ddraw_cs);
2664 return hr;
2667 static HRESULT WINAPI ddraw_surface4_GetOverlayPosition(IDirectDrawSurface4 *iface, LONG *x, LONG *y)
2669 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2670 TRACE("iface %p, x %p, y %p.\n", iface, x, y);
2672 return ddraw_surface7_GetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
2675 static HRESULT WINAPI ddraw_surface3_GetOverlayPosition(IDirectDrawSurface3 *iface, LONG *x, LONG *y)
2677 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2678 TRACE("iface %p, x %p, y %p.\n", iface, x, y);
2680 return ddraw_surface7_GetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
2683 static HRESULT WINAPI ddraw_surface2_GetOverlayPosition(IDirectDrawSurface2 *iface, LONG *x, LONG *y)
2685 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2686 TRACE("iface %p, x %p, y %p.\n", iface, x, y);
2688 return ddraw_surface7_GetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
2691 static HRESULT WINAPI ddraw_surface1_GetOverlayPosition(IDirectDrawSurface *iface, LONG *x, LONG *y)
2693 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2694 TRACE("iface %p, x %p, y %p.\n", iface, x, y);
2696 return ddraw_surface7_GetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
2699 /*****************************************************************************
2700 * IDirectDrawSurface7::GetPixelFormat
2702 * Returns the pixel format of the Surface
2704 * Params:
2705 * PixelFormat: Pointer to a DDPIXELFORMAT structure to which the pixel
2706 * format should be written
2708 * Returns:
2709 * DD_OK on success
2710 * DDERR_INVALIDPARAMS if PixelFormat is NULL
2712 *****************************************************************************/
2713 static HRESULT WINAPI ddraw_surface7_GetPixelFormat(IDirectDrawSurface7 *iface, DDPIXELFORMAT *PixelFormat)
2715 /* What is DDERR_INVALIDSURFACETYPE for here? */
2716 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2718 TRACE("iface %p, pixel_format %p.\n", iface, PixelFormat);
2720 if(!PixelFormat)
2721 return DDERR_INVALIDPARAMS;
2723 EnterCriticalSection(&ddraw_cs);
2724 DD_STRUCT_COPY_BYSIZE(PixelFormat,&This->surface_desc.u4.ddpfPixelFormat);
2725 LeaveCriticalSection(&ddraw_cs);
2727 return DD_OK;
2730 static HRESULT WINAPI ddraw_surface4_GetPixelFormat(IDirectDrawSurface4 *iface, DDPIXELFORMAT *pixel_format)
2732 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2733 TRACE("iface %p, pixel_format %p.\n", iface, pixel_format);
2735 return ddraw_surface7_GetPixelFormat(&This->IDirectDrawSurface7_iface, pixel_format);
2738 static HRESULT WINAPI ddraw_surface3_GetPixelFormat(IDirectDrawSurface3 *iface, DDPIXELFORMAT *pixel_format)
2740 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2741 TRACE("iface %p, pixel_format %p.\n", iface, pixel_format);
2743 return ddraw_surface7_GetPixelFormat(&This->IDirectDrawSurface7_iface, pixel_format);
2746 static HRESULT WINAPI ddraw_surface2_GetPixelFormat(IDirectDrawSurface2 *iface, DDPIXELFORMAT *pixel_format)
2748 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2749 TRACE("iface %p, pixel_format %p.\n", iface, pixel_format);
2751 return ddraw_surface7_GetPixelFormat(&This->IDirectDrawSurface7_iface, pixel_format);
2754 static HRESULT WINAPI ddraw_surface1_GetPixelFormat(IDirectDrawSurface *iface, DDPIXELFORMAT *pixel_format)
2756 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2757 TRACE("iface %p, pixel_format %p.\n", iface, pixel_format);
2759 return ddraw_surface7_GetPixelFormat(&This->IDirectDrawSurface7_iface, pixel_format);
2762 /*****************************************************************************
2763 * IDirectDrawSurface7::GetSurfaceDesc
2765 * Returns the description of this surface
2767 * Params:
2768 * DDSD: Address of a DDSURFACEDESC2 structure that is to be filled with the
2769 * surface desc
2771 * Returns:
2772 * DD_OK on success
2773 * DDERR_INVALIDPARAMS if DDSD is NULL
2775 *****************************************************************************/
2776 static HRESULT WINAPI ddraw_surface7_GetSurfaceDesc(IDirectDrawSurface7 *iface, DDSURFACEDESC2 *DDSD)
2778 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2780 TRACE("iface %p, surface_desc %p.\n", iface, DDSD);
2782 if(!DDSD)
2783 return DDERR_INVALIDPARAMS;
2785 if (DDSD->dwSize != sizeof(DDSURFACEDESC2))
2787 WARN("Incorrect struct size %d, returning DDERR_INVALIDPARAMS\n",DDSD->dwSize);
2788 return DDERR_INVALIDPARAMS;
2791 EnterCriticalSection(&ddraw_cs);
2792 DD_STRUCT_COPY_BYSIZE(DDSD,&This->surface_desc);
2793 TRACE("Returning surface desc:\n");
2794 if (TRACE_ON(ddraw)) DDRAW_dump_surface_desc(DDSD);
2796 LeaveCriticalSection(&ddraw_cs);
2797 return DD_OK;
2800 static HRESULT WINAPI ddraw_surface4_GetSurfaceDesc(IDirectDrawSurface4 *iface, DDSURFACEDESC2 *DDSD)
2802 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2803 TRACE("iface %p, surface_desc %p.\n", iface, DDSD);
2805 return ddraw_surface7_GetSurfaceDesc(&This->IDirectDrawSurface7_iface, DDSD);
2808 static HRESULT WINAPI ddraw_surface3_GetSurfaceDesc(IDirectDrawSurface3 *iface, DDSURFACEDESC *surface_desc)
2810 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2812 TRACE("iface %p, surface_desc %p.\n", iface, surface_desc);
2814 if (!surface_desc) return DDERR_INVALIDPARAMS;
2816 if (surface_desc->dwSize != sizeof(DDSURFACEDESC))
2818 WARN("Incorrect structure size %u, returning DDERR_INVALIDPARAMS.\n", surface_desc->dwSize);
2819 return DDERR_INVALIDPARAMS;
2822 EnterCriticalSection(&ddraw_cs);
2823 DD_STRUCT_COPY_BYSIZE(surface_desc, (DDSURFACEDESC *)&This->surface_desc);
2824 TRACE("Returning surface desc:\n");
2825 if (TRACE_ON(ddraw))
2827 /* DDRAW_dump_surface_desc handles the smaller size */
2828 DDRAW_dump_surface_desc((DDSURFACEDESC2 *)surface_desc);
2831 LeaveCriticalSection(&ddraw_cs);
2832 return DD_OK;
2835 static HRESULT WINAPI ddraw_surface2_GetSurfaceDesc(IDirectDrawSurface2 *iface, DDSURFACEDESC *DDSD)
2837 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2838 TRACE("iface %p, surface_desc %p.\n", iface, DDSD);
2840 return ddraw_surface3_GetSurfaceDesc(&This->IDirectDrawSurface3_iface, DDSD);
2843 static HRESULT WINAPI ddraw_surface1_GetSurfaceDesc(IDirectDrawSurface *iface, DDSURFACEDESC *DDSD)
2845 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2846 TRACE("iface %p, surface_desc %p.\n", iface, DDSD);
2848 return ddraw_surface3_GetSurfaceDesc(&This->IDirectDrawSurface3_iface, DDSD);
2851 /*****************************************************************************
2852 * IDirectDrawSurface7::Initialize
2854 * Initializes the surface. This is a no-op in Wine
2856 * Params:
2857 * DD: Pointer to an DirectDraw interface
2858 * DDSD: Surface description for initialization
2860 * Returns:
2861 * DDERR_ALREADYINITIALIZED
2863 *****************************************************************************/
2864 static HRESULT WINAPI ddraw_surface7_Initialize(IDirectDrawSurface7 *iface,
2865 IDirectDraw *ddraw, DDSURFACEDESC2 *surface_desc)
2867 TRACE("iface %p, ddraw %p, surface_desc %p.\n", iface, ddraw, surface_desc);
2869 return DDERR_ALREADYINITIALIZED;
2872 static HRESULT WINAPI ddraw_surface4_Initialize(IDirectDrawSurface4 *iface,
2873 IDirectDraw *ddraw, DDSURFACEDESC2 *surface_desc)
2875 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2876 TRACE("iface %p, ddraw %p, surface_desc %p.\n", iface, ddraw, surface_desc);
2878 return ddraw_surface7_Initialize(&This->IDirectDrawSurface7_iface,
2879 ddraw, surface_desc);
2882 static HRESULT WINAPI ddraw_surface3_Initialize(IDirectDrawSurface3 *iface,
2883 IDirectDraw *ddraw, DDSURFACEDESC *surface_desc)
2885 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2886 TRACE("iface %p, ddraw %p, surface_desc %p.\n", iface, ddraw, surface_desc);
2888 return ddraw_surface7_Initialize(&This->IDirectDrawSurface7_iface,
2889 ddraw, (DDSURFACEDESC2 *)surface_desc);
2892 static HRESULT WINAPI ddraw_surface2_Initialize(IDirectDrawSurface2 *iface,
2893 IDirectDraw *ddraw, DDSURFACEDESC *surface_desc)
2895 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2896 TRACE("iface %p, ddraw %p, surface_desc %p.\n", iface, ddraw, surface_desc);
2898 return ddraw_surface7_Initialize(&This->IDirectDrawSurface7_iface,
2899 ddraw, (DDSURFACEDESC2 *)surface_desc);
2902 static HRESULT WINAPI ddraw_surface1_Initialize(IDirectDrawSurface *iface,
2903 IDirectDraw *ddraw, DDSURFACEDESC *surface_desc)
2905 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
2906 TRACE("iface %p, ddraw %p, surface_desc %p.\n", iface, ddraw, surface_desc);
2908 return ddraw_surface7_Initialize(&This->IDirectDrawSurface7_iface,
2909 ddraw, (DDSURFACEDESC2 *)surface_desc);
2912 /*****************************************************************************
2913 * IDirect3DTexture1::Initialize
2915 * The sdk says it's not implemented
2917 * Params:
2920 * Returns
2921 * DDERR_UNSUPPORTED
2923 *****************************************************************************/
2924 static HRESULT WINAPI d3d_texture1_Initialize(IDirect3DTexture *iface,
2925 IDirect3DDevice *device, IDirectDrawSurface *surface)
2927 TRACE("iface %p, device %p, surface %p.\n", iface, device, surface);
2929 return DDERR_UNSUPPORTED; /* Unchecked */
2932 /*****************************************************************************
2933 * IDirectDrawSurface7::IsLost
2935 * Checks if the surface is lost
2937 * Returns:
2938 * DD_OK, if the surface is usable
2939 * DDERR_ISLOST if the surface is lost
2940 * See IWineD3DSurface::IsLost for more details
2942 *****************************************************************************/
2943 static HRESULT WINAPI ddraw_surface7_IsLost(IDirectDrawSurface7 *iface)
2945 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
2946 HRESULT hr;
2948 TRACE("iface %p.\n", iface);
2950 EnterCriticalSection(&ddraw_cs);
2951 /* We lose the surface if the implementation was changed */
2952 if(This->ImplType != This->ddraw->ImplType)
2954 /* But this shouldn't happen. When we change the implementation,
2955 * all surfaces are re-created automatically, and their content
2956 * is copied
2958 ERR(" (%p) Implementation was changed from %d to %d\n", This, This->ImplType, This->ddraw->ImplType);
2959 LeaveCriticalSection(&ddraw_cs);
2960 return DDERR_SURFACELOST;
2963 hr = wined3d_surface_is_lost(This->wined3d_surface);
2964 LeaveCriticalSection(&ddraw_cs);
2965 switch(hr)
2967 /* D3D8 and 9 loose full devices, thus there's only a DEVICELOST error.
2968 * WineD3D uses the same error for surfaces
2970 case WINED3DERR_DEVICELOST: return DDERR_SURFACELOST;
2971 default: return hr;
2975 static HRESULT WINAPI ddraw_surface4_IsLost(IDirectDrawSurface4 *iface)
2977 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
2978 TRACE("iface %p.\n", iface);
2980 return ddraw_surface7_IsLost(&This->IDirectDrawSurface7_iface);
2983 static HRESULT WINAPI ddraw_surface3_IsLost(IDirectDrawSurface3 *iface)
2985 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
2986 TRACE("iface %p.\n", iface);
2988 return ddraw_surface7_IsLost(&This->IDirectDrawSurface7_iface);
2991 static HRESULT WINAPI ddraw_surface2_IsLost(IDirectDrawSurface2 *iface)
2993 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
2994 TRACE("iface %p.\n", iface);
2996 return ddraw_surface7_IsLost(&This->IDirectDrawSurface7_iface);
2999 static HRESULT WINAPI ddraw_surface1_IsLost(IDirectDrawSurface *iface)
3001 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3002 TRACE("iface %p.\n", iface);
3004 return ddraw_surface7_IsLost(&This->IDirectDrawSurface7_iface);
3007 /*****************************************************************************
3008 * IDirectDrawSurface7::Restore
3010 * Restores a lost surface. This makes the surface usable again, but
3011 * doesn't reload its old contents
3013 * Returns:
3014 * DD_OK on success
3015 * See IWineD3DSurface::Restore for more details
3017 *****************************************************************************/
3018 static HRESULT WINAPI ddraw_surface7_Restore(IDirectDrawSurface7 *iface)
3020 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3021 HRESULT hr;
3023 TRACE("iface %p.\n", iface);
3025 EnterCriticalSection(&ddraw_cs);
3026 if(This->ImplType != This->ddraw->ImplType)
3028 /* Call the recreation callback. Make sure to AddRef first */
3029 IDirectDrawSurface_AddRef(iface);
3030 ddraw_recreate_surfaces_cb(iface, &This->surface_desc, NULL /* Not needed */);
3032 hr = wined3d_surface_restore(This->wined3d_surface);
3033 LeaveCriticalSection(&ddraw_cs);
3034 return hr;
3037 static HRESULT WINAPI ddraw_surface4_Restore(IDirectDrawSurface4 *iface)
3039 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3040 TRACE("iface %p.\n", iface);
3042 return ddraw_surface7_Restore(&This->IDirectDrawSurface7_iface);
3045 static HRESULT WINAPI ddraw_surface3_Restore(IDirectDrawSurface3 *iface)
3047 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3048 TRACE("iface %p.\n", iface);
3050 return ddraw_surface7_Restore(&This->IDirectDrawSurface7_iface);
3053 static HRESULT WINAPI ddraw_surface2_Restore(IDirectDrawSurface2 *iface)
3055 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3056 TRACE("iface %p.\n", iface);
3058 return ddraw_surface7_Restore(&This->IDirectDrawSurface7_iface);
3061 static HRESULT WINAPI ddraw_surface1_Restore(IDirectDrawSurface *iface)
3063 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3064 TRACE("iface %p.\n", iface);
3066 return ddraw_surface7_Restore(&This->IDirectDrawSurface7_iface);
3069 /*****************************************************************************
3070 * IDirectDrawSurface7::SetOverlayPosition
3072 * Changes the display coordinates of an overlay surface
3074 * Params:
3075 * X:
3076 * Y:
3078 * Returns:
3079 * DDERR_NOTAOVERLAYSURFACE, because we don't support overlays right now
3080 *****************************************************************************/
3081 static HRESULT WINAPI ddraw_surface7_SetOverlayPosition(IDirectDrawSurface7 *iface, LONG X, LONG Y)
3083 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3084 HRESULT hr;
3086 TRACE("iface %p, x %d, y %d.\n", iface, X, Y);
3088 EnterCriticalSection(&ddraw_cs);
3089 hr = wined3d_surface_set_overlay_position(This->wined3d_surface, X, Y);
3090 LeaveCriticalSection(&ddraw_cs);
3091 return hr;
3094 static HRESULT WINAPI ddraw_surface4_SetOverlayPosition(IDirectDrawSurface4 *iface, LONG x, LONG y)
3096 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3097 TRACE("iface %p, x %d, y %d.\n", iface, x, y);
3099 return ddraw_surface7_SetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
3102 static HRESULT WINAPI ddraw_surface3_SetOverlayPosition(IDirectDrawSurface3 *iface, LONG x, LONG y)
3104 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3105 TRACE("iface %p, x %d, y %d.\n", iface, x, y);
3107 return ddraw_surface7_SetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
3110 static HRESULT WINAPI ddraw_surface2_SetOverlayPosition(IDirectDrawSurface2 *iface, LONG x, LONG y)
3112 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3113 TRACE("iface %p, x %d, y %d.\n", iface, x, y);
3115 return ddraw_surface7_SetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
3118 static HRESULT WINAPI ddraw_surface1_SetOverlayPosition(IDirectDrawSurface *iface, LONG x, LONG y)
3120 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3121 TRACE("iface %p, x %d, y %d.\n", iface, x, y);
3123 return ddraw_surface7_SetOverlayPosition(&This->IDirectDrawSurface7_iface, x, y);
3126 /*****************************************************************************
3127 * IDirectDrawSurface7::UpdateOverlay
3129 * Modifies the attributes of an overlay surface.
3131 * Params:
3132 * SrcRect: The section of the source being used for the overlay
3133 * DstSurface: Address of the surface that is overlaid
3134 * DstRect: Place of the overlay
3135 * Flags: some DDOVER_* flags
3137 * Returns:
3138 * DDERR_UNSUPPORTED, because we don't support overlays
3140 *****************************************************************************/
3141 static HRESULT WINAPI ddraw_surface7_UpdateOverlay(IDirectDrawSurface7 *iface, RECT *SrcRect,
3142 IDirectDrawSurface7 *DstSurface, RECT *DstRect, DWORD Flags, DDOVERLAYFX *FX)
3144 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3145 IDirectDrawSurfaceImpl *Dst = unsafe_impl_from_IDirectDrawSurface7(DstSurface);
3146 HRESULT hr;
3148 TRACE("iface %p, src_rect %s, dst_surface %p, dst_rect %s, flags %#x, fx %p.\n",
3149 iface, wine_dbgstr_rect(SrcRect), DstSurface, wine_dbgstr_rect(DstRect), Flags, FX);
3151 EnterCriticalSection(&ddraw_cs);
3152 hr = wined3d_surface_update_overlay(This->wined3d_surface, SrcRect,
3153 Dst ? Dst->wined3d_surface : NULL, DstRect, Flags, (WINEDDOVERLAYFX *)FX);
3154 LeaveCriticalSection(&ddraw_cs);
3155 switch(hr) {
3156 case WINED3DERR_INVALIDCALL: return DDERR_INVALIDPARAMS;
3157 case WINEDDERR_NOTAOVERLAYSURFACE: return DDERR_NOTAOVERLAYSURFACE;
3158 case WINEDDERR_OVERLAYNOTVISIBLE: return DDERR_OVERLAYNOTVISIBLE;
3159 default:
3160 return hr;
3164 static HRESULT WINAPI ddraw_surface4_UpdateOverlay(IDirectDrawSurface4 *iface, RECT *src_rect,
3165 IDirectDrawSurface4 *dst_surface, RECT *dst_rect, DWORD flags, DDOVERLAYFX *fx)
3167 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3168 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface4(dst_surface);
3169 TRACE("iface %p, src_rect %s, dst_surface %p, dst_rect %s, flags %#x, fx %p.\n",
3170 iface, wine_dbgstr_rect(src_rect), dst_surface, wine_dbgstr_rect(dst_rect), flags, fx);
3172 return ddraw_surface7_UpdateOverlay(&This->IDirectDrawSurface7_iface, src_rect,
3173 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, dst_rect, flags, fx);
3176 static HRESULT WINAPI ddraw_surface3_UpdateOverlay(IDirectDrawSurface3 *iface, RECT *src_rect,
3177 IDirectDrawSurface3 *dst_surface, RECT *dst_rect, DWORD flags, DDOVERLAYFX *fx)
3179 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3180 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface3(dst_surface);
3181 TRACE("iface %p, src_rect %s, dst_surface %p, dst_rect %s, flags %#x, fx %p.\n",
3182 iface, wine_dbgstr_rect(src_rect), dst_surface, wine_dbgstr_rect(dst_rect), flags, fx);
3184 return ddraw_surface7_UpdateOverlay(&This->IDirectDrawSurface7_iface, src_rect,
3185 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, dst_rect, flags, fx);
3188 static HRESULT WINAPI ddraw_surface2_UpdateOverlay(IDirectDrawSurface2 *iface, RECT *src_rect,
3189 IDirectDrawSurface2 *dst_surface, RECT *dst_rect, DWORD flags, DDOVERLAYFX *fx)
3191 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3192 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface2(dst_surface);
3193 TRACE("iface %p, src_rect %s, dst_surface %p, dst_rect %s, flags %#x, fx %p.\n",
3194 iface, wine_dbgstr_rect(src_rect), dst_surface, wine_dbgstr_rect(dst_rect), flags, fx);
3196 return ddraw_surface7_UpdateOverlay(&This->IDirectDrawSurface7_iface, src_rect,
3197 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, dst_rect, flags, fx);
3200 static HRESULT WINAPI ddraw_surface1_UpdateOverlay(IDirectDrawSurface *iface, RECT *src_rect,
3201 IDirectDrawSurface *dst_surface, RECT *dst_rect, DWORD flags, DDOVERLAYFX *fx)
3203 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3204 IDirectDrawSurfaceImpl *dst_impl = unsafe_impl_from_IDirectDrawSurface(dst_surface);
3205 TRACE("iface %p, src_rect %s, dst_surface %p, dst_rect %s, flags %#x, fx %p.\n",
3206 iface, wine_dbgstr_rect(src_rect), dst_surface, wine_dbgstr_rect(dst_rect), flags, fx);
3208 return ddraw_surface7_UpdateOverlay(&This->IDirectDrawSurface7_iface, src_rect,
3209 dst_impl ? &dst_impl->IDirectDrawSurface7_iface : NULL, dst_rect, flags, fx);
3212 /*****************************************************************************
3213 * IDirectDrawSurface7::UpdateOverlayDisplay
3215 * The DX7 sdk says that it's not implemented
3217 * Params:
3218 * Flags: ?
3220 * Returns: DDERR_UNSUPPORTED, because we don't support overlays
3222 *****************************************************************************/
3223 static HRESULT WINAPI ddraw_surface7_UpdateOverlayDisplay(IDirectDrawSurface7 *iface, DWORD Flags)
3225 TRACE("iface %p, flags %#x.\n", iface, Flags);
3227 return DDERR_UNSUPPORTED;
3230 static HRESULT WINAPI ddraw_surface4_UpdateOverlayDisplay(IDirectDrawSurface4 *iface, DWORD flags)
3232 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3233 TRACE("iface %p, flags %#x.\n", iface, flags);
3235 return ddraw_surface7_UpdateOverlayDisplay(&This->IDirectDrawSurface7_iface, flags);
3238 static HRESULT WINAPI ddraw_surface3_UpdateOverlayDisplay(IDirectDrawSurface3 *iface, DWORD flags)
3240 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3241 TRACE("iface %p, flags %#x.\n", iface, flags);
3243 return ddraw_surface7_UpdateOverlayDisplay(&This->IDirectDrawSurface7_iface, flags);
3246 static HRESULT WINAPI ddraw_surface2_UpdateOverlayDisplay(IDirectDrawSurface2 *iface, DWORD flags)
3248 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3249 TRACE("iface %p, flags %#x.\n", iface, flags);
3251 return ddraw_surface7_UpdateOverlayDisplay(&This->IDirectDrawSurface7_iface, flags);
3254 static HRESULT WINAPI ddraw_surface1_UpdateOverlayDisplay(IDirectDrawSurface *iface, DWORD flags)
3256 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3257 TRACE("iface %p, flags %#x.\n", iface, flags);
3259 return ddraw_surface7_UpdateOverlayDisplay(&This->IDirectDrawSurface7_iface, flags);
3262 /*****************************************************************************
3263 * IDirectDrawSurface7::UpdateOverlayZOrder
3265 * Sets an overlay's Z order
3267 * Params:
3268 * Flags: DDOVERZ_* flags
3269 * DDSRef: Defines the relative position in the overlay chain
3271 * Returns:
3272 * DDERR_NOTOVERLAYSURFACE, because we don't support overlays
3274 *****************************************************************************/
3275 static HRESULT WINAPI ddraw_surface7_UpdateOverlayZOrder(IDirectDrawSurface7 *iface,
3276 DWORD Flags, IDirectDrawSurface7 *DDSRef)
3278 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3279 IDirectDrawSurfaceImpl *Ref = unsafe_impl_from_IDirectDrawSurface7(DDSRef);
3280 HRESULT hr;
3282 TRACE("iface %p, flags %#x, reference %p.\n", iface, Flags, DDSRef);
3284 EnterCriticalSection(&ddraw_cs);
3285 hr = wined3d_surface_update_overlay_z_order(This->wined3d_surface,
3286 Flags, Ref ? Ref->wined3d_surface : NULL);
3287 LeaveCriticalSection(&ddraw_cs);
3288 return hr;
3291 static HRESULT WINAPI ddraw_surface4_UpdateOverlayZOrder(IDirectDrawSurface4 *iface,
3292 DWORD flags, IDirectDrawSurface4 *reference)
3294 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3295 IDirectDrawSurfaceImpl *reference_impl = unsafe_impl_from_IDirectDrawSurface4(reference);
3296 TRACE("iface %p, flags %#x, reference %p.\n", iface, flags, reference);
3298 return ddraw_surface7_UpdateOverlayZOrder(&This->IDirectDrawSurface7_iface, flags,
3299 reference_impl ? &reference_impl->IDirectDrawSurface7_iface : NULL);
3302 static HRESULT WINAPI ddraw_surface3_UpdateOverlayZOrder(IDirectDrawSurface3 *iface,
3303 DWORD flags, IDirectDrawSurface3 *reference)
3305 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3306 IDirectDrawSurfaceImpl *reference_impl = unsafe_impl_from_IDirectDrawSurface3(reference);
3307 TRACE("iface %p, flags %#x, reference %p.\n", iface, flags, reference);
3309 return ddraw_surface7_UpdateOverlayZOrder(&This->IDirectDrawSurface7_iface, flags,
3310 reference_impl ? &reference_impl->IDirectDrawSurface7_iface : NULL);
3313 static HRESULT WINAPI ddraw_surface2_UpdateOverlayZOrder(IDirectDrawSurface2 *iface,
3314 DWORD flags, IDirectDrawSurface2 *reference)
3316 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3317 IDirectDrawSurfaceImpl *reference_impl = unsafe_impl_from_IDirectDrawSurface2(reference);
3318 TRACE("iface %p, flags %#x, reference %p.\n", iface, flags, reference);
3320 return ddraw_surface7_UpdateOverlayZOrder(&This->IDirectDrawSurface7_iface, flags,
3321 reference_impl ? &reference_impl->IDirectDrawSurface7_iface : NULL);
3324 static HRESULT WINAPI ddraw_surface1_UpdateOverlayZOrder(IDirectDrawSurface *iface,
3325 DWORD flags, IDirectDrawSurface *reference)
3327 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3328 IDirectDrawSurfaceImpl *reference_impl = unsafe_impl_from_IDirectDrawSurface(reference);
3329 TRACE("iface %p, flags %#x, reference %p.\n", iface, flags, reference);
3331 return ddraw_surface7_UpdateOverlayZOrder(&This->IDirectDrawSurface7_iface, flags,
3332 reference_impl ? &reference_impl->IDirectDrawSurface7_iface : NULL);
3335 /*****************************************************************************
3336 * IDirectDrawSurface7::GetDDInterface
3338 * Returns the IDirectDraw7 interface pointer of the DirectDraw object this
3339 * surface belongs to
3341 * Params:
3342 * DD: Address to write the interface pointer to
3344 * Returns:
3345 * DD_OK on success
3346 * DDERR_INVALIDPARAMS if DD is NULL
3348 *****************************************************************************/
3349 static HRESULT WINAPI ddraw_surface7_GetDDInterface(IDirectDrawSurface7 *iface, void **DD)
3351 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3353 TRACE("iface %p, ddraw %p.\n", iface, DD);
3355 if(!DD)
3356 return DDERR_INVALIDPARAMS;
3358 switch(This->version)
3360 case 7:
3361 *DD = &This->ddraw->IDirectDraw7_iface;
3362 break;
3364 case 4:
3365 *DD = &This->ddraw->IDirectDraw4_iface;
3366 break;
3368 case 2:
3369 *DD = &This->ddraw->IDirectDraw2_iface;
3370 break;
3372 case 1:
3373 *DD = &This->ddraw->IDirectDraw_iface;
3374 break;
3377 IUnknown_AddRef((IUnknown *)*DD);
3379 return DD_OK;
3382 static HRESULT WINAPI ddraw_surface4_GetDDInterface(IDirectDrawSurface4 *iface, void **ddraw)
3384 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3385 TRACE("iface %p, ddraw %p.\n", iface, ddraw);
3387 return ddraw_surface7_GetDDInterface(&This->IDirectDrawSurface7_iface, ddraw);
3390 static HRESULT WINAPI ddraw_surface3_GetDDInterface(IDirectDrawSurface3 *iface, void **ddraw)
3392 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3393 TRACE("iface %p, ddraw %p.\n", iface, ddraw);
3395 return ddraw_surface7_GetDDInterface(&This->IDirectDrawSurface7_iface, ddraw);
3398 static HRESULT WINAPI ddraw_surface2_GetDDInterface(IDirectDrawSurface2 *iface, void **ddraw)
3400 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3401 TRACE("iface %p, ddraw %p.\n", iface, ddraw);
3403 return ddraw_surface7_GetDDInterface(&This->IDirectDrawSurface7_iface, ddraw);
3406 /* This seems also windows implementation specific - I don't think WineD3D needs this */
3407 static HRESULT WINAPI ddraw_surface7_ChangeUniquenessValue(IDirectDrawSurface7 *iface)
3409 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3410 volatile IDirectDrawSurfaceImpl* vThis = This;
3412 TRACE("iface %p.\n", iface);
3414 EnterCriticalSection(&ddraw_cs);
3415 /* A uniqueness value of 0 is apparently special.
3416 * This needs to be checked.
3417 * TODO: Write tests for this code and check if the volatile, interlocked stuff is really needed
3419 while (1) {
3420 DWORD old_uniqueness_value = vThis->uniqueness_value;
3421 DWORD new_uniqueness_value = old_uniqueness_value+1;
3423 if (old_uniqueness_value == 0) break;
3424 if (new_uniqueness_value == 0) new_uniqueness_value = 1;
3426 if (InterlockedCompareExchange((LONG*)&vThis->uniqueness_value,
3427 old_uniqueness_value,
3428 new_uniqueness_value)
3429 == old_uniqueness_value)
3430 break;
3433 LeaveCriticalSection(&ddraw_cs);
3434 return DD_OK;
3437 static HRESULT WINAPI ddraw_surface4_ChangeUniquenessValue(IDirectDrawSurface4 *iface)
3439 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3440 TRACE("iface %p.\n", iface);
3442 return ddraw_surface7_ChangeUniquenessValue(&This->IDirectDrawSurface7_iface);
3445 static HRESULT WINAPI ddraw_surface7_GetUniquenessValue(IDirectDrawSurface7 *iface, DWORD *pValue)
3447 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3449 TRACE("iface %p, value %p.\n", iface, pValue);
3451 EnterCriticalSection(&ddraw_cs);
3452 *pValue = This->uniqueness_value;
3453 LeaveCriticalSection(&ddraw_cs);
3454 return DD_OK;
3457 static HRESULT WINAPI ddraw_surface4_GetUniquenessValue(IDirectDrawSurface4 *iface, DWORD *pValue)
3459 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3460 TRACE("iface %p, value %p.\n", iface, pValue);
3462 return ddraw_surface7_GetUniquenessValue(&This->IDirectDrawSurface7_iface, pValue);
3465 /*****************************************************************************
3466 * IDirectDrawSurface7::SetLOD
3468 * Sets the level of detail of a texture
3470 * Params:
3471 * MaxLOD: LOD to set
3473 * Returns:
3474 * DD_OK on success
3475 * DDERR_INVALIDOBJECT if the surface is invalid for this method
3477 *****************************************************************************/
3478 static HRESULT WINAPI ddraw_surface7_SetLOD(IDirectDrawSurface7 *iface, DWORD MaxLOD)
3480 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3481 HRESULT hr;
3483 TRACE("iface %p, lod %u.\n", iface, MaxLOD);
3485 EnterCriticalSection(&ddraw_cs);
3486 if (!(This->surface_desc.ddsCaps.dwCaps2 & DDSCAPS2_TEXTUREMANAGE))
3488 LeaveCriticalSection(&ddraw_cs);
3489 return DDERR_INVALIDOBJECT;
3492 if (!This->wined3d_texture)
3494 ERR("(%p) The DirectDraw texture has no WineD3DTexture!\n", This);
3495 LeaveCriticalSection(&ddraw_cs);
3496 return DDERR_INVALIDOBJECT;
3499 hr = wined3d_texture_set_lod(This->wined3d_texture, MaxLOD);
3500 LeaveCriticalSection(&ddraw_cs);
3501 return hr;
3504 /*****************************************************************************
3505 * IDirectDrawSurface7::GetLOD
3507 * Returns the level of detail of a Direct3D texture
3509 * Params:
3510 * MaxLOD: Address to write the LOD to
3512 * Returns:
3513 * DD_OK on success
3514 * DDERR_INVALIDPARAMS if MaxLOD is NULL
3515 * DDERR_INVALIDOBJECT if the surface is invalid for this method
3517 *****************************************************************************/
3518 static HRESULT WINAPI ddraw_surface7_GetLOD(IDirectDrawSurface7 *iface, DWORD *MaxLOD)
3520 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3522 TRACE("iface %p, lod %p.\n", iface, MaxLOD);
3524 if(!MaxLOD)
3525 return DDERR_INVALIDPARAMS;
3527 EnterCriticalSection(&ddraw_cs);
3528 if (!(This->surface_desc.ddsCaps.dwCaps2 & DDSCAPS2_TEXTUREMANAGE))
3530 LeaveCriticalSection(&ddraw_cs);
3531 return DDERR_INVALIDOBJECT;
3534 *MaxLOD = wined3d_texture_get_lod(This->wined3d_texture);
3535 LeaveCriticalSection(&ddraw_cs);
3536 return DD_OK;
3539 /*****************************************************************************
3540 * IDirectDrawSurface7::BltFast
3542 * Performs a fast Blit.
3544 * Params:
3545 * dstx: The x coordinate to blit to on the destination
3546 * dsty: The y coordinate to blit to on the destination
3547 * Source: The source surface
3548 * rsrc: The source rectangle
3549 * trans: Type of transfer. Some DDBLTFAST_* flags
3551 * Returns:
3552 * DD_OK on success
3553 * For more details, see IWineD3DSurface::BltFast
3555 *****************************************************************************/
3556 static HRESULT WINAPI ddraw_surface7_BltFast(IDirectDrawSurface7 *iface, DWORD dstx, DWORD dsty,
3557 IDirectDrawSurface7 *Source, RECT *rsrc, DWORD trans)
3559 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3560 IDirectDrawSurfaceImpl *src = unsafe_impl_from_IDirectDrawSurface7(Source);
3561 DWORD src_w, src_h, dst_w, dst_h;
3562 HRESULT hr;
3564 TRACE("iface %p, dst_x %u, dst_y %u, src_surface %p, src_rect %s, flags %#x.\n",
3565 iface, dstx, dsty, Source, wine_dbgstr_rect(rsrc), trans);
3567 dst_w = This->surface_desc.dwWidth;
3568 dst_h = This->surface_desc.dwHeight;
3570 /* Source must be != NULL, This is not checked by windows. Windows happily throws a 0xc0000005
3571 * in that case
3573 if(rsrc)
3575 if(rsrc->top > rsrc->bottom || rsrc->left > rsrc->right ||
3576 rsrc->right > src->surface_desc.dwWidth ||
3577 rsrc->bottom > src->surface_desc.dwHeight)
3579 WARN("Source rectangle is invalid, returning DDERR_INVALIDRECT\n");
3580 return DDERR_INVALIDRECT;
3583 src_w = rsrc->right - rsrc->left;
3584 src_h = rsrc->bottom - rsrc->top;
3586 else
3588 src_w = src->surface_desc.dwWidth;
3589 src_h = src->surface_desc.dwHeight;
3592 if (src_w > dst_w || dstx > dst_w - src_w
3593 || src_h > dst_h || dsty > dst_h - src_h)
3595 WARN("Destination area out of bounds, returning DDERR_INVALIDRECT.\n");
3596 return DDERR_INVALIDRECT;
3599 EnterCriticalSection(&ddraw_cs);
3600 hr = wined3d_surface_bltfast(This->wined3d_surface, dstx, dsty,
3601 src->wined3d_surface, rsrc, trans);
3602 LeaveCriticalSection(&ddraw_cs);
3603 switch(hr)
3605 case WINED3DERR_NOTAVAILABLE: return DDERR_UNSUPPORTED;
3606 case WINED3DERR_WRONGTEXTUREFORMAT: return DDERR_INVALIDPIXELFORMAT;
3607 default: return hr;
3611 static HRESULT WINAPI ddraw_surface4_BltFast(IDirectDrawSurface4 *iface, DWORD dst_x, DWORD dst_y,
3612 IDirectDrawSurface4 *src_surface, RECT *src_rect, DWORD flags)
3614 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3615 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface4(src_surface);
3616 TRACE("iface %p, dst_x %u, dst_y %u, src_surface %p, src_rect %s, flags %#x.\n",
3617 iface, dst_x, dst_y, src_surface, wine_dbgstr_rect(src_rect), flags);
3619 return ddraw_surface7_BltFast(&This->IDirectDrawSurface7_iface, dst_x, dst_y,
3620 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags);
3623 static HRESULT WINAPI ddraw_surface3_BltFast(IDirectDrawSurface3 *iface, DWORD dst_x, DWORD dst_y,
3624 IDirectDrawSurface3 *src_surface, RECT *src_rect, DWORD flags)
3626 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3627 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface3(src_surface);
3628 TRACE("iface %p, dst_x %u, dst_y %u, src_surface %p, src_rect %s, flags %#x.\n",
3629 iface, dst_x, dst_y, src_surface, wine_dbgstr_rect(src_rect), flags);
3631 return ddraw_surface7_BltFast(&This->IDirectDrawSurface7_iface, dst_x, dst_y,
3632 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags);
3635 static HRESULT WINAPI ddraw_surface2_BltFast(IDirectDrawSurface2 *iface, DWORD dst_x, DWORD dst_y,
3636 IDirectDrawSurface2 *src_surface, RECT *src_rect, DWORD flags)
3638 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3639 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface2(src_surface);
3640 TRACE("iface %p, dst_x %u, dst_y %u, src_surface %p, src_rect %s, flags %#x.\n",
3641 iface, dst_x, dst_y, src_surface, wine_dbgstr_rect(src_rect), flags);
3643 return ddraw_surface7_BltFast(&This->IDirectDrawSurface7_iface, dst_x, dst_y,
3644 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags);
3647 static HRESULT WINAPI ddraw_surface1_BltFast(IDirectDrawSurface *iface, DWORD dst_x, DWORD dst_y,
3648 IDirectDrawSurface *src_surface, RECT *src_rect, DWORD flags)
3650 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3651 IDirectDrawSurfaceImpl *src_impl = unsafe_impl_from_IDirectDrawSurface(src_surface);
3652 TRACE("iface %p, dst_x %u, dst_y %u, src_surface %p, src_rect %s, flags %#x.\n",
3653 iface, dst_x, dst_y, src_surface, wine_dbgstr_rect(src_rect), flags);
3655 return ddraw_surface7_BltFast(&This->IDirectDrawSurface7_iface, dst_x, dst_y,
3656 src_impl ? &src_impl->IDirectDrawSurface7_iface : NULL, src_rect, flags);
3659 /*****************************************************************************
3660 * IDirectDrawSurface7::GetClipper
3662 * Returns the IDirectDrawClipper interface of the clipper assigned to this
3663 * surface
3665 * Params:
3666 * Clipper: Address to store the interface pointer at
3668 * Returns:
3669 * DD_OK on success
3670 * DDERR_INVALIDPARAMS if Clipper is NULL
3671 * DDERR_NOCLIPPERATTACHED if there's no clipper attached
3673 *****************************************************************************/
3674 static HRESULT WINAPI ddraw_surface7_GetClipper(IDirectDrawSurface7 *iface, IDirectDrawClipper **Clipper)
3676 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3678 TRACE("iface %p, clipper %p.\n", iface, Clipper);
3680 if(!Clipper)
3682 LeaveCriticalSection(&ddraw_cs);
3683 return DDERR_INVALIDPARAMS;
3686 EnterCriticalSection(&ddraw_cs);
3687 if(This->clipper == NULL)
3689 LeaveCriticalSection(&ddraw_cs);
3690 return DDERR_NOCLIPPERATTACHED;
3693 *Clipper = (IDirectDrawClipper *)This->clipper;
3694 IDirectDrawClipper_AddRef(*Clipper);
3695 LeaveCriticalSection(&ddraw_cs);
3696 return DD_OK;
3699 static HRESULT WINAPI ddraw_surface4_GetClipper(IDirectDrawSurface4 *iface, IDirectDrawClipper **clipper)
3701 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3702 TRACE("iface %p, clipper %p.\n", iface, clipper);
3704 return ddraw_surface7_GetClipper(&This->IDirectDrawSurface7_iface, clipper);
3707 static HRESULT WINAPI ddraw_surface3_GetClipper(IDirectDrawSurface3 *iface, IDirectDrawClipper **clipper)
3709 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3710 TRACE("iface %p, clipper %p.\n", iface, clipper);
3712 return ddraw_surface7_GetClipper(&This->IDirectDrawSurface7_iface, clipper);
3715 static HRESULT WINAPI ddraw_surface2_GetClipper(IDirectDrawSurface2 *iface, IDirectDrawClipper **clipper)
3717 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3718 TRACE("iface %p, clipper %p.\n", iface, clipper);
3720 return ddraw_surface7_GetClipper(&This->IDirectDrawSurface7_iface, clipper);
3723 static HRESULT WINAPI ddraw_surface1_GetClipper(IDirectDrawSurface *iface, IDirectDrawClipper **clipper)
3725 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3726 TRACE("iface %p, clipper %p.\n", iface, clipper);
3728 return ddraw_surface7_GetClipper(&This->IDirectDrawSurface7_iface, clipper);
3731 /*****************************************************************************
3732 * IDirectDrawSurface7::SetClipper
3734 * Sets a clipper for the surface
3736 * Params:
3737 * Clipper: IDirectDrawClipper interface of the clipper to set
3739 * Returns:
3740 * DD_OK on success
3742 *****************************************************************************/
3743 static HRESULT WINAPI ddraw_surface7_SetClipper(IDirectDrawSurface7 *iface,
3744 IDirectDrawClipper *iclipper)
3746 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3747 IDirectDrawClipperImpl *clipper = unsafe_impl_from_IDirectDrawClipper(iclipper);
3748 IDirectDrawClipperImpl *oldClipper = This->clipper;
3749 HWND clipWindow;
3750 HRESULT hr;
3752 TRACE("iface %p, clipper %p.\n", iface, iclipper);
3754 EnterCriticalSection(&ddraw_cs);
3755 if (clipper == This->clipper)
3757 LeaveCriticalSection(&ddraw_cs);
3758 return DD_OK;
3761 This->clipper = clipper;
3763 if (clipper != NULL)
3764 IDirectDrawClipper_AddRef(iclipper);
3765 if(oldClipper)
3766 IDirectDrawClipper_Release(&oldClipper->IDirectDrawClipper_iface);
3768 hr = wined3d_surface_set_clipper(This->wined3d_surface,
3769 This->clipper ? This->clipper->wineD3DClipper : NULL);
3771 if (This->wined3d_swapchain)
3773 clipWindow = NULL;
3774 if(clipper) {
3775 IDirectDrawClipper_GetHWnd(iclipper, &clipWindow);
3778 if (clipWindow)
3779 wined3d_swapchain_set_window(This->wined3d_swapchain, clipWindow);
3780 else
3781 wined3d_swapchain_set_window(This->wined3d_swapchain, This->ddraw->d3d_window);
3784 LeaveCriticalSection(&ddraw_cs);
3785 return hr;
3788 static HRESULT WINAPI ddraw_surface4_SetClipper(IDirectDrawSurface4 *iface, IDirectDrawClipper *clipper)
3790 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3791 TRACE("iface %p, clipper %p.\n", iface, clipper);
3793 return ddraw_surface7_SetClipper(&This->IDirectDrawSurface7_iface, clipper);
3796 static HRESULT WINAPI ddraw_surface3_SetClipper(IDirectDrawSurface3 *iface, IDirectDrawClipper *clipper)
3798 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3799 TRACE("iface %p, clipper %p.\n", iface, clipper);
3801 return ddraw_surface7_SetClipper(&This->IDirectDrawSurface7_iface, clipper);
3804 static HRESULT WINAPI ddraw_surface2_SetClipper(IDirectDrawSurface2 *iface, IDirectDrawClipper *clipper)
3806 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3807 TRACE("iface %p, clipper %p.\n", iface, clipper);
3809 return ddraw_surface7_SetClipper(&This->IDirectDrawSurface7_iface, clipper);
3812 static HRESULT WINAPI ddraw_surface1_SetClipper(IDirectDrawSurface *iface, IDirectDrawClipper *clipper)
3814 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
3815 TRACE("iface %p, clipper %p.\n", iface, clipper);
3817 return ddraw_surface7_SetClipper(&This->IDirectDrawSurface7_iface, clipper);
3820 /*****************************************************************************
3821 * IDirectDrawSurface7::SetSurfaceDesc
3823 * Sets the surface description. It can override the pixel format, the surface
3824 * memory, ...
3825 * It's not really tested.
3827 * Params:
3828 * DDSD: Pointer to the new surface description to set
3829 * Flags: Some flags
3831 * Returns:
3832 * DD_OK on success
3833 * DDERR_INVALIDPARAMS if DDSD is NULL
3835 *****************************************************************************/
3836 static HRESULT WINAPI ddraw_surface7_SetSurfaceDesc(IDirectDrawSurface7 *iface, DDSURFACEDESC2 *DDSD, DWORD Flags)
3838 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3839 enum wined3d_format_id newFormat = WINED3DFMT_UNKNOWN;
3840 HRESULT hr;
3842 TRACE("iface %p, surface_desc %p, flags %#x.\n", iface, DDSD, Flags);
3844 if(!DDSD)
3845 return DDERR_INVALIDPARAMS;
3847 EnterCriticalSection(&ddraw_cs);
3848 if (DDSD->dwFlags & DDSD_PIXELFORMAT)
3850 newFormat = PixelFormat_DD2WineD3D(&DDSD->u4.ddpfPixelFormat);
3852 if(newFormat == WINED3DFMT_UNKNOWN)
3854 ERR("Requested to set an unknown pixelformat\n");
3855 LeaveCriticalSection(&ddraw_cs);
3856 return DDERR_INVALIDPARAMS;
3858 if(newFormat != PixelFormat_DD2WineD3D(&This->surface_desc.u4.ddpfPixelFormat) )
3860 hr = wined3d_surface_set_format(This->wined3d_surface, newFormat);
3861 if (FAILED(hr))
3863 LeaveCriticalSection(&ddraw_cs);
3864 return hr;
3868 if (DDSD->dwFlags & DDSD_CKDESTOVERLAY)
3870 wined3d_surface_set_color_key(This->wined3d_surface, DDCKEY_DESTOVERLAY,
3871 (WINEDDCOLORKEY *)&DDSD->u3.ddckCKDestOverlay);
3873 if (DDSD->dwFlags & DDSD_CKDESTBLT)
3875 wined3d_surface_set_color_key(This->wined3d_surface, DDCKEY_DESTBLT,
3876 (WINEDDCOLORKEY *)&DDSD->ddckCKDestBlt);
3878 if (DDSD->dwFlags & DDSD_CKSRCOVERLAY)
3880 wined3d_surface_set_color_key(This->wined3d_surface, DDCKEY_SRCOVERLAY,
3881 (WINEDDCOLORKEY *)&DDSD->ddckCKSrcOverlay);
3883 if (DDSD->dwFlags & DDSD_CKSRCBLT)
3885 wined3d_surface_set_color_key(This->wined3d_surface, DDCKEY_SRCBLT,
3886 (WINEDDCOLORKEY *)&DDSD->ddckCKSrcBlt);
3888 if (DDSD->dwFlags & DDSD_LPSURFACE && DDSD->lpSurface)
3890 hr = wined3d_surface_set_mem(This->wined3d_surface, DDSD->lpSurface);
3891 if (FAILED(hr))
3893 /* No need for a trace here, wined3d does that for us */
3894 switch(hr)
3896 case WINED3DERR_INVALIDCALL:
3897 LeaveCriticalSection(&ddraw_cs);
3898 return DDERR_INVALIDPARAMS;
3899 default:
3900 break; /* Go on */
3905 This->surface_desc = *DDSD;
3907 LeaveCriticalSection(&ddraw_cs);
3908 return DD_OK;
3911 static HRESULT WINAPI ddraw_surface4_SetSurfaceDesc(IDirectDrawSurface4 *iface,
3912 DDSURFACEDESC2 *surface_desc, DWORD flags)
3914 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3915 TRACE("iface %p, surface_desc %p, flags %#x.\n", iface, surface_desc, flags);
3917 return ddraw_surface7_SetSurfaceDesc(&This->IDirectDrawSurface7_iface,
3918 surface_desc, flags);
3921 static HRESULT WINAPI ddraw_surface3_SetSurfaceDesc(IDirectDrawSurface3 *iface,
3922 DDSURFACEDESC *surface_desc, DWORD flags)
3924 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3925 TRACE("iface %p, surface_desc %p, flags %#x.\n", iface, surface_desc, flags);
3927 return ddraw_surface7_SetSurfaceDesc(&This->IDirectDrawSurface7_iface,
3928 (DDSURFACEDESC2 *)surface_desc, flags);
3931 /*****************************************************************************
3932 * IDirectDrawSurface7::GetPalette
3934 * Returns the IDirectDrawPalette interface of the palette currently assigned
3935 * to the surface
3937 * Params:
3938 * Pal: Address to write the interface pointer to
3940 * Returns:
3941 * DD_OK on success
3942 * DDERR_INVALIDPARAMS if Pal is NULL
3944 *****************************************************************************/
3945 static HRESULT WINAPI ddraw_surface7_GetPalette(IDirectDrawSurface7 *iface, IDirectDrawPalette **Pal)
3947 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
3948 struct wined3d_palette *wined3d_palette;
3949 HRESULT hr = DD_OK;
3951 TRACE("iface %p, palette %p.\n", iface, Pal);
3953 if(!Pal)
3954 return DDERR_INVALIDPARAMS;
3956 EnterCriticalSection(&ddraw_cs);
3957 wined3d_palette = wined3d_surface_get_palette(This->wined3d_surface);
3958 if (wined3d_palette)
3960 *Pal = wined3d_palette_get_parent(wined3d_palette);
3961 IDirectDrawPalette_AddRef(*Pal);
3963 else
3965 *Pal = NULL;
3966 hr = DDERR_NOPALETTEATTACHED;
3969 LeaveCriticalSection(&ddraw_cs);
3970 return hr;
3973 static HRESULT WINAPI ddraw_surface4_GetPalette(IDirectDrawSurface4 *iface, IDirectDrawPalette **palette)
3975 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
3976 TRACE("iface %p, palette %p.\n", iface, palette);
3978 return ddraw_surface7_GetPalette(&This->IDirectDrawSurface7_iface, palette);
3981 static HRESULT WINAPI ddraw_surface3_GetPalette(IDirectDrawSurface3 *iface, IDirectDrawPalette **palette)
3983 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
3984 TRACE("iface %p, palette %p.\n", iface, palette);
3986 return ddraw_surface7_GetPalette(&This->IDirectDrawSurface7_iface, palette);
3989 static HRESULT WINAPI ddraw_surface2_GetPalette(IDirectDrawSurface2 *iface, IDirectDrawPalette **palette)
3991 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
3992 TRACE("iface %p, palette %p.\n", iface, palette);
3994 return ddraw_surface7_GetPalette(&This->IDirectDrawSurface7_iface, palette);
3997 static HRESULT WINAPI ddraw_surface1_GetPalette(IDirectDrawSurface *iface, IDirectDrawPalette **palette)
3999 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
4000 TRACE("iface %p, palette %p.\n", iface, palette);
4002 return ddraw_surface7_GetPalette(&This->IDirectDrawSurface7_iface, palette);
4005 /*****************************************************************************
4006 * SetColorKeyEnum
4008 * EnumAttachedSurface callback for SetColorKey. Used to set color keys
4009 * recursively in the surface tree
4011 *****************************************************************************/
4012 struct SCKContext
4014 HRESULT ret;
4015 WINEDDCOLORKEY *CKey;
4016 DWORD Flags;
4019 static HRESULT WINAPI
4020 SetColorKeyEnum(IDirectDrawSurface7 *surface,
4021 DDSURFACEDESC2 *desc,
4022 void *context)
4024 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(surface);
4025 struct SCKContext *ctx = context;
4026 HRESULT hr;
4028 hr = wined3d_surface_set_color_key(This->wined3d_surface, ctx->Flags, ctx->CKey);
4029 if (FAILED(hr))
4031 WARN("IWineD3DSurface_SetColorKey failed, hr = %08x\n", hr);
4032 ctx->ret = hr;
4035 ddraw_surface7_EnumAttachedSurfaces(surface, context, SetColorKeyEnum);
4036 ddraw_surface7_Release(surface);
4038 return DDENUMRET_OK;
4041 /*****************************************************************************
4042 * IDirectDrawSurface7::SetColorKey
4044 * Sets the color keying options for the surface. Observations showed that
4045 * in case of complex surfaces the color key has to be assigned to all
4046 * sublevels.
4048 * Params:
4049 * Flags: DDCKEY_*
4050 * CKey: The new color key
4052 * Returns:
4053 * DD_OK on success
4054 * See IWineD3DSurface::SetColorKey for details
4056 *****************************************************************************/
4057 static HRESULT WINAPI ddraw_surface7_SetColorKey(IDirectDrawSurface7 *iface, DWORD Flags, DDCOLORKEY *CKey)
4059 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
4060 DDCOLORKEY FixedCKey;
4061 struct SCKContext ctx = { DD_OK, (WINEDDCOLORKEY *) (CKey ? &FixedCKey : NULL), Flags };
4063 TRACE("iface %p, flags %#x, color_key %p.\n", iface, Flags, CKey);
4065 EnterCriticalSection(&ddraw_cs);
4066 if (CKey)
4068 FixedCKey = *CKey;
4069 /* Handle case where dwColorSpaceHighValue < dwColorSpaceLowValue */
4070 if (FixedCKey.dwColorSpaceHighValue < FixedCKey.dwColorSpaceLowValue)
4071 FixedCKey.dwColorSpaceHighValue = FixedCKey.dwColorSpaceLowValue;
4073 switch (Flags & ~DDCKEY_COLORSPACE)
4075 case DDCKEY_DESTBLT:
4076 This->surface_desc.ddckCKDestBlt = FixedCKey;
4077 This->surface_desc.dwFlags |= DDSD_CKDESTBLT;
4078 break;
4080 case DDCKEY_DESTOVERLAY:
4081 This->surface_desc.u3.ddckCKDestOverlay = FixedCKey;
4082 This->surface_desc.dwFlags |= DDSD_CKDESTOVERLAY;
4083 break;
4085 case DDCKEY_SRCOVERLAY:
4086 This->surface_desc.ddckCKSrcOverlay = FixedCKey;
4087 This->surface_desc.dwFlags |= DDSD_CKSRCOVERLAY;
4088 break;
4090 case DDCKEY_SRCBLT:
4091 This->surface_desc.ddckCKSrcBlt = FixedCKey;
4092 This->surface_desc.dwFlags |= DDSD_CKSRCBLT;
4093 break;
4095 default:
4096 LeaveCriticalSection(&ddraw_cs);
4097 return DDERR_INVALIDPARAMS;
4100 else
4102 switch (Flags & ~DDCKEY_COLORSPACE)
4104 case DDCKEY_DESTBLT:
4105 This->surface_desc.dwFlags &= ~DDSD_CKDESTBLT;
4106 break;
4108 case DDCKEY_DESTOVERLAY:
4109 This->surface_desc.dwFlags &= ~DDSD_CKDESTOVERLAY;
4110 break;
4112 case DDCKEY_SRCOVERLAY:
4113 This->surface_desc.dwFlags &= ~DDSD_CKSRCOVERLAY;
4114 break;
4116 case DDCKEY_SRCBLT:
4117 This->surface_desc.dwFlags &= ~DDSD_CKSRCBLT;
4118 break;
4120 default:
4121 LeaveCriticalSection(&ddraw_cs);
4122 return DDERR_INVALIDPARAMS;
4125 ctx.ret = wined3d_surface_set_color_key(This->wined3d_surface, Flags, ctx.CKey);
4126 ddraw_surface7_EnumAttachedSurfaces(iface, &ctx, SetColorKeyEnum);
4127 LeaveCriticalSection(&ddraw_cs);
4128 switch(ctx.ret)
4130 case WINED3DERR_INVALIDCALL: return DDERR_INVALIDPARAMS;
4131 default: return ctx.ret;
4135 static HRESULT WINAPI ddraw_surface4_SetColorKey(IDirectDrawSurface4 *iface, DWORD flags, DDCOLORKEY *color_key)
4137 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
4138 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
4140 return ddraw_surface7_SetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
4143 static HRESULT WINAPI ddraw_surface3_SetColorKey(IDirectDrawSurface3 *iface, DWORD flags, DDCOLORKEY *color_key)
4145 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
4146 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
4148 return ddraw_surface7_SetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
4151 static HRESULT WINAPI ddraw_surface2_SetColorKey(IDirectDrawSurface2 *iface, DWORD flags, DDCOLORKEY *color_key)
4153 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
4154 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
4156 return ddraw_surface7_SetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
4159 static HRESULT WINAPI ddraw_surface1_SetColorKey(IDirectDrawSurface *iface, DWORD flags, DDCOLORKEY *color_key)
4161 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
4162 TRACE("iface %p, flags %#x, color_key %p.\n", iface, flags, color_key);
4164 return ddraw_surface7_SetColorKey(&This->IDirectDrawSurface7_iface, flags, color_key);
4167 /*****************************************************************************
4168 * IDirectDrawSurface7::SetPalette
4170 * Assigns a DirectDrawPalette object to the surface
4172 * Params:
4173 * Pal: Interface to the palette to set
4175 * Returns:
4176 * DD_OK on success
4178 *****************************************************************************/
4179 static HRESULT WINAPI ddraw_surface7_SetPalette(IDirectDrawSurface7 *iface, IDirectDrawPalette *Pal)
4181 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface7(iface);
4182 IDirectDrawPalette *oldPal;
4183 IDirectDrawSurfaceImpl *surf;
4184 IDirectDrawPaletteImpl *PalImpl = (IDirectDrawPaletteImpl *)Pal;
4185 HRESULT hr;
4187 TRACE("iface %p, palette %p.\n", iface, Pal);
4189 if (!(This->surface_desc.u4.ddpfPixelFormat.dwFlags & (DDPF_PALETTEINDEXED1 | DDPF_PALETTEINDEXED2 |
4190 DDPF_PALETTEINDEXED4 | DDPF_PALETTEINDEXED8 | DDPF_PALETTEINDEXEDTO8))) {
4191 return DDERR_INVALIDPIXELFORMAT;
4194 /* Find the old palette */
4195 EnterCriticalSection(&ddraw_cs);
4196 hr = IDirectDrawSurface_GetPalette(iface, &oldPal);
4197 if(hr != DD_OK && hr != DDERR_NOPALETTEATTACHED)
4199 LeaveCriticalSection(&ddraw_cs);
4200 return hr;
4202 if(oldPal) IDirectDrawPalette_Release(oldPal); /* For the GetPalette */
4204 /* Set the new Palette */
4205 wined3d_surface_set_palette(This->wined3d_surface, PalImpl ? PalImpl->wineD3DPalette : NULL);
4206 /* AddRef the Palette */
4207 if(Pal) IDirectDrawPalette_AddRef(Pal);
4209 /* Release the old palette */
4210 if(oldPal) IDirectDrawPalette_Release(oldPal);
4212 /* If this is a front buffer, also update the back buffers
4213 * TODO: How do things work for palettized cube textures?
4215 if(This->surface_desc.ddsCaps.dwCaps & DDSCAPS_FRONTBUFFER)
4217 /* For primary surfaces the tree is just a list, so the simpler scheme fits too */
4218 DDSCAPS2 caps2 = { DDSCAPS_PRIMARYSURFACE, 0, 0, 0 };
4220 surf = This;
4221 while(1)
4223 IDirectDrawSurface7 *attach;
4224 HRESULT hr;
4225 hr = ddraw_surface7_GetAttachedSurface(&surf->IDirectDrawSurface7_iface, &caps2, &attach);
4226 if(hr != DD_OK)
4228 break;
4231 TRACE("Setting palette on %p\n", attach);
4232 ddraw_surface7_SetPalette(attach, Pal);
4233 surf = impl_from_IDirectDrawSurface7(attach);
4234 ddraw_surface7_Release(attach);
4238 LeaveCriticalSection(&ddraw_cs);
4239 return DD_OK;
4242 static HRESULT WINAPI ddraw_surface4_SetPalette(IDirectDrawSurface4 *iface, IDirectDrawPalette *palette)
4244 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface4(iface);
4245 TRACE("iface %p, palette %p.\n", iface, palette);
4247 return ddraw_surface7_SetPalette(&This->IDirectDrawSurface7_iface, palette);
4250 static HRESULT WINAPI ddraw_surface3_SetPalette(IDirectDrawSurface3 *iface, IDirectDrawPalette *palette)
4252 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface3(iface);
4253 TRACE("iface %p, palette %p.\n", iface, palette);
4255 return ddraw_surface7_SetPalette(&This->IDirectDrawSurface7_iface, palette);
4258 static HRESULT WINAPI ddraw_surface2_SetPalette(IDirectDrawSurface2 *iface, IDirectDrawPalette *palette)
4260 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface2(iface);
4261 TRACE("iface %p, palette %p.\n", iface, palette);
4263 return ddraw_surface7_SetPalette(&This->IDirectDrawSurface7_iface, palette);
4266 static HRESULT WINAPI ddraw_surface1_SetPalette(IDirectDrawSurface *iface, IDirectDrawPalette *palette)
4268 IDirectDrawSurfaceImpl *This = impl_from_IDirectDrawSurface(iface);
4269 TRACE("iface %p, palette %p.\n", iface, palette);
4271 return ddraw_surface7_SetPalette(&This->IDirectDrawSurface7_iface, palette);
4274 /**********************************************************
4275 * IDirectDrawGammaControl::GetGammaRamp
4277 * Returns the current gamma ramp for a surface
4279 * Params:
4280 * flags: Ignored
4281 * gamma_ramp: Address to write the ramp to
4283 * Returns:
4284 * DD_OK on success
4285 * DDERR_INVALIDPARAMS if gamma_ramp is NULL
4287 **********************************************************/
4288 static HRESULT WINAPI ddraw_gamma_control_GetGammaRamp(IDirectDrawGammaControl *iface,
4289 DWORD flags, DDGAMMARAMP *gamma_ramp)
4291 IDirectDrawSurfaceImpl *surface = impl_from_IDirectDrawGammaControl(iface);
4293 TRACE("iface %p, flags %#x, gamma_ramp %p.\n", iface, flags, gamma_ramp);
4295 if (!gamma_ramp)
4297 WARN("Invalid gamma_ramp passed.\n");
4298 return DDERR_INVALIDPARAMS;
4301 EnterCriticalSection(&ddraw_cs);
4302 if (surface->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE)
4304 /* Note: DDGAMMARAMP is compatible with WINED3DGAMMARAMP. */
4305 wined3d_device_get_gamma_ramp(surface->ddraw->wined3d_device, 0, (WINED3DGAMMARAMP *)gamma_ramp);
4307 else
4309 ERR("Not implemented for non-primary surfaces.\n");
4311 LeaveCriticalSection(&ddraw_cs);
4313 return DD_OK;
4316 /**********************************************************
4317 * IDirectDrawGammaControl::SetGammaRamp
4319 * Sets the red, green and blue gamma ramps for
4321 * Params:
4322 * flags: Can be DDSGR_CALIBRATE to request calibration
4323 * gamma_ramp: Structure containing the new gamma ramp
4325 * Returns:
4326 * DD_OK on success
4327 * DDERR_INVALIDPARAMS if gamma_ramp is NULL
4329 **********************************************************/
4330 static HRESULT WINAPI ddraw_gamma_control_SetGammaRamp(IDirectDrawGammaControl *iface,
4331 DWORD flags, DDGAMMARAMP *gamma_ramp)
4333 IDirectDrawSurfaceImpl *surface = impl_from_IDirectDrawGammaControl(iface);
4335 TRACE("iface %p, flags %#x, gamma_ramp %p.\n", iface, flags, gamma_ramp);
4337 if (!gamma_ramp)
4339 WARN("Invalid gamma_ramp passed.\n");
4340 return DDERR_INVALIDPARAMS;
4343 EnterCriticalSection(&ddraw_cs);
4344 if (surface->surface_desc.ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE)
4346 /* Note: DDGAMMARAMP is compatible with WINED3DGAMMARAMP */
4347 wined3d_device_set_gamma_ramp(surface->ddraw->wined3d_device, 0, flags, (WINED3DGAMMARAMP *)gamma_ramp);
4349 else
4351 ERR("Not implemented for non-primary surfaces.\n");
4353 LeaveCriticalSection(&ddraw_cs);
4355 return DD_OK;
4358 /*****************************************************************************
4359 * IDirect3DTexture2::PaletteChanged
4361 * Informs the texture about a palette change
4363 * Params:
4364 * start: Start index of the change
4365 * count: The number of changed entries
4367 * Returns
4368 * D3D_OK, because it's a stub
4370 *****************************************************************************/
4371 static HRESULT WINAPI d3d_texture2_PaletteChanged(IDirect3DTexture2 *iface, DWORD start, DWORD count)
4373 FIXME("iface %p, start %u, count %u stub!\n", iface, start, count);
4375 return D3D_OK;
4378 static HRESULT WINAPI d3d_texture1_PaletteChanged(IDirect3DTexture *iface, DWORD start, DWORD count)
4380 IDirectDrawSurfaceImpl *surface = surface_from_texture1(iface);
4382 TRACE("iface %p, start %u, count %u.\n", iface, start, count);
4384 return d3d_texture2_PaletteChanged((IDirect3DTexture2 *)&surface->IDirect3DTexture2_vtbl, start, count);
4387 /*****************************************************************************
4388 * IDirect3DTexture::Unload
4390 * DX5 SDK: "The IDirect3DTexture2::Unload method is not implemented
4393 * Returns:
4394 * DDERR_UNSUPPORTED
4396 *****************************************************************************/
4397 static HRESULT WINAPI d3d_texture1_Unload(IDirect3DTexture *iface)
4399 WARN("iface %p. Not implemented.\n", iface);
4401 return DDERR_UNSUPPORTED;
4404 /*****************************************************************************
4405 * IDirect3DTexture2::GetHandle
4407 * Returns handle for the texture. At the moment, the interface
4408 * to the IWineD3DTexture is used.
4410 * Params:
4411 * device: Device this handle is assigned to
4412 * handle: Address to store the handle at.
4414 * Returns:
4415 * D3D_OK
4417 *****************************************************************************/
4418 static HRESULT WINAPI d3d_texture2_GetHandle(IDirect3DTexture2 *iface,
4419 IDirect3DDevice2 *device, D3DTEXTUREHANDLE *handle)
4421 IDirectDrawSurfaceImpl *surface = surface_from_texture2(iface);
4423 TRACE("iface %p, device %p, handle %p.\n", iface, device, handle);
4425 EnterCriticalSection(&ddraw_cs);
4427 if (!surface->Handle)
4429 DWORD h = ddraw_allocate_handle(&device_from_device2(device)->handle_table, surface, DDRAW_HANDLE_SURFACE);
4430 if (h == DDRAW_INVALID_HANDLE)
4432 ERR("Failed to allocate a texture handle.\n");
4433 LeaveCriticalSection(&ddraw_cs);
4434 return DDERR_OUTOFMEMORY;
4437 surface->Handle = h + 1;
4440 TRACE("Returning handle %08x.\n", surface->Handle);
4441 *handle = surface->Handle;
4443 LeaveCriticalSection(&ddraw_cs);
4445 return D3D_OK;
4448 static HRESULT WINAPI d3d_texture1_GetHandle(IDirect3DTexture *iface,
4449 IDirect3DDevice *device, D3DTEXTUREHANDLE *handle)
4451 IDirect3DTexture2 *texture2 = (IDirect3DTexture2 *)&surface_from_texture1(iface)->IDirect3DTexture2_vtbl;
4452 IDirect3DDevice2 *device2 = (IDirect3DDevice2 *)&device_from_device1(device)->IDirect3DDevice2_vtbl;
4454 TRACE("iface %p, device %p, handle %p.\n", iface, device, handle);
4456 return d3d_texture2_GetHandle(texture2, device2, handle);
4459 /*****************************************************************************
4460 * get_sub_mimaplevel
4462 * Helper function that returns the next mipmap level
4464 * tex_ptr: Surface of which to return the next level
4466 *****************************************************************************/
4467 static IDirectDrawSurfaceImpl *get_sub_mimaplevel(IDirectDrawSurfaceImpl *surface)
4469 /* Now go down the mipmap chain to the next surface */
4470 static DDSCAPS2 mipmap_caps = { DDSCAPS_MIPMAP | DDSCAPS_TEXTURE, 0, 0, 0 };
4471 IDirectDrawSurface7 *next_level;
4472 HRESULT hr;
4474 hr = ddraw_surface7_GetAttachedSurface(&surface->IDirectDrawSurface7_iface, &mipmap_caps, &next_level);
4475 if (FAILED(hr)) return NULL;
4477 ddraw_surface7_Release(next_level);
4479 return impl_from_IDirectDrawSurface7(next_level);
4482 /*****************************************************************************
4483 * IDirect3DTexture2::Load
4485 * Loads a texture created with the DDSCAPS_ALLOCONLOAD
4487 * This function isn't relayed to WineD3D because the whole interface is
4488 * implemented in DDraw only. For speed improvements a implementation which
4489 * takes OpenGL more into account could be placed into WineD3D.
4491 * Params:
4492 * src_texture: Address of the texture to load
4494 * Returns:
4495 * D3D_OK on success
4496 * D3DERR_TEXTURE_LOAD_FAILED.
4498 *****************************************************************************/
4499 static HRESULT WINAPI d3d_texture2_Load(IDirect3DTexture2 *iface, IDirect3DTexture2 *src_texture)
4501 IDirectDrawSurfaceImpl *dst_surface = surface_from_texture2(iface);
4502 IDirectDrawSurfaceImpl *src_surface = surface_from_texture2(src_texture);
4503 HRESULT hr;
4505 TRACE("iface %p, src_texture %p.\n", iface, src_texture);
4507 if (src_surface == dst_surface)
4509 TRACE("copying surface %p to surface %p, why?\n", src_surface, dst_surface);
4510 return D3D_OK;
4513 EnterCriticalSection(&ddraw_cs);
4515 if (((src_surface->surface_desc.ddsCaps.dwCaps & DDSCAPS_MIPMAP)
4516 != (dst_surface->surface_desc.ddsCaps.dwCaps & DDSCAPS_MIPMAP))
4517 || (src_surface->surface_desc.u2.dwMipMapCount != dst_surface->surface_desc.u2.dwMipMapCount))
4519 ERR("Trying to load surfaces with different mip-map counts.\n");
4522 for (;;)
4524 struct wined3d_palette *wined3d_dst_pal, *wined3d_src_pal;
4525 IDirectDrawPalette *dst_pal = NULL, *src_pal = NULL;
4526 DDSURFACEDESC *src_desc, *dst_desc;
4528 TRACE("Copying surface %p to surface %p (mipmap level %d).\n",
4529 src_surface, dst_surface, src_surface->mipmap_level);
4531 /* Suppress the ALLOCONLOAD flag */
4532 dst_surface->surface_desc.ddsCaps.dwCaps &= ~DDSCAPS_ALLOCONLOAD;
4534 /* Get the palettes */
4535 wined3d_dst_pal = wined3d_surface_get_palette(dst_surface->wined3d_surface);
4536 if (wined3d_dst_pal)
4537 dst_pal = wined3d_palette_get_parent(wined3d_dst_pal);
4539 wined3d_src_pal = wined3d_surface_get_palette(src_surface->wined3d_surface);
4540 if (wined3d_src_pal)
4541 src_pal = wined3d_palette_get_parent(wined3d_src_pal);
4543 if (src_pal)
4545 PALETTEENTRY palent[256];
4547 if (!dst_pal)
4549 LeaveCriticalSection(&ddraw_cs);
4550 return DDERR_NOPALETTEATTACHED;
4552 IDirectDrawPalette_GetEntries(src_pal, 0, 0, 256, palent);
4553 IDirectDrawPalette_SetEntries(dst_pal, 0, 0, 256, palent);
4556 /* Copy one surface on the other */
4557 dst_desc = (DDSURFACEDESC *)&(dst_surface->surface_desc);
4558 src_desc = (DDSURFACEDESC *)&(src_surface->surface_desc);
4560 if ((src_desc->dwWidth != dst_desc->dwWidth) || (src_desc->dwHeight != dst_desc->dwHeight))
4562 /* Should also check for same pixel format, u1.lPitch, ... */
4563 ERR("Error in surface sizes.\n");
4564 LeaveCriticalSection(&ddraw_cs);
4565 return D3DERR_TEXTURE_LOAD_FAILED;
4567 else
4569 WINED3DLOCKED_RECT src_rect, dst_rect;
4571 /* Copy also the ColorKeying stuff */
4572 if (src_desc->dwFlags & DDSD_CKSRCBLT)
4574 dst_desc->dwFlags |= DDSD_CKSRCBLT;
4575 dst_desc->ddckCKSrcBlt.dwColorSpaceLowValue = src_desc->ddckCKSrcBlt.dwColorSpaceLowValue;
4576 dst_desc->ddckCKSrcBlt.dwColorSpaceHighValue = src_desc->ddckCKSrcBlt.dwColorSpaceHighValue;
4579 /* Copy the main memory texture into the surface that corresponds
4580 * to the OpenGL texture object. */
4582 hr = wined3d_surface_map(src_surface->wined3d_surface, &src_rect, NULL, 0);
4583 if (FAILED(hr))
4585 ERR("Failed to lock source surface, hr %#x.\n", hr);
4586 LeaveCriticalSection(&ddraw_cs);
4587 return D3DERR_TEXTURE_LOAD_FAILED;
4590 hr = wined3d_surface_map(dst_surface->wined3d_surface, &dst_rect, NULL, 0);
4591 if (FAILED(hr))
4593 ERR("Failed to lock destination surface, hr %#x.\n", hr);
4594 wined3d_surface_unmap(src_surface->wined3d_surface);
4595 LeaveCriticalSection(&ddraw_cs);
4596 return D3DERR_TEXTURE_LOAD_FAILED;
4599 if (dst_surface->surface_desc.u4.ddpfPixelFormat.dwFlags & DDPF_FOURCC)
4600 memcpy(dst_rect.pBits, src_rect.pBits, src_surface->surface_desc.u1.dwLinearSize);
4601 else
4602 memcpy(dst_rect.pBits, src_rect.pBits, src_rect.Pitch * src_desc->dwHeight);
4604 wined3d_surface_unmap(src_surface->wined3d_surface);
4605 wined3d_surface_unmap(dst_surface->wined3d_surface);
4608 if (src_surface->surface_desc.ddsCaps.dwCaps & DDSCAPS_MIPMAP)
4609 src_surface = get_sub_mimaplevel(src_surface);
4610 else
4611 src_surface = NULL;
4613 if (dst_surface->surface_desc.ddsCaps.dwCaps & DDSCAPS_MIPMAP)
4614 dst_surface = get_sub_mimaplevel(dst_surface);
4615 else
4616 dst_surface = NULL;
4618 if (!src_surface || !dst_surface)
4620 if (src_surface != dst_surface)
4621 ERR("Loading surface with different mipmap structure.\n");
4622 break;
4626 LeaveCriticalSection(&ddraw_cs);
4628 return hr;
4631 static HRESULT WINAPI d3d_texture1_Load(IDirect3DTexture *iface, IDirect3DTexture *src_texture)
4633 TRACE("iface %p, src_texture %p.\n", iface, src_texture);
4635 return d3d_texture2_Load((IDirect3DTexture2 *)&surface_from_texture1(iface)->IDirect3DTexture2_vtbl,
4636 src_texture ? (IDirect3DTexture2 *)&surface_from_texture1(src_texture)->IDirect3DTexture2_vtbl : NULL);
4639 /*****************************************************************************
4640 * The VTable
4641 *****************************************************************************/
4643 static const struct IDirectDrawSurface7Vtbl ddraw_surface7_vtbl =
4645 /* IUnknown */
4646 ddraw_surface7_QueryInterface,
4647 ddraw_surface7_AddRef,
4648 ddraw_surface7_Release,
4649 /* IDirectDrawSurface */
4650 ddraw_surface7_AddAttachedSurface,
4651 ddraw_surface7_AddOverlayDirtyRect,
4652 ddraw_surface7_Blt,
4653 ddraw_surface7_BltBatch,
4654 ddraw_surface7_BltFast,
4655 ddraw_surface7_DeleteAttachedSurface,
4656 ddraw_surface7_EnumAttachedSurfaces,
4657 ddraw_surface7_EnumOverlayZOrders,
4658 ddraw_surface7_Flip,
4659 ddraw_surface7_GetAttachedSurface,
4660 ddraw_surface7_GetBltStatus,
4661 ddraw_surface7_GetCaps,
4662 ddraw_surface7_GetClipper,
4663 ddraw_surface7_GetColorKey,
4664 ddraw_surface7_GetDC,
4665 ddraw_surface7_GetFlipStatus,
4666 ddraw_surface7_GetOverlayPosition,
4667 ddraw_surface7_GetPalette,
4668 ddraw_surface7_GetPixelFormat,
4669 ddraw_surface7_GetSurfaceDesc,
4670 ddraw_surface7_Initialize,
4671 ddraw_surface7_IsLost,
4672 ddraw_surface7_Lock,
4673 ddraw_surface7_ReleaseDC,
4674 ddraw_surface7_Restore,
4675 ddraw_surface7_SetClipper,
4676 ddraw_surface7_SetColorKey,
4677 ddraw_surface7_SetOverlayPosition,
4678 ddraw_surface7_SetPalette,
4679 ddraw_surface7_Unlock,
4680 ddraw_surface7_UpdateOverlay,
4681 ddraw_surface7_UpdateOverlayDisplay,
4682 ddraw_surface7_UpdateOverlayZOrder,
4683 /* IDirectDrawSurface2 */
4684 ddraw_surface7_GetDDInterface,
4685 ddraw_surface7_PageLock,
4686 ddraw_surface7_PageUnlock,
4687 /* IDirectDrawSurface3 */
4688 ddraw_surface7_SetSurfaceDesc,
4689 /* IDirectDrawSurface4 */
4690 ddraw_surface7_SetPrivateData,
4691 ddraw_surface7_GetPrivateData,
4692 ddraw_surface7_FreePrivateData,
4693 ddraw_surface7_GetUniquenessValue,
4694 ddraw_surface7_ChangeUniquenessValue,
4695 /* IDirectDrawSurface7 */
4696 ddraw_surface7_SetPriority,
4697 ddraw_surface7_GetPriority,
4698 ddraw_surface7_SetLOD,
4699 ddraw_surface7_GetLOD,
4702 static const struct IDirectDrawSurface4Vtbl ddraw_surface4_vtbl =
4704 /* IUnknown */
4705 ddraw_surface4_QueryInterface,
4706 ddraw_surface4_AddRef,
4707 ddraw_surface4_Release,
4708 /* IDirectDrawSurface */
4709 ddraw_surface4_AddAttachedSurface,
4710 ddraw_surface4_AddOverlayDirtyRect,
4711 ddraw_surface4_Blt,
4712 ddraw_surface4_BltBatch,
4713 ddraw_surface4_BltFast,
4714 ddraw_surface4_DeleteAttachedSurface,
4715 ddraw_surface4_EnumAttachedSurfaces,
4716 ddraw_surface4_EnumOverlayZOrders,
4717 ddraw_surface4_Flip,
4718 ddraw_surface4_GetAttachedSurface,
4719 ddraw_surface4_GetBltStatus,
4720 ddraw_surface4_GetCaps,
4721 ddraw_surface4_GetClipper,
4722 ddraw_surface4_GetColorKey,
4723 ddraw_surface4_GetDC,
4724 ddraw_surface4_GetFlipStatus,
4725 ddraw_surface4_GetOverlayPosition,
4726 ddraw_surface4_GetPalette,
4727 ddraw_surface4_GetPixelFormat,
4728 ddraw_surface4_GetSurfaceDesc,
4729 ddraw_surface4_Initialize,
4730 ddraw_surface4_IsLost,
4731 ddraw_surface4_Lock,
4732 ddraw_surface4_ReleaseDC,
4733 ddraw_surface4_Restore,
4734 ddraw_surface4_SetClipper,
4735 ddraw_surface4_SetColorKey,
4736 ddraw_surface4_SetOverlayPosition,
4737 ddraw_surface4_SetPalette,
4738 ddraw_surface4_Unlock,
4739 ddraw_surface4_UpdateOverlay,
4740 ddraw_surface4_UpdateOverlayDisplay,
4741 ddraw_surface4_UpdateOverlayZOrder,
4742 /* IDirectDrawSurface2 */
4743 ddraw_surface4_GetDDInterface,
4744 ddraw_surface4_PageLock,
4745 ddraw_surface4_PageUnlock,
4746 /* IDirectDrawSurface3 */
4747 ddraw_surface4_SetSurfaceDesc,
4748 /* IDirectDrawSurface4 */
4749 ddraw_surface4_SetPrivateData,
4750 ddraw_surface4_GetPrivateData,
4751 ddraw_surface4_FreePrivateData,
4752 ddraw_surface4_GetUniquenessValue,
4753 ddraw_surface4_ChangeUniquenessValue,
4756 static const struct IDirectDrawSurface3Vtbl ddraw_surface3_vtbl =
4758 /* IUnknown */
4759 ddraw_surface3_QueryInterface,
4760 ddraw_surface3_AddRef,
4761 ddraw_surface3_Release,
4762 /* IDirectDrawSurface */
4763 ddraw_surface3_AddAttachedSurface,
4764 ddraw_surface3_AddOverlayDirtyRect,
4765 ddraw_surface3_Blt,
4766 ddraw_surface3_BltBatch,
4767 ddraw_surface3_BltFast,
4768 ddraw_surface3_DeleteAttachedSurface,
4769 ddraw_surface3_EnumAttachedSurfaces,
4770 ddraw_surface3_EnumOverlayZOrders,
4771 ddraw_surface3_Flip,
4772 ddraw_surface3_GetAttachedSurface,
4773 ddraw_surface3_GetBltStatus,
4774 ddraw_surface3_GetCaps,
4775 ddraw_surface3_GetClipper,
4776 ddraw_surface3_GetColorKey,
4777 ddraw_surface3_GetDC,
4778 ddraw_surface3_GetFlipStatus,
4779 ddraw_surface3_GetOverlayPosition,
4780 ddraw_surface3_GetPalette,
4781 ddraw_surface3_GetPixelFormat,
4782 ddraw_surface3_GetSurfaceDesc,
4783 ddraw_surface3_Initialize,
4784 ddraw_surface3_IsLost,
4785 ddraw_surface3_Lock,
4786 ddraw_surface3_ReleaseDC,
4787 ddraw_surface3_Restore,
4788 ddraw_surface3_SetClipper,
4789 ddraw_surface3_SetColorKey,
4790 ddraw_surface3_SetOverlayPosition,
4791 ddraw_surface3_SetPalette,
4792 ddraw_surface3_Unlock,
4793 ddraw_surface3_UpdateOverlay,
4794 ddraw_surface3_UpdateOverlayDisplay,
4795 ddraw_surface3_UpdateOverlayZOrder,
4796 /* IDirectDrawSurface2 */
4797 ddraw_surface3_GetDDInterface,
4798 ddraw_surface3_PageLock,
4799 ddraw_surface3_PageUnlock,
4800 /* IDirectDrawSurface3 */
4801 ddraw_surface3_SetSurfaceDesc,
4804 static const struct IDirectDrawSurface2Vtbl ddraw_surface2_vtbl =
4806 /* IUnknown */
4807 ddraw_surface2_QueryInterface,
4808 ddraw_surface2_AddRef,
4809 ddraw_surface2_Release,
4810 /* IDirectDrawSurface */
4811 ddraw_surface2_AddAttachedSurface,
4812 ddraw_surface2_AddOverlayDirtyRect,
4813 ddraw_surface2_Blt,
4814 ddraw_surface2_BltBatch,
4815 ddraw_surface2_BltFast,
4816 ddraw_surface2_DeleteAttachedSurface,
4817 ddraw_surface2_EnumAttachedSurfaces,
4818 ddraw_surface2_EnumOverlayZOrders,
4819 ddraw_surface2_Flip,
4820 ddraw_surface2_GetAttachedSurface,
4821 ddraw_surface2_GetBltStatus,
4822 ddraw_surface2_GetCaps,
4823 ddraw_surface2_GetClipper,
4824 ddraw_surface2_GetColorKey,
4825 ddraw_surface2_GetDC,
4826 ddraw_surface2_GetFlipStatus,
4827 ddraw_surface2_GetOverlayPosition,
4828 ddraw_surface2_GetPalette,
4829 ddraw_surface2_GetPixelFormat,
4830 ddraw_surface2_GetSurfaceDesc,
4831 ddraw_surface2_Initialize,
4832 ddraw_surface2_IsLost,
4833 ddraw_surface2_Lock,
4834 ddraw_surface2_ReleaseDC,
4835 ddraw_surface2_Restore,
4836 ddraw_surface2_SetClipper,
4837 ddraw_surface2_SetColorKey,
4838 ddraw_surface2_SetOverlayPosition,
4839 ddraw_surface2_SetPalette,
4840 ddraw_surface2_Unlock,
4841 ddraw_surface2_UpdateOverlay,
4842 ddraw_surface2_UpdateOverlayDisplay,
4843 ddraw_surface2_UpdateOverlayZOrder,
4844 /* IDirectDrawSurface2 */
4845 ddraw_surface2_GetDDInterface,
4846 ddraw_surface2_PageLock,
4847 ddraw_surface2_PageUnlock,
4850 static const struct IDirectDrawSurfaceVtbl ddraw_surface1_vtbl =
4852 /* IUnknown */
4853 ddraw_surface1_QueryInterface,
4854 ddraw_surface1_AddRef,
4855 ddraw_surface1_Release,
4856 /* IDirectDrawSurface */
4857 ddraw_surface1_AddAttachedSurface,
4858 ddraw_surface1_AddOverlayDirtyRect,
4859 ddraw_surface1_Blt,
4860 ddraw_surface1_BltBatch,
4861 ddraw_surface1_BltFast,
4862 ddraw_surface1_DeleteAttachedSurface,
4863 ddraw_surface1_EnumAttachedSurfaces,
4864 ddraw_surface1_EnumOverlayZOrders,
4865 ddraw_surface1_Flip,
4866 ddraw_surface1_GetAttachedSurface,
4867 ddraw_surface1_GetBltStatus,
4868 ddraw_surface1_GetCaps,
4869 ddraw_surface1_GetClipper,
4870 ddraw_surface1_GetColorKey,
4871 ddraw_surface1_GetDC,
4872 ddraw_surface1_GetFlipStatus,
4873 ddraw_surface1_GetOverlayPosition,
4874 ddraw_surface1_GetPalette,
4875 ddraw_surface1_GetPixelFormat,
4876 ddraw_surface1_GetSurfaceDesc,
4877 ddraw_surface1_Initialize,
4878 ddraw_surface1_IsLost,
4879 ddraw_surface1_Lock,
4880 ddraw_surface1_ReleaseDC,
4881 ddraw_surface1_Restore,
4882 ddraw_surface1_SetClipper,
4883 ddraw_surface1_SetColorKey,
4884 ddraw_surface1_SetOverlayPosition,
4885 ddraw_surface1_SetPalette,
4886 ddraw_surface1_Unlock,
4887 ddraw_surface1_UpdateOverlay,
4888 ddraw_surface1_UpdateOverlayDisplay,
4889 ddraw_surface1_UpdateOverlayZOrder,
4892 static const struct IDirectDrawGammaControlVtbl ddraw_gamma_control_vtbl =
4894 ddraw_gamma_control_QueryInterface,
4895 ddraw_gamma_control_AddRef,
4896 ddraw_gamma_control_Release,
4897 ddraw_gamma_control_GetGammaRamp,
4898 ddraw_gamma_control_SetGammaRamp,
4901 static const struct IDirect3DTexture2Vtbl d3d_texture2_vtbl =
4903 d3d_texture2_QueryInterface,
4904 d3d_texture2_AddRef,
4905 d3d_texture2_Release,
4906 d3d_texture2_GetHandle,
4907 d3d_texture2_PaletteChanged,
4908 d3d_texture2_Load,
4911 static const struct IDirect3DTextureVtbl d3d_texture1_vtbl =
4913 d3d_texture1_QueryInterface,
4914 d3d_texture1_AddRef,
4915 d3d_texture1_Release,
4916 d3d_texture1_Initialize,
4917 d3d_texture1_GetHandle,
4918 d3d_texture1_PaletteChanged,
4919 d3d_texture1_Load,
4920 d3d_texture1_Unload,
4923 IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface7(IDirectDrawSurface7 *iface)
4925 if (!iface) return NULL;
4926 assert(iface->lpVtbl == &ddraw_surface7_vtbl);
4927 return CONTAINING_RECORD(iface, IDirectDrawSurfaceImpl, IDirectDrawSurface7_iface);
4930 IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface4(IDirectDrawSurface4 *iface)
4932 if (!iface) return NULL;
4933 assert(iface->lpVtbl == &ddraw_surface4_vtbl);
4934 return CONTAINING_RECORD(iface, IDirectDrawSurfaceImpl, IDirectDrawSurface4_iface);
4937 static IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface3(IDirectDrawSurface3 *iface)
4939 if (!iface) return NULL;
4940 assert(iface->lpVtbl == &ddraw_surface3_vtbl);
4941 return CONTAINING_RECORD(iface, IDirectDrawSurfaceImpl, IDirectDrawSurface3_iface);
4944 static IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface2(IDirectDrawSurface2 *iface)
4946 if (!iface) return NULL;
4947 assert(iface->lpVtbl == &ddraw_surface2_vtbl);
4948 return CONTAINING_RECORD(iface, IDirectDrawSurfaceImpl, IDirectDrawSurface2_iface);
4951 IDirectDrawSurfaceImpl *unsafe_impl_from_IDirectDrawSurface(IDirectDrawSurface *iface)
4953 if (!iface) return NULL;
4954 assert(iface->lpVtbl == &ddraw_surface1_vtbl);
4955 return CONTAINING_RECORD(iface, IDirectDrawSurfaceImpl, IDirectDrawSurface_iface);
4958 static void STDMETHODCALLTYPE ddraw_surface_wined3d_object_destroyed(void *parent)
4960 IDirectDrawSurfaceImpl *surface = parent;
4962 TRACE("surface %p.\n", surface);
4964 /* Check for attached surfaces and detach them. */
4965 if (surface->first_attached != surface)
4967 IDirectDrawSurface7 *root = &surface->first_attached->IDirectDrawSurface7_iface;
4968 IDirectDrawSurface7 *detach = &surface->IDirectDrawSurface7_iface;
4970 /* Well, this shouldn't happen: The surface being attached is
4971 * referenced in AddAttachedSurface(), so it shouldn't be released
4972 * until DeleteAttachedSurface() is called, because the refcount is
4973 * held. It looks like the application released it often enough to
4974 * force this. */
4975 WARN("Surface is still attached to surface %p.\n", surface->first_attached);
4977 /* The refcount will drop to -1 here */
4978 if (FAILED(IDirectDrawSurface7_DeleteAttachedSurface(root, 0, detach)))
4979 ERR("DeleteAttachedSurface failed.\n");
4982 while (surface->next_attached)
4984 IDirectDrawSurface7 *root = &surface->IDirectDrawSurface7_iface;
4985 IDirectDrawSurface7 *detach = &surface->next_attached->IDirectDrawSurface7_iface;
4987 if (FAILED(IDirectDrawSurface7_DeleteAttachedSurface(root, 0, detach)))
4988 ERR("DeleteAttachedSurface failed.\n");
4991 /* Having a texture handle set implies that the device still exists. */
4992 if (surface->Handle)
4993 ddraw_free_handle(&surface->ddraw->d3ddevice->handle_table, surface->Handle - 1, DDRAW_HANDLE_SURFACE);
4995 /* Reduce the ddraw surface count. */
4996 InterlockedDecrement(&surface->ddraw->surfaces);
4997 list_remove(&surface->surface_list_entry);
4999 HeapFree(GetProcessHeap(), 0, surface);
5002 const struct wined3d_parent_ops ddraw_surface_wined3d_parent_ops =
5004 ddraw_surface_wined3d_object_destroyed,
5007 static void STDMETHODCALLTYPE ddraw_texture_wined3d_object_destroyed(void *parent)
5009 IDirectDrawSurfaceImpl *surface = parent;
5011 TRACE("surface %p.\n", surface);
5013 ddraw_surface_cleanup(surface);
5016 static const struct wined3d_parent_ops ddraw_texture_wined3d_parent_ops =
5018 ddraw_texture_wined3d_object_destroyed,
5021 HRESULT ddraw_surface_create_texture(IDirectDrawSurfaceImpl *surface)
5023 const DDSURFACEDESC2 *desc = &surface->surface_desc;
5024 enum wined3d_format_id format;
5025 WINED3DPOOL pool;
5026 UINT levels;
5028 if (desc->ddsCaps.dwCaps & DDSCAPS_MIPMAP)
5029 levels = desc->u2.dwMipMapCount;
5030 else
5031 levels = 1;
5033 /* DDSCAPS_SYSTEMMEMORY textures are in WINED3DPOOL_SYSTEMMEM.
5034 * Should I forward the MANAGED cap to the managed pool? */
5035 if (desc->ddsCaps.dwCaps & DDSCAPS_SYSTEMMEMORY)
5036 pool = WINED3DPOOL_SYSTEMMEM;
5037 else
5038 pool = WINED3DPOOL_DEFAULT;
5040 format = PixelFormat_DD2WineD3D(&surface->surface_desc.u4.ddpfPixelFormat);
5041 if (desc->ddsCaps.dwCaps2 & DDSCAPS2_CUBEMAP)
5042 return wined3d_texture_create_cube(surface->ddraw->wined3d_device, desc->dwWidth,
5043 levels, 0, format, pool, surface, &ddraw_texture_wined3d_parent_ops, &surface->wined3d_texture);
5044 else
5045 return wined3d_texture_create_2d(surface->ddraw->wined3d_device, desc->dwWidth, desc->dwHeight,
5046 levels, 0, format, pool, surface, &ddraw_texture_wined3d_parent_ops, &surface->wined3d_texture);
5049 HRESULT ddraw_surface_init(IDirectDrawSurfaceImpl *surface, IDirectDrawImpl *ddraw,
5050 DDSURFACEDESC2 *desc, UINT mip_level, WINED3DSURFTYPE surface_type, UINT version)
5052 struct wined3d_resource_desc wined3d_desc;
5053 struct wined3d_resource *wined3d_resource;
5054 WINED3DPOOL pool = WINED3DPOOL_DEFAULT;
5055 enum wined3d_format_id format;
5056 DWORD usage = 0;
5057 HRESULT hr;
5059 if (!(desc->ddsCaps.dwCaps & (DDSCAPS_VIDEOMEMORY | DDSCAPS_SYSTEMMEMORY))
5060 && !((desc->ddsCaps.dwCaps & DDSCAPS_TEXTURE)
5061 && (desc->ddsCaps.dwCaps2 & DDSCAPS2_TEXTUREMANAGE)))
5063 /* Tests show surfaces without memory flags get these flags added
5064 * right after creation. */
5065 desc->ddsCaps.dwCaps |= DDSCAPS_LOCALVIDMEM | DDSCAPS_VIDEOMEMORY;
5068 if (desc->ddsCaps.dwCaps & DDSCAPS_PRIMARYSURFACE)
5070 usage |= WINED3DUSAGE_RENDERTARGET;
5071 desc->ddsCaps.dwCaps |= DDSCAPS_VISIBLE;
5074 if ((desc->ddsCaps.dwCaps & DDSCAPS_3DDEVICE) && !(desc->ddsCaps.dwCaps & DDSCAPS_ZBUFFER))
5076 usage |= WINED3DUSAGE_RENDERTARGET;
5079 if (desc->ddsCaps.dwCaps & (DDSCAPS_OVERLAY))
5081 usage |= WINED3DUSAGE_OVERLAY;
5084 if (ddraw->depthstencil || (desc->ddsCaps.dwCaps & DDSCAPS_ZBUFFER))
5086 /* The depth stencil creation callback sets this flag. Set the
5087 * wined3d usage to let it know it's a depth/stencil surface. */
5088 usage |= WINED3DUSAGE_DEPTHSTENCIL;
5091 if (desc->ddsCaps.dwCaps & DDSCAPS_SYSTEMMEMORY)
5093 pool = WINED3DPOOL_SYSTEMMEM;
5095 else if (desc->ddsCaps.dwCaps2 & DDSCAPS2_TEXTUREMANAGE)
5097 pool = WINED3DPOOL_MANAGED;
5098 /* Managed textures have the system memory flag set. */
5099 desc->ddsCaps.dwCaps |= DDSCAPS_SYSTEMMEMORY;
5101 else if (desc->ddsCaps.dwCaps & DDSCAPS_VIDEOMEMORY)
5103 /* Videomemory adds localvidmem. This is mutually exclusive with
5104 * systemmemory and texturemanage. */
5105 desc->ddsCaps.dwCaps |= DDSCAPS_LOCALVIDMEM;
5108 format = PixelFormat_DD2WineD3D(&desc->u4.ddpfPixelFormat);
5109 if (format == WINED3DFMT_UNKNOWN)
5111 WARN("Unsupported / unknown pixelformat.\n");
5112 return DDERR_INVALIDPIXELFORMAT;
5115 surface->IDirectDrawSurface7_iface.lpVtbl = &ddraw_surface7_vtbl;
5116 surface->IDirectDrawSurface4_iface.lpVtbl = &ddraw_surface4_vtbl;
5117 surface->IDirectDrawSurface3_iface.lpVtbl = &ddraw_surface3_vtbl;
5118 surface->IDirectDrawSurface2_iface.lpVtbl = &ddraw_surface2_vtbl;
5119 surface->IDirectDrawSurface_iface.lpVtbl = &ddraw_surface1_vtbl;
5120 surface->IDirectDrawGammaControl_iface.lpVtbl = &ddraw_gamma_control_vtbl;
5121 surface->IDirect3DTexture2_vtbl = &d3d_texture2_vtbl;
5122 surface->IDirect3DTexture_vtbl = &d3d_texture1_vtbl;
5123 surface->ref = 1;
5124 surface->iface_count = 1;
5125 surface->version = version;
5126 surface->ddraw = ddraw;
5128 copy_to_surfacedesc2(&surface->surface_desc, desc);
5130 surface->first_attached = surface;
5131 surface->ImplType = surface_type;
5133 hr = wined3d_surface_create(ddraw->wined3d_device, desc->dwWidth, desc->dwHeight, format,
5134 TRUE /* Lockable */, FALSE /* Discard */, mip_level, usage, pool,
5135 WINED3DMULTISAMPLE_NONE, 0 /* MultiSampleQuality */, surface_type, surface,
5136 &ddraw_surface_wined3d_parent_ops, &surface->wined3d_surface);
5137 if (FAILED(hr))
5139 WARN("Failed to create wined3d surface, hr %#x.\n", hr);
5140 return hr;
5143 surface->surface_desc.dwFlags |= DDSD_PIXELFORMAT;
5144 wined3d_resource = wined3d_surface_get_resource(surface->wined3d_surface);
5145 wined3d_resource_get_desc(wined3d_resource, &wined3d_desc);
5147 format = wined3d_desc.format;
5148 if (format == WINED3DFMT_UNKNOWN)
5150 FIXME("IWineD3DSurface::GetDesc returned WINED3DFMT_UNKNOWN.\n");
5152 PixelFormat_WineD3DtoDD(&surface->surface_desc.u4.ddpfPixelFormat, format);
5154 /* Anno 1602 stores the pitch right after surface creation, so make sure
5155 * it's there. TODO: Test other fourcc formats. */
5156 if (format == WINED3DFMT_DXT1 || format == WINED3DFMT_DXT2 || format == WINED3DFMT_DXT3
5157 || format == WINED3DFMT_DXT4 || format == WINED3DFMT_DXT5)
5159 surface->surface_desc.dwFlags |= DDSD_LINEARSIZE;
5160 if (format == WINED3DFMT_DXT1)
5162 surface->surface_desc.u1.dwLinearSize = max(4, wined3d_desc.width) * max(4, wined3d_desc.height) / 2;
5164 else
5166 surface->surface_desc.u1.dwLinearSize = max(4, wined3d_desc.width) * max(4, wined3d_desc.height);
5169 else
5171 surface->surface_desc.dwFlags |= DDSD_PITCH;
5172 surface->surface_desc.u1.lPitch = wined3d_surface_get_pitch(surface->wined3d_surface);
5175 if (desc->dwFlags & DDSD_CKDESTOVERLAY)
5177 wined3d_surface_set_color_key(surface->wined3d_surface, DDCKEY_DESTOVERLAY,
5178 (WINEDDCOLORKEY *)&desc->u3.ddckCKDestOverlay);
5180 if (desc->dwFlags & DDSD_CKDESTBLT)
5182 wined3d_surface_set_color_key(surface->wined3d_surface, DDCKEY_DESTBLT,
5183 (WINEDDCOLORKEY *)&desc->ddckCKDestBlt);
5185 if (desc->dwFlags & DDSD_CKSRCOVERLAY)
5187 wined3d_surface_set_color_key(surface->wined3d_surface, DDCKEY_SRCOVERLAY,
5188 (WINEDDCOLORKEY *)&desc->ddckCKSrcOverlay);
5190 if (desc->dwFlags & DDSD_CKSRCBLT)
5192 wined3d_surface_set_color_key(surface->wined3d_surface, DDCKEY_SRCBLT,
5193 (WINEDDCOLORKEY *)&desc->ddckCKSrcBlt);
5195 if (desc->dwFlags & DDSD_LPSURFACE)
5197 hr = wined3d_surface_set_mem(surface->wined3d_surface, desc->lpSurface);
5198 if (FAILED(hr))
5200 ERR("Failed to set surface memory, hr %#x.\n", hr);
5201 wined3d_surface_decref(surface->wined3d_surface);
5202 return hr;
5206 return DD_OK;