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_adapter
*impl_from_IWineDXGIAdapter(IWineDXGIAdapter
*iface
)
29 return CONTAINING_RECORD(iface
, struct dxgi_adapter
, IWineDXGIAdapter_iface
);
32 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_QueryInterface(IWineDXGIAdapter
*iface
, REFIID iid
, void **out
)
34 TRACE("iface %p, iid %s, out %p.\n", iface
, debugstr_guid(iid
), out
);
36 if (IsEqualGUID(iid
, &IID_IWineDXGIAdapter
)
37 || IsEqualGUID(iid
, &IID_IDXGIAdapter4
)
38 || IsEqualGUID(iid
, &IID_IDXGIAdapter3
)
39 || IsEqualGUID(iid
, &IID_IDXGIAdapter2
)
40 || IsEqualGUID(iid
, &IID_IDXGIAdapter1
)
41 || IsEqualGUID(iid
, &IID_IDXGIAdapter
)
42 || IsEqualGUID(iid
, &IID_IDXGIObject
)
43 || IsEqualGUID(iid
, &IID_IUnknown
))
45 IUnknown_AddRef(iface
);
50 WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid
));
56 static ULONG STDMETHODCALLTYPE
dxgi_adapter_AddRef(IWineDXGIAdapter
*iface
)
58 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
59 ULONG refcount
= InterlockedIncrement(&adapter
->refcount
);
61 TRACE("%p increasing refcount to %u.\n", iface
, refcount
);
66 static ULONG STDMETHODCALLTYPE
dxgi_adapter_Release(IWineDXGIAdapter
*iface
)
68 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
69 ULONG refcount
= InterlockedDecrement(&adapter
->refcount
);
71 TRACE("%p decreasing refcount to %u.\n", iface
, refcount
);
75 wined3d_private_store_cleanup(&adapter
->private_store
);
76 IWineDXGIFactory_Release(&adapter
->factory
->IWineDXGIFactory_iface
);
83 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_SetPrivateData(IWineDXGIAdapter
*iface
,
84 REFGUID guid
, UINT data_size
, const void *data
)
86 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
88 TRACE("iface %p, guid %s, data_size %u, data %p.\n", iface
, debugstr_guid(guid
), data_size
, data
);
90 return dxgi_set_private_data(&adapter
->private_store
, guid
, data_size
, data
);
93 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_SetPrivateDataInterface(IWineDXGIAdapter
*iface
,
94 REFGUID guid
, const IUnknown
*object
)
96 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
98 TRACE("iface %p, guid %s, object %p.\n", iface
, debugstr_guid(guid
), object
);
100 return dxgi_set_private_data_interface(&adapter
->private_store
, guid
, object
);
103 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_GetPrivateData(IWineDXGIAdapter
*iface
,
104 REFGUID guid
, UINT
*data_size
, void *data
)
106 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
108 TRACE("iface %p, guid %s, data_size %p, data %p.\n", iface
, debugstr_guid(guid
), data_size
, data
);
110 return dxgi_get_private_data(&adapter
->private_store
, guid
, data_size
, data
);
113 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_GetParent(IWineDXGIAdapter
*iface
, REFIID iid
, void **parent
)
115 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
117 TRACE("iface %p, iid %s, parent %p.\n", iface
, debugstr_guid(iid
), parent
);
119 return IWineDXGIFactory_QueryInterface(&adapter
->factory
->IWineDXGIFactory_iface
, iid
, parent
);
122 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_EnumOutputs(IWineDXGIAdapter
*iface
,
123 UINT output_idx
, IDXGIOutput
**output
)
125 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
126 struct dxgi_output
*output_object
;
127 unsigned int output_count
;
130 TRACE("iface %p, output_idx %u, output %p.\n", iface
, output_idx
, output
);
135 output_count
= wined3d_adapter_get_output_count(adapter
->wined3d_adapter
);
136 if (output_idx
>= output_count
)
139 return DXGI_ERROR_NOT_FOUND
;
142 if (FAILED(hr
= dxgi_output_create(adapter
, output_idx
, &output_object
)))
148 *output
= (IDXGIOutput
*)&output_object
->IDXGIOutput6_iface
;
150 TRACE("Returning output %p.\n", *output
);
155 static HRESULT
dxgi_adapter_get_desc(struct dxgi_adapter
*adapter
, DXGI_ADAPTER_DESC3
*desc
)
157 char description
[ARRAY_SIZE(desc
->Description
)];
158 struct wined3d_adapter_identifier adapter_id
;
161 adapter_id
.driver_size
= 0;
162 adapter_id
.description
= description
;
163 adapter_id
.description_size
= sizeof(description
);
165 if (FAILED(hr
= wined3d_adapter_get_identifier(adapter
->wined3d_adapter
, 0, &adapter_id
)))
168 if (!MultiByteToWideChar(CP_ACP
, 0, description
, -1, desc
->Description
, ARRAY_SIZE(description
)))
170 DWORD err
= GetLastError();
171 ERR("Failed to translate description %s (%#x).\n", debugstr_a(description
), err
);
175 desc
->VendorId
= adapter_id
.vendor_id
;
176 desc
->DeviceId
= adapter_id
.device_id
;
177 desc
->SubSysId
= adapter_id
.subsystem_id
;
178 desc
->Revision
= adapter_id
.revision
;
179 desc
->DedicatedVideoMemory
= adapter_id
.video_memory
;
180 desc
->DedicatedSystemMemory
= 0; /* FIXME */
181 desc
->SharedSystemMemory
= adapter_id
.shared_system_memory
;
182 desc
->AdapterLuid
= adapter_id
.adapter_luid
;
184 desc
->GraphicsPreemptionGranularity
= 0; /* FIXME */
185 desc
->ComputePreemptionGranularity
= 0; /* FIXME */
190 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_GetDesc(IWineDXGIAdapter
*iface
, DXGI_ADAPTER_DESC
*desc
)
192 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
193 DXGI_ADAPTER_DESC3 desc3
;
196 TRACE("iface %p, desc %p.\n", iface
, desc
);
201 if (SUCCEEDED(hr
= dxgi_adapter_get_desc(adapter
, &desc3
)))
202 memcpy(desc
, &desc3
, sizeof(*desc
));
207 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_CheckInterfaceSupport(IWineDXGIAdapter
*iface
,
208 REFGUID guid
, LARGE_INTEGER
*umd_version
)
210 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
211 struct wined3d_adapter_identifier adapter_id
;
212 struct wined3d_caps caps
;
215 TRACE("iface %p, guid %s, umd_version %p.\n", iface
, debugstr_guid(guid
), umd_version
);
217 /* This method works only for D3D10 interfaces. */
218 if (!(IsEqualGUID(guid
, &IID_IDXGIDevice
)
219 || IsEqualGUID(guid
, &IID_ID3D10Device
)
220 || IsEqualGUID(guid
, &IID_ID3D10Device1
)))
222 WARN("Returning DXGI_ERROR_UNSUPPORTED for %s.\n", debugstr_guid(guid
));
223 return DXGI_ERROR_UNSUPPORTED
;
226 adapter_id
.driver_size
= 0;
227 adapter_id
.description_size
= 0;
229 wined3d_mutex_lock();
230 hr
= wined3d_get_device_caps(adapter
->wined3d_adapter
, WINED3D_DEVICE_TYPE_HAL
, &caps
);
232 hr
= wined3d_adapter_get_identifier(adapter
->wined3d_adapter
, 0, &adapter_id
);
233 wined3d_mutex_unlock();
237 if (caps
.max_feature_level
< WINED3D_FEATURE_LEVEL_10
)
238 return DXGI_ERROR_UNSUPPORTED
;
241 *umd_version
= adapter_id
.driver_version
;
246 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_GetDesc1(IWineDXGIAdapter
*iface
, DXGI_ADAPTER_DESC1
*desc
)
248 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
249 DXGI_ADAPTER_DESC3 desc3
;
252 TRACE("iface %p, desc %p.\n", iface
, desc
);
257 if (SUCCEEDED(hr
= dxgi_adapter_get_desc(adapter
, &desc3
)))
258 memcpy(desc
, &desc3
, sizeof(*desc
));
263 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_GetDesc2(IWineDXGIAdapter
*iface
, DXGI_ADAPTER_DESC2
*desc
)
265 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
266 DXGI_ADAPTER_DESC3 desc3
;
269 TRACE("iface %p, desc %p.\n", iface
, desc
);
274 if (SUCCEEDED(hr
= dxgi_adapter_get_desc(adapter
, &desc3
)))
275 memcpy(desc
, &desc3
, sizeof(*desc
));
280 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_RegisterHardwareContentProtectionTeardownStatusEvent(
281 IWineDXGIAdapter
*iface
, HANDLE event
, DWORD
*cookie
)
283 FIXME("iface %p, event %p, cookie %p stub!\n", iface
, event
, cookie
);
288 static void STDMETHODCALLTYPE
dxgi_adapter_UnregisterHardwareContentProtectionTeardownStatus(
289 IWineDXGIAdapter
*iface
, DWORD cookie
)
291 FIXME("iface %p, cookie %#x stub!\n", iface
, cookie
);
294 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_QueryVideoMemoryInfo(IWineDXGIAdapter
*iface
,
295 UINT node_index
, DXGI_MEMORY_SEGMENT_GROUP segment_group
, DXGI_QUERY_VIDEO_MEMORY_INFO
*info
)
297 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
298 struct wined3d_adapter_identifier adapter_id
;
299 static unsigned int once
;
302 TRACE("iface %p, node_index %u, segment_group %#x, info %p.\n",
303 iface
, node_index
, segment_group
, info
);
306 FIXME("Returning fake video memory info.\n");
309 FIXME("Ignoring node index %u.\n", node_index
);
311 adapter_id
.driver_size
= 0;
312 adapter_id
.description_size
= 0;
314 if (FAILED(hr
= wined3d_adapter_get_identifier(adapter
->wined3d_adapter
, 0, &adapter_id
)))
317 switch (segment_group
)
319 case DXGI_MEMORY_SEGMENT_GROUP_LOCAL
:
320 info
->Budget
= adapter_id
.video_memory
;
321 info
->CurrentUsage
= 0;
322 info
->AvailableForReservation
= adapter_id
.video_memory
/ 2;
323 info
->CurrentReservation
= 0;
325 case DXGI_MEMORY_SEGMENT_GROUP_NON_LOCAL
:
326 memset(info
, 0, sizeof(*info
));
329 WARN("Invalid memory segment group %#x.\n", segment_group
);
333 TRACE("Budget 0x%s, usage 0x%s, available for reservation 0x%s, reservation 0x%s.\n",
334 wine_dbgstr_longlong(info
->Budget
), wine_dbgstr_longlong(info
->CurrentUsage
),
335 wine_dbgstr_longlong(info
->AvailableForReservation
), wine_dbgstr_longlong(info
->CurrentReservation
));
340 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_SetVideoMemoryReservation(IWineDXGIAdapter
*iface
,
341 UINT node_index
, DXGI_MEMORY_SEGMENT_GROUP segment_group
, UINT64 reservation
)
343 FIXME("iface %p, node_index %u, segment_group %#x, reservation 0x%s stub!\n",
344 iface
, node_index
, segment_group
, wine_dbgstr_longlong(reservation
));
349 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_RegisterVideoMemoryBudgetChangeNotificationEvent(
350 IWineDXGIAdapter
*iface
, HANDLE event
, DWORD
*cookie
)
352 FIXME("iface %p, event %p, cookie %p stub!\n", iface
, event
, cookie
);
357 static void STDMETHODCALLTYPE
dxgi_adapter_UnregisterVideoMemoryBudgetChangeNotification(
358 IWineDXGIAdapter
*iface
, DWORD cookie
)
360 FIXME("iface %p, cookie %#x stub!\n", iface
, cookie
);
363 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_GetDesc3(IWineDXGIAdapter
*iface
, DXGI_ADAPTER_DESC3
*desc
)
365 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
367 TRACE("iface %p, desc %p.\n", iface
, desc
);
372 return dxgi_adapter_get_desc(adapter
, desc
);
375 static HRESULT STDMETHODCALLTYPE
dxgi_adapter_get_adapter_info(IWineDXGIAdapter
*iface
,
376 struct wine_dxgi_adapter_info
*info
)
378 struct dxgi_adapter
*adapter
= impl_from_IWineDXGIAdapter(iface
);
379 struct wined3d_adapter_identifier adapter_id
;
382 TRACE("iface %p, info %p.\n", iface
, info
);
384 memset(&adapter_id
, 0, sizeof(adapter_id
));
385 if (SUCCEEDED(hr
= wined3d_adapter_get_identifier(adapter
->wined3d_adapter
, 0, &adapter_id
)))
387 info
->driver_uuid
= adapter_id
.driver_uuid
;
388 info
->device_uuid
= adapter_id
.device_uuid
;
389 info
->vendor_id
= adapter_id
.vendor_id
;
390 info
->device_id
= adapter_id
.device_id
;
391 info
->luid
= adapter_id
.adapter_luid
;
397 static const struct IWineDXGIAdapterVtbl dxgi_adapter_vtbl
=
399 dxgi_adapter_QueryInterface
,
401 dxgi_adapter_Release
,
402 dxgi_adapter_SetPrivateData
,
403 dxgi_adapter_SetPrivateDataInterface
,
404 dxgi_adapter_GetPrivateData
,
405 dxgi_adapter_GetParent
,
406 /* IDXGIAdapter methods */
407 dxgi_adapter_EnumOutputs
,
408 dxgi_adapter_GetDesc
,
409 dxgi_adapter_CheckInterfaceSupport
,
410 /* IDXGIAdapter1 methods */
411 dxgi_adapter_GetDesc1
,
412 /* IDXGIAdapter2 methods */
413 dxgi_adapter_GetDesc2
,
414 /* IDXGIAdapter3 methods */
415 dxgi_adapter_RegisterHardwareContentProtectionTeardownStatusEvent
,
416 dxgi_adapter_UnregisterHardwareContentProtectionTeardownStatus
,
417 dxgi_adapter_QueryVideoMemoryInfo
,
418 dxgi_adapter_SetVideoMemoryReservation
,
419 dxgi_adapter_RegisterVideoMemoryBudgetChangeNotificationEvent
,
420 dxgi_adapter_UnregisterVideoMemoryBudgetChangeNotification
,
421 /* IDXGIAdapter4 methods */
422 dxgi_adapter_GetDesc3
,
423 /* IWineDXGIAdapter methods */
424 dxgi_adapter_get_adapter_info
,
427 struct dxgi_adapter
*unsafe_impl_from_IDXGIAdapter(IDXGIAdapter
*iface
)
429 IWineDXGIAdapter
*wine_adapter
;
430 struct dxgi_adapter
*adapter
;
435 if (FAILED(hr
= IDXGIAdapter_QueryInterface(iface
, &IID_IWineDXGIAdapter
, (void **)&wine_adapter
)))
437 ERR("Failed to get IWineDXGIAdapter interface, hr %#x.\n", hr
);
440 assert(wine_adapter
->lpVtbl
== &dxgi_adapter_vtbl
);
441 adapter
= CONTAINING_RECORD(wine_adapter
, struct dxgi_adapter
, IWineDXGIAdapter_iface
);
442 IWineDXGIAdapter_Release(wine_adapter
);
446 static void dxgi_adapter_init(struct dxgi_adapter
*adapter
, struct dxgi_factory
*factory
, UINT ordinal
)
448 adapter
->IWineDXGIAdapter_iface
.lpVtbl
= &dxgi_adapter_vtbl
;
449 adapter
->refcount
= 1;
450 adapter
->wined3d_adapter
= wined3d_get_adapter(factory
->wined3d
, ordinal
);
451 wined3d_private_store_init(&adapter
->private_store
);
452 adapter
->ordinal
= ordinal
;
453 adapter
->factory
= factory
;
454 IWineDXGIFactory_AddRef(&adapter
->factory
->IWineDXGIFactory_iface
);
457 HRESULT
dxgi_adapter_create(struct dxgi_factory
*factory
, UINT ordinal
, struct dxgi_adapter
**adapter
)
459 if (!(*adapter
= heap_alloc(sizeof(**adapter
))))
460 return E_OUTOFMEMORY
;
462 dxgi_adapter_init(*adapter
, factory
, ordinal
);