325 lines
12 KiB
C
325 lines
12 KiB
C
/*
|
|
* Copyright 2008 Henri Verbeet for CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#define COBJMACROS
|
|
#include "initguid.h"
|
|
#include "d3d10.h"
|
|
#include "wine/test.h"
|
|
|
|
HRESULT WINAPI DXGID3D10CreateDevice(HMODULE d3d10core, IDXGIFactory *factory,
|
|
IDXGIAdapter *adapter, UINT flags, void *unknown0, void **device);
|
|
|
|
static IDXGIDevice *create_device(HMODULE d3d10core)
|
|
{
|
|
IDXGIDevice *dxgi_device = NULL;
|
|
IDXGIFactory *factory = NULL;
|
|
IDXGIAdapter *adapter = NULL;
|
|
IUnknown *device = NULL;
|
|
HRESULT hr;
|
|
|
|
hr = CreateDXGIFactory(&IID_IDXGIFactory, (void *)&factory);
|
|
if (FAILED(hr)) goto cleanup;
|
|
|
|
hr = IDXGIFactory_EnumAdapters(factory, 0, &adapter);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = DXGID3D10CreateDevice(d3d10core, factory, adapter, 0, NULL, (void **)&device);
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
HMODULE d3d10ref;
|
|
|
|
trace("Failed to create a HW device, trying REF\n");
|
|
if (adapter) IDXGIAdapter_Release(adapter);
|
|
adapter = NULL;
|
|
|
|
d3d10ref = LoadLibraryA("d3d10ref.dll");
|
|
if (!d3d10ref)
|
|
{
|
|
trace("d3d10ref.dll not available, unable to create a REF device\n");
|
|
goto cleanup;
|
|
}
|
|
|
|
hr = IDXGIFactory_CreateSoftwareAdapter(factory, d3d10ref, &adapter);
|
|
FreeLibrary(d3d10ref);
|
|
ok(SUCCEEDED(hr), "CreateSoftwareAdapter failed, hr %#x\n", hr);
|
|
if (FAILED(hr)) goto cleanup;
|
|
|
|
hr = DXGID3D10CreateDevice(d3d10core, factory, adapter, 0, NULL, (void **)&device);
|
|
ok(SUCCEEDED(hr), "Failed to create a REF device, hr %#x\n", hr);
|
|
if (FAILED(hr)) goto cleanup;
|
|
}
|
|
|
|
hr = IUnknown_QueryInterface(device, &IID_IDXGIDevice, (void **)&dxgi_device);
|
|
ok(SUCCEEDED(hr), "Created device does not implement IDXGIDevice\n");
|
|
IUnknown_Release(device);
|
|
|
|
cleanup:
|
|
if (adapter) IDXGIAdapter_Release(adapter);
|
|
if (factory) IDXGIFactory_Release(factory);
|
|
|
|
return dxgi_device;
|
|
}
|
|
|
|
static void test_device_interfaces(IDXGIDevice *device)
|
|
{
|
|
IUnknown *obj;
|
|
HRESULT hr;
|
|
|
|
if (SUCCEEDED(hr = IDXGIDevice_QueryInterface(device, &IID_IUnknown, (void **)&obj)))
|
|
IUnknown_Release(obj);
|
|
ok(SUCCEEDED(hr), "IDXGIDevice does not implement IUnknown\n");
|
|
|
|
if (SUCCEEDED(hr = IDXGIDevice_QueryInterface(device, &IID_IDXGIObject, (void **)&obj)))
|
|
IUnknown_Release(obj);
|
|
ok(SUCCEEDED(hr), "IDXGIDevice does not implement IDXGIObject\n");
|
|
|
|
if (SUCCEEDED(hr = IDXGIDevice_QueryInterface(device, &IID_IDXGIDevice, (void **)&obj)))
|
|
IUnknown_Release(obj);
|
|
ok(SUCCEEDED(hr), "IDXGIDevice does not implement IDXGIDevice\n");
|
|
|
|
if (SUCCEEDED(hr = IDXGIDevice_QueryInterface(device, &IID_ID3D10Device, (void **)&obj)))
|
|
IUnknown_Release(obj);
|
|
ok(SUCCEEDED(hr), "IDXGIDevice does not implement ID3D10Device\n");
|
|
}
|
|
|
|
static void test_adapter_desc(IDXGIDevice *device)
|
|
{
|
|
DXGI_ADAPTER_DESC desc;
|
|
IDXGIAdapter *adapter;
|
|
HRESULT hr;
|
|
|
|
hr = IDXGIDevice_GetAdapter(device, &adapter);
|
|
ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr);
|
|
|
|
hr = IDXGIAdapter_GetDesc(adapter, NULL);
|
|
ok(hr == E_INVALIDARG, "GetDesc returned %#x, expected %#x.\n",
|
|
hr, E_INVALIDARG);
|
|
|
|
hr = IDXGIAdapter_GetDesc(adapter, &desc);
|
|
ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr);
|
|
|
|
trace("%s.\n", wine_dbgstr_w(desc.Description));
|
|
trace("%04x: %04x:%04x (rev %02x).\n",
|
|
desc.SubSysId, desc.VendorId, desc.DeviceId, desc.Revision);
|
|
trace("Dedicated video memory: %lu (%lu MB).\n",
|
|
desc.DedicatedVideoMemory, desc.DedicatedVideoMemory / (1024 * 1024));
|
|
trace("Dedicated system memory: %lu (%lu MB).\n",
|
|
desc.DedicatedSystemMemory, desc.DedicatedSystemMemory / (1024 * 1024));
|
|
trace("Shared system memory: %lu (%lu MB).\n",
|
|
desc.SharedSystemMemory, desc.SharedSystemMemory / (1024 * 1024));
|
|
trace("LUID: %08x:%08x.\n", desc.AdapterLuid.HighPart, desc.AdapterLuid.LowPart);
|
|
|
|
IDXGIAdapter_Release(adapter);
|
|
}
|
|
|
|
static void test_create_surface(IDXGIDevice *device)
|
|
{
|
|
ID3D10Texture2D *texture;
|
|
IDXGISurface *surface;
|
|
DXGI_SURFACE_DESC desc;
|
|
HRESULT hr;
|
|
|
|
desc.Width = 512;
|
|
desc.Height = 512;
|
|
desc.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
|
|
desc.SampleDesc.Count = 1;
|
|
desc.SampleDesc.Quality = 0;
|
|
|
|
hr = IDXGIDevice_CreateSurface(device, &desc, 1, DXGI_USAGE_RENDER_TARGET_OUTPUT, NULL, &surface);
|
|
ok(SUCCEEDED(hr), "Failed to create a dxgi surface, hr %#x\n", hr);
|
|
|
|
hr = IDXGISurface_QueryInterface(surface, &IID_ID3D10Texture2D, (void **)&texture);
|
|
ok(SUCCEEDED(hr), "Surface should implement ID3D10Texture2D\n");
|
|
if (SUCCEEDED(hr)) ID3D10Texture2D_Release(texture);
|
|
|
|
IDXGISurface_Release(surface);
|
|
}
|
|
|
|
static void test_parents(IDXGIDevice *device)
|
|
{
|
|
DXGI_SURFACE_DESC surface_desc;
|
|
IDXGISurface *surface;
|
|
IDXGIFactory *factory;
|
|
IDXGIAdapter *adapter;
|
|
IDXGIOutput *output;
|
|
IUnknown *parent;
|
|
HRESULT hr;
|
|
|
|
surface_desc.Width = 512;
|
|
surface_desc.Height = 512;
|
|
surface_desc.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
|
|
surface_desc.SampleDesc.Count = 1;
|
|
surface_desc.SampleDesc.Quality = 0;
|
|
|
|
hr = IDXGIDevice_CreateSurface(device, &surface_desc, 1, DXGI_USAGE_RENDER_TARGET_OUTPUT, NULL, &surface);
|
|
ok(SUCCEEDED(hr), "Failed to create a dxgi surface, hr %#x\n", hr);
|
|
|
|
hr = IDXGISurface_GetParent(surface, &IID_IDXGIDevice, (void **)&parent);
|
|
IDXGISurface_Release(surface);
|
|
ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr);
|
|
ok(parent == (IUnknown *)device, "Got parent %p, expected %p.\n", parent, device);
|
|
IUnknown_Release(parent);
|
|
|
|
hr = IDXGIDevice_GetAdapter(device, &adapter);
|
|
ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr);
|
|
|
|
hr = IDXGIAdapter_EnumOutputs(adapter, 0, &output);
|
|
if (hr == DXGI_ERROR_NOT_FOUND)
|
|
{
|
|
skip("Adapter has not outputs, skipping output tests.\n");
|
|
}
|
|
else
|
|
{
|
|
ok(SUCCEEDED(hr), "EnumOutputs failed, hr %#x.\n", hr);
|
|
|
|
hr = IDXGIOutput_GetParent(output, &IID_IDXGIAdapter, (void **)&parent);
|
|
IDXGIOutput_Release(output);
|
|
ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr);
|
|
ok(parent == (IUnknown *)adapter, "Got parent %p, expected %p.\n", parent, adapter);
|
|
IUnknown_Release(parent);
|
|
}
|
|
|
|
hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory);
|
|
ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr);
|
|
|
|
hr = IDXGIFactory_GetParent(factory, &IID_IUnknown, (void **)&parent);
|
|
ok(hr == E_NOINTERFACE, "GetParent returned %#x, expected %#x.\n", hr, E_NOINTERFACE);
|
|
ok(parent == NULL, "Got parent %p, expected %p.\n", parent, NULL);
|
|
IDXGIFactory_Release(factory);
|
|
|
|
hr = IDXGIDevice_GetParent(device, &IID_IDXGIAdapter, (void **)&parent);
|
|
ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr);
|
|
ok(parent == (IUnknown *)adapter, "Got parent %p, expected %p.\n", parent, adapter);
|
|
IUnknown_Release(parent);
|
|
|
|
IDXGIAdapter_Release(adapter);
|
|
}
|
|
|
|
static void test_output(IDXGIDevice *device)
|
|
{
|
|
IDXGIAdapter *adapter;
|
|
HRESULT hr;
|
|
IDXGIOutput *output;
|
|
UINT mode_count, mode_count_comp, i;
|
|
DXGI_MODE_DESC *modes;
|
|
|
|
hr = IDXGIDevice_GetAdapter(device, &adapter);
|
|
ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr);
|
|
|
|
hr = IDXGIAdapter_EnumOutputs(adapter, 0, &output);
|
|
if (hr == DXGI_ERROR_NOT_FOUND)
|
|
{
|
|
skip("Adapter has not outputs, skipping output tests.\n");
|
|
IDXGIAdapter_Release(adapter);
|
|
return;
|
|
}
|
|
|
|
ok(SUCCEEDED(hr), "EnumOutputs failed, hr %#x.\n", hr);
|
|
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, 0, NULL, NULL);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, 0, &mode_count, NULL);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
mode_count_comp = mode_count;
|
|
|
|
IDXGIOutput_GetDisplayModeList(output, 0, 0, &mode_count, NULL);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(!mode_count, "Expected 0 got %d\n", mode_count);
|
|
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_ENUM_MODES_SCALING, &mode_count, NULL);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(mode_count >= mode_count_comp, "Flag implies trying to enumerate more modes\n");
|
|
mode_count_comp = mode_count;
|
|
|
|
modes = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(DXGI_MODE_DESC) * mode_count+10);
|
|
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_ENUM_MODES_SCALING, NULL, modes);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(!modes[0].Height, "No output was expected\n");
|
|
|
|
mode_count = 0;
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_ENUM_MODES_SCALING, &mode_count, modes);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(!modes[0].Height, "No output was expected\n");
|
|
|
|
mode_count = mode_count_comp;
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_ENUM_MODES_SCALING, &mode_count, modes);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(mode_count == mode_count_comp, "Expected %d, got %d\n", mode_count_comp, mode_count);
|
|
|
|
for (i = 0; i < mode_count; i++)
|
|
{
|
|
ok(modes[i].Height && modes[i].Width, "Proper mode was expected\n");
|
|
}
|
|
|
|
mode_count += 5;
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_ENUM_MODES_SCALING, &mode_count, modes);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(mode_count == mode_count_comp, "Expected %d, got %d\n", mode_count_comp, mode_count);
|
|
|
|
if (mode_count_comp)
|
|
{
|
|
mode_count = mode_count_comp - 1;
|
|
IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_ENUM_MODES_SCALING, &mode_count, modes);
|
|
ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr);
|
|
ok(mode_count == mode_count_comp -1, "Expected %d, got %d\n", mode_count_comp, mode_count);
|
|
}
|
|
else
|
|
{
|
|
skip("Not enough modes for test, skipping\n");
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, modes);
|
|
IDXGIOutput_Release(output);
|
|
IDXGIAdapter_Release(adapter);
|
|
}
|
|
|
|
START_TEST(device)
|
|
{
|
|
HMODULE d3d10core = LoadLibraryA("d3d10core.dll");
|
|
IDXGIDevice *device;
|
|
ULONG refcount;
|
|
|
|
if (!d3d10core)
|
|
{
|
|
win_skip("d3d10core.dll not available, skipping tests\n");
|
|
return;
|
|
}
|
|
|
|
device = create_device(d3d10core);
|
|
if (!device)
|
|
{
|
|
skip("Failed to create device, skipping tests\n");
|
|
FreeLibrary(d3d10core);
|
|
return;
|
|
}
|
|
|
|
test_adapter_desc(device);
|
|
test_device_interfaces(device);
|
|
test_create_surface(device);
|
|
test_parents(device);
|
|
test_output(device);
|
|
|
|
refcount = IDXGIDevice_Release(device);
|
|
ok(!refcount, "Device has %u references left\n", refcount);
|
|
FreeLibrary(d3d10core);
|
|
}
|