2 * Copyright 2008 Henri Verbeet for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "wine/port.h"
23 #include "dxgi_private.h"
25 WINE_DEFAULT_DEBUG_CHANNEL(dxgi
);
27 static inline struct dxgi_factory
*impl_from_IWineDXGIFactory(IWineDXGIFactory
*iface
)
29 return CONTAINING_RECORD(iface
, struct dxgi_factory
, IWineDXGIFactory_iface
);
32 /* IUnknown methods */
34 static HRESULT STDMETHODCALLTYPE
dxgi_factory_QueryInterface(IWineDXGIFactory
*iface
, REFIID riid
, void **object
)
36 TRACE("iface %p, riid %s, object %p\n", iface
, debugstr_guid(riid
), object
);
38 if (IsEqualGUID(riid
, &IID_IUnknown
)
39 || IsEqualGUID(riid
, &IID_IDXGIObject
)
40 || IsEqualGUID(riid
, &IID_IDXGIFactory
)
41 || IsEqualGUID(riid
, &IID_IWineDXGIFactory
))
43 IUnknown_AddRef(iface
);
48 WARN("%s not implemented, returning E_NOINTERFACE\n", debugstr_guid(riid
));
54 static ULONG STDMETHODCALLTYPE
dxgi_factory_AddRef(IWineDXGIFactory
*iface
)
56 struct dxgi_factory
*This
= impl_from_IWineDXGIFactory(iface
);
57 ULONG refcount
= InterlockedIncrement(&This
->refcount
);
59 TRACE("%p increasing refcount to %u\n", This
, refcount
);
64 static ULONG STDMETHODCALLTYPE
dxgi_factory_Release(IWineDXGIFactory
*iface
)
66 struct dxgi_factory
*This
= impl_from_IWineDXGIFactory(iface
);
67 ULONG refcount
= InterlockedDecrement(&This
->refcount
);
69 TRACE("%p decreasing refcount to %u\n", This
, refcount
);
75 for (i
= 0; i
< This
->adapter_count
; ++i
)
77 IDXGIAdapter_Release(This
->adapters
[i
]);
79 HeapFree(GetProcessHeap(), 0, This
->adapters
);
81 EnterCriticalSection(&dxgi_cs
);
82 wined3d_decref(This
->wined3d
);
83 LeaveCriticalSection(&dxgi_cs
);
84 HeapFree(GetProcessHeap(), 0, This
);
90 /* IDXGIObject methods */
92 static HRESULT STDMETHODCALLTYPE
dxgi_factory_SetPrivateData(IWineDXGIFactory
*iface
,
93 REFGUID guid
, UINT data_size
, const void *data
)
95 FIXME("iface %p, guid %s, data_size %u, data %p stub!\n", iface
, debugstr_guid(guid
), data_size
, data
);
100 static HRESULT STDMETHODCALLTYPE
dxgi_factory_SetPrivateDataInterface(IWineDXGIFactory
*iface
,
101 REFGUID guid
, const IUnknown
*object
)
103 FIXME("iface %p, guid %s, object %p stub!\n", iface
, debugstr_guid(guid
), object
);
108 static HRESULT STDMETHODCALLTYPE
dxgi_factory_GetPrivateData(IWineDXGIFactory
*iface
,
109 REFGUID guid
, UINT
*data_size
, void *data
)
111 FIXME("iface %p, guid %s, data_size %p, data %p stub!\n", iface
, debugstr_guid(guid
), data_size
, data
);
116 static HRESULT STDMETHODCALLTYPE
dxgi_factory_GetParent(IWineDXGIFactory
*iface
, REFIID riid
, void **parent
)
118 WARN("iface %p, riid %s, parent %p.\n", iface
, debugstr_guid(riid
), parent
);
122 return E_NOINTERFACE
;
125 /* IDXGIFactory methods */
127 static HRESULT STDMETHODCALLTYPE
dxgi_factory_EnumAdapters(IWineDXGIFactory
*iface
,
128 UINT adapter_idx
, IDXGIAdapter
**adapter
)
130 struct dxgi_factory
*This
= impl_from_IWineDXGIFactory(iface
);
132 TRACE("iface %p, adapter_idx %u, adapter %p\n", iface
, adapter_idx
, adapter
);
134 if (!adapter
) return DXGI_ERROR_INVALID_CALL
;
136 if (adapter_idx
>= This
->adapter_count
)
139 return DXGI_ERROR_NOT_FOUND
;
142 *adapter
= This
->adapters
[adapter_idx
];
143 IDXGIAdapter_AddRef(*adapter
);
145 TRACE("Returning adapter %p\n", *adapter
);
150 static HRESULT STDMETHODCALLTYPE
dxgi_factory_MakeWindowAssociation(IWineDXGIFactory
*iface
, HWND window
, UINT flags
)
152 FIXME("iface %p, window %p, flags %#x stub!\n\n", iface
, window
, flags
);
157 static HRESULT STDMETHODCALLTYPE
dxgi_factory_GetWindowAssociation(IWineDXGIFactory
*iface
, HWND
*window
)
159 FIXME("iface %p, window %p stub!\n", iface
, window
);
164 static UINT
dxgi_rational_to_uint(const DXGI_RATIONAL
*rational
)
166 if (rational
->Denominator
)
167 return rational
->Numerator
/ rational
->Denominator
;
169 return rational
->Numerator
;
172 static HRESULT STDMETHODCALLTYPE
dxgi_factory_CreateSwapChain(IWineDXGIFactory
*iface
,
173 IUnknown
*device
, DXGI_SWAP_CHAIN_DESC
*desc
, IDXGISwapChain
**swapchain
)
175 struct wined3d_swapchain
*wined3d_swapchain
;
176 struct wined3d_swapchain_desc wined3d_desc
;
177 struct wined3d_device
*wined3d_device
;
178 IWineDXGIDevice
*dxgi_device
;
182 FIXME("iface %p, device %p, desc %p, swapchain %p partial stub!\n", iface
, device
, desc
, swapchain
);
184 hr
= IUnknown_QueryInterface(device
, &IID_IWineDXGIDevice
, (void **)&dxgi_device
);
187 ERR("This is not the device we're looking for\n");
191 wined3d_device
= IWineDXGIDevice_get_wined3d_device(dxgi_device
);
192 IWineDXGIDevice_Release(dxgi_device
);
194 count
= wined3d_device_get_swapchain_count(wined3d_device
);
197 FIXME("Only a single swapchain supported.\n");
198 wined3d_device_decref(wined3d_device
);
202 if (!desc
->OutputWindow
)
204 FIXME("No output window, should use factory output window\n");
207 FIXME("Ignoring SwapEffect and Flags\n");
209 wined3d_desc
.backbuffer_width
= desc
->BufferDesc
.Width
;
210 wined3d_desc
.backbuffer_height
= desc
->BufferDesc
.Height
;
211 wined3d_desc
.backbuffer_format
= wined3dformat_from_dxgi_format(desc
->BufferDesc
.Format
);
212 wined3d_desc
.backbuffer_count
= desc
->BufferCount
;
213 if (desc
->SampleDesc
.Count
> 1)
215 wined3d_desc
.multisample_type
= desc
->SampleDesc
.Count
;
216 wined3d_desc
.multisample_quality
= desc
->SampleDesc
.Quality
;
220 wined3d_desc
.multisample_type
= WINED3D_MULTISAMPLE_NONE
;
221 wined3d_desc
.multisample_quality
= 0;
223 wined3d_desc
.swap_effect
= WINED3D_SWAP_EFFECT_DISCARD
;
224 wined3d_desc
.device_window
= desc
->OutputWindow
;
225 wined3d_desc
.windowed
= desc
->Windowed
;
226 wined3d_desc
.enable_auto_depth_stencil
= FALSE
;
227 wined3d_desc
.auto_depth_stencil_format
= 0;
228 wined3d_desc
.flags
= 0; /* WINED3DPRESENTFLAG_DISCARD_DEPTHSTENCIL? */
229 wined3d_desc
.refresh_rate
= dxgi_rational_to_uint(&desc
->BufferDesc
.RefreshRate
);
230 wined3d_desc
.swap_interval
= WINED3DPRESENT_INTERVAL_DEFAULT
;
232 hr
= wined3d_device_init_3d(wined3d_device
, &wined3d_desc
);
235 WARN("Failed to initialize 3D, returning %#x\n", hr
);
236 wined3d_device_decref(wined3d_device
);
240 wined3d_swapchain
= wined3d_device_get_swapchain(wined3d_device
, 0);
241 wined3d_device_decref(wined3d_device
);
242 if (!wined3d_swapchain
)
244 WARN("Failed to get swapchain.\n");
248 *swapchain
= wined3d_swapchain_get_parent(wined3d_swapchain
);
250 /* FIXME? The swapchain is created with refcount 1 by the wined3d device,
251 * but the wined3d device can't hold a real reference. */
253 TRACE("Created IDXGISwapChain %p\n", *swapchain
);
258 static HRESULT STDMETHODCALLTYPE
dxgi_factory_CreateSoftwareAdapter(IWineDXGIFactory
*iface
,
259 HMODULE swrast
, IDXGIAdapter
**adapter
)
261 FIXME("iface %p, swrast %p, adapter %p stub!\n", iface
, swrast
, adapter
);
266 /* IWineDXGIFactory methods */
268 static struct wined3d
* STDMETHODCALLTYPE
dxgi_factory_get_wined3d(IWineDXGIFactory
*iface
)
270 struct dxgi_factory
*This
= impl_from_IWineDXGIFactory(iface
);
272 TRACE("iface %p\n", iface
);
274 EnterCriticalSection(&dxgi_cs
);
275 wined3d_incref(This
->wined3d
);
276 LeaveCriticalSection(&dxgi_cs
);
277 return This
->wined3d
;
280 static const struct IWineDXGIFactoryVtbl dxgi_factory_vtbl
=
282 /* IUnknown methods */
283 dxgi_factory_QueryInterface
,
285 dxgi_factory_Release
,
286 /* IDXGIObject methods */
287 dxgi_factory_SetPrivateData
,
288 dxgi_factory_SetPrivateDataInterface
,
289 dxgi_factory_GetPrivateData
,
290 dxgi_factory_GetParent
,
291 /* IDXGIFactory methods */
292 dxgi_factory_EnumAdapters
,
293 dxgi_factory_MakeWindowAssociation
,
294 dxgi_factory_GetWindowAssociation
,
295 dxgi_factory_CreateSwapChain
,
296 dxgi_factory_CreateSoftwareAdapter
,
297 /* IWineDXGIFactory methods */
298 dxgi_factory_get_wined3d
,
301 HRESULT
dxgi_factory_init(struct dxgi_factory
*factory
)
306 factory
->IWineDXGIFactory_iface
.lpVtbl
= &dxgi_factory_vtbl
;
307 factory
->refcount
= 1;
309 EnterCriticalSection(&dxgi_cs
);
310 factory
->wined3d
= wined3d_create(10, 0);
311 if (!factory
->wined3d
)
313 LeaveCriticalSection(&dxgi_cs
);
314 return DXGI_ERROR_UNSUPPORTED
;
317 factory
->adapter_count
= wined3d_get_adapter_count(factory
->wined3d
);
318 LeaveCriticalSection(&dxgi_cs
);
319 factory
->adapters
= HeapAlloc(GetProcessHeap(), 0, factory
->adapter_count
* sizeof(*factory
->adapters
));
320 if (!factory
->adapters
)
322 ERR("Failed to allocate DXGI adapter array memory.\n");
327 for (i
= 0; i
< factory
->adapter_count
; ++i
)
329 struct dxgi_adapter
*adapter
= HeapAlloc(GetProcessHeap(), 0, sizeof(*adapter
));
334 ERR("Failed to allocate DXGI adapter memory.\n");
336 for (j
= 0; j
< i
; ++j
)
338 IDXGIAdapter_Release(factory
->adapters
[j
]);
344 hr
= dxgi_adapter_init(adapter
, &factory
->IWineDXGIFactory_iface
, i
);
349 ERR("Failed to initialize adapter, hr %#x.\n", hr
);
351 HeapFree(GetProcessHeap(), 0, adapter
);
352 for (j
= 0; j
< i
; ++j
)
354 IDXGIAdapter_Release(factory
->adapters
[j
]);
359 factory
->adapters
[i
] = (IDXGIAdapter
*)adapter
;
365 HeapFree(GetProcessHeap(), 0, factory
->adapters
);
366 EnterCriticalSection(&dxgi_cs
);
367 wined3d_decref(factory
->wined3d
);
368 LeaveCriticalSection(&dxgi_cs
);