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
22 #include "wine/test.h"
24 HRESULT WINAPI
DXGID3D10CreateDevice(HMODULE d3d10core
, IDXGIFactory
*factory
,
25 IDXGIAdapter
*adapter
, UINT flags
, void *unknown0
, void **device
);
27 static IDXGIDevice
*create_device(HMODULE d3d10core
)
29 IDXGIDevice
*dxgi_device
= NULL
;
30 IDXGIFactory
*factory
= NULL
;
31 IDXGIAdapter
*adapter
= NULL
;
32 IUnknown
*device
= NULL
;
35 hr
= CreateDXGIFactory(&IID_IDXGIFactory
, (void *)&factory
);
36 if (FAILED(hr
)) goto cleanup
;
38 hr
= IDXGIFactory_EnumAdapters(factory
, 0, &adapter
);
41 hr
= DXGID3D10CreateDevice(d3d10core
, factory
, adapter
, 0, NULL
, (void **)&device
);
48 trace("Failed to create a HW device, trying REF\n");
49 if (adapter
) IDXGIAdapter_Release(adapter
);
52 d3d10ref
= LoadLibraryA("d3d10ref.dll");
55 trace("d3d10ref.dll not available, unable to create a REF device\n");
59 hr
= IDXGIFactory_CreateSoftwareAdapter(factory
, d3d10ref
, &adapter
);
60 FreeLibrary(d3d10ref
);
61 ok(SUCCEEDED(hr
), "CreateSoftwareAdapter failed, hr %#x\n", hr
);
62 if (FAILED(hr
)) goto cleanup
;
64 hr
= DXGID3D10CreateDevice(d3d10core
, factory
, adapter
, 0, NULL
, (void **)&device
);
65 ok(SUCCEEDED(hr
), "Failed to create a REF device, hr %#x\n", hr
);
66 if (FAILED(hr
)) goto cleanup
;
69 hr
= IUnknown_QueryInterface(device
, &IID_IDXGIDevice
, (void **)&dxgi_device
);
70 ok(SUCCEEDED(hr
), "Created device does not implement IDXGIDevice\n");
71 IUnknown_Release(device
);
74 if (adapter
) IDXGIAdapter_Release(adapter
);
75 if (factory
) IDXGIFactory_Release(factory
);
80 static void test_device_interfaces(IDXGIDevice
*device
)
85 if (SUCCEEDED(hr
= IDXGIDevice_QueryInterface(device
, &IID_IUnknown
, (void **)&obj
)))
86 IUnknown_Release(obj
);
87 ok(SUCCEEDED(hr
), "IDXGIDevice does not implement IUnknown\n");
89 if (SUCCEEDED(hr
= IDXGIDevice_QueryInterface(device
, &IID_IDXGIObject
, (void **)&obj
)))
90 IUnknown_Release(obj
);
91 ok(SUCCEEDED(hr
), "IDXGIDevice does not implement IDXGIObject\n");
93 if (SUCCEEDED(hr
= IDXGIDevice_QueryInterface(device
, &IID_IDXGIDevice
, (void **)&obj
)))
94 IUnknown_Release(obj
);
95 ok(SUCCEEDED(hr
), "IDXGIDevice does not implement IDXGIDevice\n");
97 if (SUCCEEDED(hr
= IDXGIDevice_QueryInterface(device
, &IID_ID3D10Device
, (void **)&obj
)))
98 IUnknown_Release(obj
);
99 ok(SUCCEEDED(hr
), "IDXGIDevice does not implement ID3D10Device\n");
102 static void test_adapter_desc(IDXGIDevice
*device
)
104 DXGI_ADAPTER_DESC desc
;
105 IDXGIAdapter
*adapter
;
108 hr
= IDXGIDevice_GetAdapter(device
, &adapter
);
109 ok(SUCCEEDED(hr
), "GetAdapter failed, hr %#x.\n", hr
);
111 hr
= IDXGIAdapter_GetDesc(adapter
, NULL
);
112 ok(hr
== E_INVALIDARG
, "GetDesc returned %#x, expected %#x.\n",
115 hr
= IDXGIAdapter_GetDesc(adapter
, &desc
);
116 ok(SUCCEEDED(hr
), "GetDesc failed, hr %#x.\n", hr
);
118 trace("%s.\n", wine_dbgstr_w(desc
.Description
));
119 trace("%04x: %04x:%04x (rev %02x).\n",
120 desc
.SubSysId
, desc
.VendorId
, desc
.DeviceId
, desc
.Revision
);
121 trace("Dedicated video memory: %lu (%lu MB).\n",
122 desc
.DedicatedVideoMemory
, desc
.DedicatedVideoMemory
/ (1024 * 1024));
123 trace("Dedicated system memory: %lu (%lu MB).\n",
124 desc
.DedicatedSystemMemory
, desc
.DedicatedSystemMemory
/ (1024 * 1024));
125 trace("Shared system memory: %lu (%lu MB).\n",
126 desc
.SharedSystemMemory
, desc
.SharedSystemMemory
/ (1024 * 1024));
127 trace("LUID: %08x:%08x.\n", desc
.AdapterLuid
.HighPart
, desc
.AdapterLuid
.LowPart
);
129 IDXGIAdapter_Release(adapter
);
132 static void test_create_surface(IDXGIDevice
*device
)
134 ID3D10Texture2D
*texture
;
135 IDXGISurface
*surface
;
136 DXGI_SURFACE_DESC desc
;
141 desc
.Format
= DXGI_FORMAT_R8G8B8A8_UNORM
;
142 desc
.SampleDesc
.Count
= 1;
143 desc
.SampleDesc
.Quality
= 0;
145 hr
= IDXGIDevice_CreateSurface(device
, &desc
, 1, DXGI_USAGE_RENDER_TARGET_OUTPUT
, NULL
, &surface
);
146 ok(SUCCEEDED(hr
), "Failed to create a dxgi surface, hr %#x\n", hr
);
148 hr
= IDXGISurface_QueryInterface(surface
, &IID_ID3D10Texture2D
, (void **)&texture
);
149 ok(SUCCEEDED(hr
), "Surface should implement ID3D10Texture2D\n");
150 if (SUCCEEDED(hr
)) ID3D10Texture2D_Release(texture
);
152 IDXGISurface_Release(surface
);
155 static void test_parents(IDXGIDevice
*device
)
157 DXGI_SURFACE_DESC surface_desc
;
158 IDXGISurface
*surface
;
159 IDXGIFactory
*factory
;
160 IDXGIAdapter
*adapter
;
165 surface_desc
.Width
= 512;
166 surface_desc
.Height
= 512;
167 surface_desc
.Format
= DXGI_FORMAT_R8G8B8A8_UNORM
;
168 surface_desc
.SampleDesc
.Count
= 1;
169 surface_desc
.SampleDesc
.Quality
= 0;
171 hr
= IDXGIDevice_CreateSurface(device
, &surface_desc
, 1, DXGI_USAGE_RENDER_TARGET_OUTPUT
, NULL
, &surface
);
172 ok(SUCCEEDED(hr
), "Failed to create a dxgi surface, hr %#x\n", hr
);
174 hr
= IDXGISurface_GetParent(surface
, &IID_IDXGIDevice
, (void **)&parent
);
175 IDXGISurface_Release(surface
);
176 ok(SUCCEEDED(hr
), "GetParent failed, hr %#x.\n", hr
);
177 ok(parent
== (IUnknown
*)device
, "Got parent %p, expected %p.\n", parent
, device
);
178 IUnknown_Release(parent
);
180 hr
= IDXGIDevice_GetAdapter(device
, &adapter
);
181 ok(SUCCEEDED(hr
), "GetAdapter failed, hr %#x.\n", hr
);
183 hr
= IDXGIAdapter_EnumOutputs(adapter
, 0, &output
);
184 if (hr
== DXGI_ERROR_NOT_FOUND
)
186 skip("Adapter has not outputs, skipping output tests.\n");
190 ok(SUCCEEDED(hr
), "EnumOutputs failed, hr %#x.\n", hr
);
192 hr
= IDXGIOutput_GetParent(output
, &IID_IDXGIAdapter
, (void **)&parent
);
193 IDXGIOutput_Release(output
);
194 ok(SUCCEEDED(hr
), "GetParent failed, hr %#x.\n", hr
);
195 ok(parent
== (IUnknown
*)adapter
, "Got parent %p, expected %p.\n", parent
, adapter
);
196 IUnknown_Release(parent
);
199 hr
= IDXGIAdapter_GetParent(adapter
, &IID_IDXGIFactory
, (void **)&factory
);
200 ok(SUCCEEDED(hr
), "GetParent failed, hr %#x.\n", hr
);
202 hr
= IDXGIFactory_GetParent(factory
, &IID_IUnknown
, (void **)&parent
);
203 ok(hr
== E_NOINTERFACE
, "GetParent returned %#x, expected %#x.\n", hr
, E_NOINTERFACE
);
204 ok(parent
== NULL
, "Got parent %p, expected %p.\n", parent
, NULL
);
205 IDXGIFactory_Release(factory
);
207 hr
= IDXGIDevice_GetParent(device
, &IID_IDXGIAdapter
, (void **)&parent
);
208 ok(SUCCEEDED(hr
), "GetParent failed, hr %#x.\n", hr
);
209 ok(parent
== (IUnknown
*)adapter
, "Got parent %p, expected %p.\n", parent
, adapter
);
210 IUnknown_Release(parent
);
212 IDXGIAdapter_Release(adapter
);
215 static void test_output(IDXGIDevice
*device
)
217 IDXGIAdapter
*adapter
;
220 UINT mode_count
, mode_count_comp
, i
;
221 DXGI_MODE_DESC
*modes
;
223 hr
= IDXGIDevice_GetAdapter(device
, &adapter
);
224 ok(SUCCEEDED(hr
), "GetAdapter failed, hr %#x.\n", hr
);
226 hr
= IDXGIAdapter_EnumOutputs(adapter
, 0, &output
);
227 if (hr
== DXGI_ERROR_NOT_FOUND
)
229 skip("Adapter has not outputs, skipping output tests.\n");
230 IDXGIAdapter_Release(adapter
);
234 ok(SUCCEEDED(hr
), "EnumOutputs failed, hr %#x.\n", hr
);
236 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, 0, NULL
, NULL
);
237 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
239 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, 0, &mode_count
, NULL
);
240 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
241 mode_count_comp
= mode_count
;
243 IDXGIOutput_GetDisplayModeList(output
, 0, 0, &mode_count
, NULL
);
244 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
245 ok(!mode_count
, "Expected 0 got %d\n", mode_count
);
247 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, DXGI_ENUM_MODES_SCALING
, &mode_count
, NULL
);
248 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
249 ok(mode_count
>= mode_count_comp
, "Flag implies trying to enumerate more modes\n");
250 mode_count_comp
= mode_count
;
252 modes
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(DXGI_MODE_DESC
) * mode_count
+10);
254 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, DXGI_ENUM_MODES_SCALING
, NULL
, modes
);
255 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
256 ok(!modes
[0].Height
, "No output was expected\n");
259 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, DXGI_ENUM_MODES_SCALING
, &mode_count
, modes
);
260 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
261 ok(!modes
[0].Height
, "No output was expected\n");
263 mode_count
= mode_count_comp
;
264 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, DXGI_ENUM_MODES_SCALING
, &mode_count
, modes
);
265 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
266 ok(mode_count
== mode_count_comp
, "Expected %d, got %d\n", mode_count_comp
, mode_count
);
268 for (i
= 0; i
< mode_count
; i
++)
270 ok(modes
[i
].Height
&& modes
[i
].Width
, "Proper mode was expected\n");
274 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, DXGI_ENUM_MODES_SCALING
, &mode_count
, modes
);
275 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
276 ok(mode_count
== mode_count_comp
, "Expected %d, got %d\n", mode_count_comp
, mode_count
);
280 mode_count
= mode_count_comp
- 1;
281 IDXGIOutput_GetDisplayModeList(output
, DXGI_FORMAT_R8G8B8A8_UNORM
, DXGI_ENUM_MODES_SCALING
, &mode_count
, modes
);
282 ok(SUCCEEDED(hr
), "Failed to list modes, hr %#x.\n", hr
);
283 ok(mode_count
== mode_count_comp
-1, "Expected %d, got %d\n", mode_count_comp
, mode_count
);
287 skip("Not enough modes for test, skipping\n");
290 HeapFree(GetProcessHeap(), 0, modes
);
291 IDXGIOutput_Release(output
);
292 IDXGIAdapter_Release(adapter
);
299 BOOL numerator_should_pass
;
300 BOOL denominator_should_pass
;
303 static void test_createswapchain(IDXGIDevice
*device
)
306 IDXGIAdapter
*adapter
;
307 IDXGIFactory
*factory
;
308 IDXGISwapChain
*swapchain
;
309 DXGI_SWAP_CHAIN_DESC creation_desc
, result_desc
;
314 const struct refresh_rates refresh_list
[] =
316 {60, 60, FALSE
, FALSE
},
317 {60, 0, TRUE
, FALSE
},
319 { 0, 60, TRUE
, FALSE
},
320 { 0, 0, TRUE
, FALSE
},
324 wc
.lpfnWndProc
= DefWindowProc
;
325 wc
.lpszClassName
= "dxgi_test_wc";
329 creation_desc
.OutputWindow
= 0;
330 creation_desc
.BufferDesc
.Width
= 800;
331 creation_desc
.BufferDesc
.Height
= 600;
332 creation_desc
.BufferDesc
.RefreshRate
.Numerator
= 60;
333 creation_desc
.BufferDesc
.RefreshRate
.Denominator
= 60;
334 creation_desc
.BufferDesc
.Format
= DXGI_FORMAT_R8G8B8A8_UNORM
;
335 creation_desc
.BufferDesc
.ScanlineOrdering
= DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED
;
336 creation_desc
.BufferDesc
.Scaling
= DXGI_MODE_SCALING_UNSPECIFIED
;
337 creation_desc
.SampleDesc
.Count
= 1;
338 creation_desc
.SampleDesc
.Quality
= 0;
339 creation_desc
.BufferUsage
= DXGI_USAGE_RENDER_TARGET_OUTPUT
;
340 creation_desc
.BufferCount
= 1;
341 creation_desc
.OutputWindow
= CreateWindow("dxgi_test_wc", "dxgi_test", 0, 0, 0, 0, 0, 0, 0, 0, 0);
342 creation_desc
.Windowed
= TRUE
;
343 creation_desc
.SwapEffect
= DXGI_SWAP_EFFECT_DISCARD
;
344 creation_desc
.Flags
= 0;
346 hr
= IDXGIDevice_QueryInterface(device
, &IID_IUnknown
, (void **)&obj
);
347 ok(SUCCEEDED(hr
), "IDXGIDevice does not implement IUnknown\n");
349 hr
= IDXGIDevice_GetAdapter(device
, &adapter
);
350 ok(SUCCEEDED(hr
), "GetAdapter failed, hr %#x.\n", hr
);
352 hr
= IDXGIAdapter_GetParent(adapter
, &IID_IDXGIFactory
, (void **)&factory
);
353 ok(SUCCEEDED(hr
), "GetParent failed, hr %#x.\n", hr
);
355 hr
= IDXGIFactory_CreateSwapChain(factory
, obj
, &creation_desc
, &swapchain
);
356 ok(SUCCEEDED(hr
), "CreateSwapChain failed, hr %#x.\n", hr
);
358 hr
= IDXGISwapChain_GetDesc(swapchain
, NULL
);
359 ok(hr
== E_INVALIDARG
, "GetDesc unexpectedly returned %#x.\n", hr
);
361 IDXGISwapChain_Release(swapchain
);
363 for (i
= 0; i
< sizeof(refresh_list
)/sizeof(refresh_list
[0]); i
++)
365 creation_desc
.BufferDesc
.RefreshRate
.Numerator
= refresh_list
[i
].numerator
;
366 creation_desc
.BufferDesc
.RefreshRate
.Denominator
= refresh_list
[i
].denominator
;
368 hr
= IDXGIFactory_CreateSwapChain(factory
, obj
, &creation_desc
, &swapchain
);
369 ok(SUCCEEDED(hr
), "CreateSwapChain failed, hr %#x.\n", hr
);
371 hr
= IDXGISwapChain_GetDesc(swapchain
, &result_desc
);
372 ok(SUCCEEDED(hr
), "GetDesc failed, hr %#x.\n", hr
);
374 if (refresh_list
[i
].numerator_should_pass
)
375 ok(result_desc
.BufferDesc
.RefreshRate
.Numerator
== refresh_list
[i
].numerator
,
376 "Numerator %u is %u.\n", i
, result_desc
.BufferDesc
.RefreshRate
.Numerator
);
378 todo_wine
ok(result_desc
.BufferDesc
.RefreshRate
.Numerator
== refresh_list
[i
].numerator
,
379 "Numerator %u is %u.\n", i
, result_desc
.BufferDesc
.RefreshRate
.Numerator
);
381 if (refresh_list
[i
].denominator_should_pass
)
382 ok(result_desc
.BufferDesc
.RefreshRate
.Denominator
== refresh_list
[i
].denominator
,
383 "Denominator %u is %u.\n", i
,result_desc
.BufferDesc
.RefreshRate
.Denominator
);
385 todo_wine
ok(result_desc
.BufferDesc
.RefreshRate
.Denominator
== refresh_list
[i
].denominator
,
386 "Denominator %u is %u.\n", i
, result_desc
.BufferDesc
.RefreshRate
.Denominator
);
388 IDXGISwapChain_Release(swapchain
);
391 creation_desc
.Windowed
= FALSE
;
393 for (i
= 0; i
< sizeof(refresh_list
)/sizeof(refresh_list
[0]); i
++)
395 creation_desc
.BufferDesc
.RefreshRate
.Numerator
= refresh_list
[i
].numerator
;
396 creation_desc
.BufferDesc
.RefreshRate
.Denominator
= refresh_list
[i
].denominator
;
398 hr
= IDXGIFactory_CreateSwapChain(factory
, obj
, &creation_desc
, &swapchain
);
399 ok(SUCCEEDED(hr
), "CreateSwapChain failed, hr %#x.\n", hr
);
401 hr
= IDXGISwapChain_GetDesc(swapchain
, &result_desc
);
402 ok(SUCCEEDED(hr
), "GetDesc failed, hr %#x.\n", hr
);
404 if (refresh_list
[i
].numerator_should_pass
)
405 ok(result_desc
.BufferDesc
.RefreshRate
.Numerator
== refresh_list
[i
].numerator
,
406 "Numerator %u is %u.\n", i
, result_desc
.BufferDesc
.RefreshRate
.Numerator
);
408 todo_wine
ok(result_desc
.BufferDesc
.RefreshRate
.Numerator
== refresh_list
[i
].numerator
,
409 "Numerator %u is %u.\n", i
, result_desc
.BufferDesc
.RefreshRate
.Numerator
);
411 if (refresh_list
[i
].denominator_should_pass
)
412 ok(result_desc
.BufferDesc
.RefreshRate
.Denominator
== refresh_list
[i
].denominator
,
413 "Denominator %u is %u.\n", i
,result_desc
.BufferDesc
.RefreshRate
.Denominator
);
415 todo_wine
ok(result_desc
.BufferDesc
.RefreshRate
.Denominator
== refresh_list
[i
].denominator
,
416 "Denominator %u is %u.\n", i
, result_desc
.BufferDesc
.RefreshRate
.Denominator
);
418 hr
= IDXGISwapChain_SetFullscreenState(swapchain
, FALSE
, NULL
);
419 todo_wine
ok(SUCCEEDED(hr
), "SetFullscreenState failed, hr %#x.\n", hr
);
421 IDXGISwapChain_Release(swapchain
);
424 IDXGIFactory_Release(factory
);
425 IDXGIAdapter_Release(adapter
);
426 IUnknown_Release(obj
);
431 HMODULE d3d10core
= LoadLibraryA("d3d10core.dll");
437 win_skip("d3d10core.dll not available, skipping tests\n");
441 device
= create_device(d3d10core
);
444 skip("Failed to create device, skipping tests\n");
445 FreeLibrary(d3d10core
);
449 test_adapter_desc(device
);
450 test_device_interfaces(device
);
451 test_create_surface(device
);
452 test_parents(device
);
454 test_createswapchain(device
);
456 refcount
= IDXGIDevice_Release(device
);
457 ok(!refcount
, "Device has %u references left\n", refcount
);
458 FreeLibrary(d3d10core
);