| /* |
| * 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 |
| */ |
| |
| #include <assert.h> |
| #define COBJMACROS |
| #include "initguid.h" |
| #include "d3d11.h" |
| #include "wine/test.h" |
| |
| enum frame_latency |
| { |
| DEFAULT_FRAME_LATENCY = 3, |
| MAX_FRAME_LATENCY = 16, |
| }; |
| |
| static DEVMODEW registry_mode; |
| |
| static HRESULT (WINAPI *pCreateDXGIFactory1)(REFIID iid, void **factory); |
| |
| static ULONG get_refcount(IUnknown *iface) |
| { |
| IUnknown_AddRef(iface); |
| return IUnknown_Release(iface); |
| } |
| |
| #define MODE_DESC_IGNORE_RESOLUTION 0x00000001u |
| #define MODE_DESC_IGNORE_REFRESH_RATE 0x00000002u |
| #define MODE_DESC_IGNORE_FORMAT 0x00000004u |
| #define MODE_DESC_IGNORE_SCANLINE_ORDERING 0x00000008u |
| #define MODE_DESC_IGNORE_SCALING 0x00000010u |
| |
| #define MODE_DESC_CHECK_RESOLUTION (~MODE_DESC_IGNORE_RESOLUTION) |
| #define MODE_DESC_CHECK_FORMAT (~MODE_DESC_IGNORE_FORMAT) |
| |
| #define check_mode_desc(a, b, c) check_mode_desc_(__LINE__, a, b, c) |
| static void check_mode_desc_(unsigned int line, const DXGI_MODE_DESC *desc, |
| const DXGI_MODE_DESC *expected_desc, unsigned int ignore_flags) |
| { |
| if (!(ignore_flags & MODE_DESC_IGNORE_RESOLUTION)) |
| { |
| ok_(__FILE__, line)(desc->Width == expected_desc->Width |
| && desc->Height == expected_desc->Height, |
| "Got resolution %ux%u, expected %ux%u.\n", |
| desc->Width, desc->Height, expected_desc->Width, expected_desc->Height); |
| } |
| if (!(ignore_flags & MODE_DESC_IGNORE_REFRESH_RATE)) |
| { |
| ok_(__FILE__, line)(desc->RefreshRate.Numerator == expected_desc->RefreshRate.Numerator |
| && desc->RefreshRate.Denominator == expected_desc->RefreshRate.Denominator, |
| "Got refresh rate %u / %u, expected %u / %u.\n", |
| desc->RefreshRate.Numerator, desc->RefreshRate.Denominator, |
| expected_desc->RefreshRate.Denominator, expected_desc->RefreshRate.Denominator); |
| } |
| if (!(ignore_flags & MODE_DESC_IGNORE_FORMAT)) |
| { |
| ok_(__FILE__, line)(desc->Format == expected_desc->Format, |
| "Got format %#x, expected %#x.\n", desc->Format, expected_desc->Format); |
| } |
| if (!(ignore_flags & MODE_DESC_IGNORE_SCANLINE_ORDERING)) |
| { |
| ok_(__FILE__, line)(desc->ScanlineOrdering == expected_desc->ScanlineOrdering, |
| "Got scanline ordering %#x, expected %#x.\n", |
| desc->ScanlineOrdering, expected_desc->ScanlineOrdering); |
| } |
| if (!(ignore_flags & MODE_DESC_IGNORE_SCALING)) |
| { |
| ok_(__FILE__, line)(desc->Scaling == expected_desc->Scaling, |
| "Got scaling %#x, expected %#x.\n", |
| desc->Scaling, expected_desc->Scaling); |
| } |
| } |
| |
| #define check_output_desc(a, b) check_output_desc_(__LINE__, a, b) |
| static void check_output_desc_(unsigned int line, const DXGI_OUTPUT_DESC *desc, |
| const struct DXGI_OUTPUT_DESC *expected_desc) |
| { |
| ok_(__FILE__, line)(!lstrcmpW(desc->DeviceName, expected_desc->DeviceName), |
| "Got unexpected device name %s, expected %s.\n", |
| wine_dbgstr_w(desc->DeviceName), wine_dbgstr_w(expected_desc->DeviceName)); |
| ok_(__FILE__, line)(EqualRect(&desc->DesktopCoordinates, &expected_desc->DesktopCoordinates), |
| "Got unexpected desktop coordinates %s, expected %s.\n", |
| wine_dbgstr_rect(&desc->DesktopCoordinates), |
| wine_dbgstr_rect(&expected_desc->DesktopCoordinates)); |
| } |
| |
| #define check_output_equal(a, b) check_output_equal_(__LINE__, a, b) |
| static void check_output_equal_(unsigned int line, IDXGIOutput *output1, IDXGIOutput *output2) |
| { |
| DXGI_OUTPUT_DESC desc1, desc2; |
| HRESULT hr; |
| |
| hr = IDXGIOutput_GetDesc(output1, &desc1); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| hr = IDXGIOutput_GetDesc(output2, &desc2); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| check_output_desc_(line, &desc1, &desc2); |
| } |
| |
| static BOOL output_belongs_to_adapter(IDXGIOutput *output, IDXGIAdapter *adapter) |
| { |
| DXGI_OUTPUT_DESC output_desc, desc; |
| unsigned int output_idx; |
| IDXGIOutput *o; |
| HRESULT hr; |
| |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "Failed to get output desc, hr %#x.\n", hr); |
| |
| for (output_idx = 0; IDXGIAdapter_EnumOutputs(adapter, output_idx, &o) != DXGI_ERROR_NOT_FOUND; ++output_idx) |
| { |
| hr = IDXGIOutput_GetDesc(o, &desc); |
| ok(SUCCEEDED(hr), "Failed to get output desc, hr %#x.\n", hr); |
| IDXGIOutput_Release(o); |
| |
| if (!lstrcmpW(desc.DeviceName, output_desc.DeviceName) |
| && EqualRect(&desc.DesktopCoordinates, &output_desc.DesktopCoordinates)) |
| return TRUE; |
| } |
| |
| return FALSE; |
| } |
| |
| struct fullscreen_state |
| { |
| RECT window_rect; |
| RECT client_rect; |
| HMONITOR monitor; |
| RECT monitor_rect; |
| }; |
| |
| struct swapchain_fullscreen_state |
| { |
| struct fullscreen_state fullscreen_state; |
| BOOL fullscreen; |
| IDXGIOutput *target; |
| }; |
| |
| #define capture_fullscreen_state(a, b) capture_fullscreen_state_(__LINE__, a, b) |
| static void capture_fullscreen_state_(unsigned int line, struct fullscreen_state *state, HWND window) |
| { |
| MONITORINFOEXW monitor_info; |
| BOOL ret; |
| |
| ret = GetWindowRect(window, &state->window_rect); |
| ok_(__FILE__, line)(ret, "GetWindowRect failed.\n"); |
| ret = GetClientRect(window, &state->client_rect); |
| ok_(__FILE__, line)(ret, "GetClientRect failed.\n"); |
| |
| state->monitor = MonitorFromWindow(window, MONITOR_DEFAULTTONULL); |
| ok_(__FILE__, line)(!!state->monitor, "Failed to get monitor from window.\n"); |
| |
| monitor_info.cbSize = sizeof(monitor_info); |
| ret = GetMonitorInfoW(state->monitor, (MONITORINFO *)&monitor_info); |
| ok_(__FILE__, line)(ret, "Failed to get monitor info.\n"); |
| state->monitor_rect = monitor_info.rcMonitor; |
| } |
| |
| #define check_fullscreen_state(a, b) check_fullscreen_state_(__LINE__, a, b) |
| static void check_fullscreen_state_(unsigned int line, const struct fullscreen_state *state, |
| const struct fullscreen_state *expected_state) |
| { |
| ok_(__FILE__, line)(EqualRect(&state->window_rect, &expected_state->window_rect), |
| "Got window rect %s, expected %s.\n", |
| wine_dbgstr_rect(&state->window_rect), wine_dbgstr_rect(&expected_state->window_rect)); |
| ok_(__FILE__, line)(EqualRect(&state->client_rect, &expected_state->client_rect), |
| "Got client rect %s, expected %s.\n", |
| wine_dbgstr_rect(&state->client_rect), wine_dbgstr_rect(&expected_state->client_rect)); |
| ok_(__FILE__, line)(state->monitor == expected_state->monitor, |
| "Got monitor %p, expected %p.\n", |
| state->monitor, expected_state->monitor); |
| ok_(__FILE__, line)(EqualRect(&state->monitor_rect, &expected_state->monitor_rect), |
| "Got monitor rect %s, expected %s.\n", |
| wine_dbgstr_rect(&state->monitor_rect), wine_dbgstr_rect(&expected_state->monitor_rect)); |
| } |
| |
| #define check_window_fullscreen_state(a, b) check_window_fullscreen_state_(__LINE__, a, b) |
| static void check_window_fullscreen_state_(unsigned int line, HWND window, |
| const struct fullscreen_state *expected_state) |
| { |
| struct fullscreen_state current_state; |
| capture_fullscreen_state_(line, ¤t_state, window); |
| check_fullscreen_state_(line, ¤t_state, expected_state); |
| } |
| |
| #define check_swapchain_fullscreen_state(a, b) check_swapchain_fullscreen_state_(__LINE__, a, b) |
| static void check_swapchain_fullscreen_state_(unsigned int line, IDXGISwapChain *swapchain, |
| const struct swapchain_fullscreen_state *expected_state) |
| { |
| IDXGIOutput *containing_output, *target; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| BOOL fullscreen; |
| HRESULT hr; |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok_(__FILE__, line)(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| check_window_fullscreen_state_(line, swapchain_desc.OutputWindow, &expected_state->fullscreen_state); |
| |
| ok_(__FILE__, line)(swapchain_desc.Windowed == !expected_state->fullscreen, |
| "Got windowed %#x, expected %#x.\n", |
| swapchain_desc.Windowed, !expected_state->fullscreen); |
| |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, &target); |
| ok_(__FILE__, line)(SUCCEEDED(hr), "GetFullscreenState failed, hr %#x.\n", hr); |
| ok_(__FILE__, line)(fullscreen == expected_state->fullscreen, "Got fullscreen %#x, expected %#x.\n", |
| fullscreen, expected_state->fullscreen); |
| |
| if (!swapchain_desc.Windowed && expected_state->fullscreen) |
| { |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| |
| hr = IDXGISwapChain_GetDevice(swapchain, &IID_IDXGIDevice, (void **)&device); |
| ok_(__FILE__, line)(SUCCEEDED(hr), "GetDevice failed, hr %#x.\n", hr); |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok_(__FILE__, line)(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| IDXGIDevice_Release(device); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &containing_output); |
| ok_(__FILE__, line)(SUCCEEDED(hr), "GetContainingOutput failed, hr %#x.\n", hr); |
| |
| check_output_equal_(line, target, expected_state->target); |
| ok_(__FILE__, line)(target == containing_output, "Got target %p, expected %p.\n", |
| target, containing_output); |
| ok_(__FILE__, line)(output_belongs_to_adapter(target, adapter), |
| "Output %p doesn't belong to adapter %p.\n", |
| target, adapter); |
| |
| IDXGIOutput_Release(target); |
| IDXGIOutput_Release(containing_output); |
| IDXGIAdapter_Release(adapter); |
| } |
| else |
| { |
| ok_(__FILE__, line)(!target, "Got unexpected target %p.\n", target); |
| } |
| } |
| |
| #define compute_expected_swapchain_fullscreen_state_after_fullscreen_change(a, b, c, d, e, f) \ |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change_(__LINE__, a, b, c, d, e, f) |
| static void compute_expected_swapchain_fullscreen_state_after_fullscreen_change_(unsigned int line, |
| struct swapchain_fullscreen_state *state, const DXGI_SWAP_CHAIN_DESC *swapchain_desc, |
| const RECT *old_monitor_rect, unsigned int new_width, unsigned int new_height, IDXGIOutput *target) |
| { |
| if (!new_width && !new_height) |
| { |
| RECT client_rect; |
| GetClientRect(swapchain_desc->OutputWindow, &client_rect); |
| new_width = client_rect.right - client_rect.left; |
| new_height = client_rect.bottom - client_rect.top; |
| } |
| |
| if (target) |
| { |
| DXGI_MODE_DESC mode_desc = swapchain_desc->BufferDesc; |
| HRESULT hr; |
| |
| mode_desc.Width = new_width; |
| mode_desc.Height = new_height; |
| hr = IDXGIOutput_FindClosestMatchingMode(target, &mode_desc, &mode_desc, NULL); |
| ok_(__FILE__, line)(SUCCEEDED(hr), "FindClosestMatchingMode failed, hr %#x.\n", hr); |
| new_width = mode_desc.Width; |
| new_height = mode_desc.Height; |
| } |
| |
| state->fullscreen = TRUE; |
| if (swapchain_desc->Flags & DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH) |
| { |
| unsigned int new_x = (old_monitor_rect->left >= 0) |
| ? old_monitor_rect->left : old_monitor_rect->right - new_width; |
| unsigned new_y = (old_monitor_rect->top >= 0) |
| ? old_monitor_rect->top : old_monitor_rect->bottom - new_height; |
| RECT new_monitor_rect = {0, 0, new_width, new_height}; |
| OffsetRect(&new_monitor_rect, new_x, new_y); |
| |
| SetRect(&state->fullscreen_state.client_rect, 0, 0, new_width, new_height); |
| state->fullscreen_state.monitor_rect = new_monitor_rect; |
| state->fullscreen_state.window_rect = new_monitor_rect; |
| |
| if (target) |
| state->target = target; |
| } |
| else |
| { |
| state->fullscreen_state.window_rect = *old_monitor_rect; |
| SetRect(&state->fullscreen_state.client_rect, 0, 0, |
| old_monitor_rect->right - old_monitor_rect->left, |
| old_monitor_rect->bottom - old_monitor_rect->top); |
| } |
| } |
| |
| static IDXGIDevice *create_device(UINT flags) |
| { |
| IDXGIDevice *dxgi_device; |
| ID3D10Device1 *device; |
| HRESULT hr; |
| |
| if (SUCCEEDED(D3D10CreateDevice1(NULL, D3D10_DRIVER_TYPE_HARDWARE, NULL, |
| flags, D3D10_FEATURE_LEVEL_10_0, D3D10_1_SDK_VERSION, &device))) |
| goto success; |
| if (SUCCEEDED(D3D10CreateDevice1(NULL, D3D10_DRIVER_TYPE_WARP, NULL, |
| flags, D3D10_FEATURE_LEVEL_10_0, D3D10_1_SDK_VERSION, &device))) |
| goto success; |
| if (SUCCEEDED(D3D10CreateDevice1(NULL, D3D10_DRIVER_TYPE_REFERENCE, NULL, |
| flags, D3D10_FEATURE_LEVEL_10_0, D3D10_1_SDK_VERSION, &device))) |
| goto success; |
| |
| return NULL; |
| |
| success: |
| hr = ID3D10Device1_QueryInterface(device, &IID_IDXGIDevice, (void **)&dxgi_device); |
| ok(SUCCEEDED(hr), "Created device does not implement IDXGIDevice\n"); |
| ID3D10Device1_Release(device); |
| |
| return dxgi_device; |
| } |
| |
| static void test_adapter_desc(void) |
| { |
| DXGI_ADAPTER_DESC1 desc1; |
| IDXGIAdapter1 *adapter1; |
| DXGI_ADAPTER_DESC desc; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| |
| 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); |
| |
| hr = IDXGIAdapter_QueryInterface(adapter, &IID_IDXGIAdapter1, (void **)&adapter1); |
| ok(SUCCEEDED(hr) || broken(hr == E_NOINTERFACE), "Got unexpected hr %#x.\n", hr); |
| if (hr == E_NOINTERFACE) |
| goto done; |
| |
| hr = IDXGIAdapter1_GetDesc1(adapter1, &desc1); |
| ok(SUCCEEDED(hr), "GetDesc1 failed, hr %#x.\n", hr); |
| |
| ok(!lstrcmpW(desc.Description, desc1.Description), |
| "Got unexpected description %s.\n", wine_dbgstr_w(desc1.Description)); |
| ok(desc1.VendorId == desc.VendorId, "Got unexpected vendor ID %04x.\n", desc1.VendorId); |
| ok(desc1.DeviceId == desc.DeviceId, "Got unexpected device ID %04x.\n", desc1.DeviceId); |
| ok(desc1.SubSysId == desc.SubSysId, "Got unexpected sub system ID %04x.\n", desc1.SubSysId); |
| ok(desc1.Revision == desc.Revision, "Got unexpected revision %02x.\n", desc1.Revision); |
| ok(desc1.DedicatedVideoMemory == desc.DedicatedVideoMemory, |
| "Got unexpected dedicated video memory %lu.\n", desc1.DedicatedVideoMemory); |
| ok(desc1.DedicatedSystemMemory == desc.DedicatedSystemMemory, |
| "Got unexpected dedicated system memory %lu.\n", desc1.DedicatedSystemMemory); |
| ok(desc1.SharedSystemMemory == desc.SharedSystemMemory, |
| "Got unexpected shared system memory %lu.\n", desc1.SharedSystemMemory); |
| ok(!memcmp(&desc.AdapterLuid, &desc1.AdapterLuid, sizeof(desc.AdapterLuid)), |
| "Got unexpected adapter LUID %08x:%08x.\n", desc1.AdapterLuid.HighPart, desc1.AdapterLuid.LowPart); |
| trace("Flags: %08x.\n", desc1.Flags); |
| |
| IDXGIAdapter1_Release(adapter1); |
| |
| done: |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| static void test_check_interface_support(void) |
| { |
| LARGE_INTEGER driver_version; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| IUnknown *iface; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_CheckInterfaceSupport(adapter, &IID_ID3D10Device, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIAdapter_CheckInterfaceSupport(adapter, &IID_ID3D10Device, &driver_version); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| |
| trace("UMD version: %u.%u.%u.%u.\n", |
| HIWORD(U(driver_version).HighPart), LOWORD(U(driver_version).HighPart), |
| HIWORD(U(driver_version).LowPart), LOWORD(U(driver_version).LowPart)); |
| |
| hr = IDXGIDevice_QueryInterface(device, &IID_ID3D10Device1, (void **)&iface); |
| if (SUCCEEDED(hr)) |
| { |
| IUnknown_Release(iface); |
| hr = IDXGIAdapter_CheckInterfaceSupport(adapter, &IID_ID3D10Device1, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIAdapter_CheckInterfaceSupport(adapter, &IID_ID3D10Device1, &driver_version); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| } |
| else |
| { |
| win_skip("D3D10.1 is not supported.\n"); |
| } |
| |
| hr = IDXGIAdapter_CheckInterfaceSupport(adapter, &IID_ID3D11Device, NULL); |
| ok(hr == DXGI_ERROR_UNSUPPORTED, "Got unexpected hr %#x.\n", hr); |
| driver_version.HighPart = driver_version.LowPart = 0xdeadbeef; |
| hr = IDXGIAdapter_CheckInterfaceSupport(adapter, &IID_ID3D11Device, &driver_version); |
| ok(hr == DXGI_ERROR_UNSUPPORTED, "Got unexpected hr %#x.\n", hr); |
| ok(driver_version.HighPart == 0xdeadbeef, "Got unexpected driver version %#x.\n", driver_version.HighPart); |
| ok(driver_version.LowPart == 0xdeadbeef, "Got unexpected driver version %#x.\n", driver_version.LowPart); |
| |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| static void test_create_surface(void) |
| { |
| DXGI_SURFACE_DESC desc; |
| IDXGISurface *surface; |
| IDXGIDevice *device; |
| IUnknown *surface1; |
| IUnknown *texture; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| |
| 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"); |
| IUnknown_Release(texture); |
| |
| hr = IDXGISurface_QueryInterface(surface, &IID_ID3D11Texture2D, (void **)&texture); |
| ok(SUCCEEDED(hr) || broken(hr == E_NOINTERFACE) /* Not available on all Windows versions. */, |
| "Surface should implement ID3D11Texture2D.\n"); |
| if (SUCCEEDED(hr)) IUnknown_Release(texture); |
| |
| hr = IDXGISurface_QueryInterface(surface, &IID_IDXGISurface1, (void **)&surface1); |
| ok(SUCCEEDED(hr) || broken(hr == E_NOINTERFACE) /* Not available on all Windows versions. */, |
| "Surface should implement IDXGISurface1.\n"); |
| if (SUCCEEDED(hr)) IUnknown_Release(surface1); |
| |
| IDXGISurface_Release(surface); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| static void test_parents(void) |
| { |
| DXGI_SURFACE_DESC surface_desc; |
| IDXGISurface *surface; |
| IDXGIFactory *factory; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| IDXGIOutput *output; |
| IUnknown *parent; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| |
| 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); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| static void test_output(void) |
| { |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| HRESULT hr; |
| IDXGIOutput *output; |
| ULONG refcount; |
| UINT mode_count, mode_count_comp, i; |
| DXGI_MODE_DESC *modes; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| |
| 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 doesn't have any outputs, skipping tests.\n"); |
| IDXGIAdapter_Release(adapter); |
| IDXGIDevice_Release(device); |
| return; |
| } |
| ok(SUCCEEDED(hr), "EnumOutputs failed, hr %#x.\n", hr); |
| |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, 0, NULL, NULL); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, 0, &mode_count, NULL); |
| ok(SUCCEEDED(hr) |
| || broken(hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE), /* Remote Desktop Services / Win 7 testbot */ |
| "Failed to list modes, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE) |
| { |
| win_skip("GetDisplayModeList() not supported.\n"); |
| IDXGIOutput_Release(output); |
| IDXGIAdapter_Release(adapter); |
| IDXGIDevice_Release(device); |
| return; |
| } |
| mode_count_comp = mode_count; |
| |
| hr = IDXGIOutput_GetDisplayModeList(output, 0, 0, &mode_count, NULL); |
| ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr); |
| ok(!mode_count, "Got unexpected mode_count %u.\n", mode_count); |
| |
| hr = 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, "Got unexpected mode_count %u, expected >= %u.\n", mode_count, mode_count_comp); |
| mode_count_comp = mode_count; |
| |
| modes = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*modes) * (mode_count + 10)); |
| |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, |
| DXGI_ENUM_MODES_SCALING, NULL, modes); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| ok(!modes[0].Height, "No output was expected.\n"); |
| |
| mode_count = 0; |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, |
| DXGI_ENUM_MODES_SCALING, &mode_count, modes); |
| ok(hr == DXGI_ERROR_MORE_DATA, "Got unexpected hr %#x.\n", hr); |
| ok(!modes[0].Height, "No output was expected.\n"); |
| |
| mode_count = mode_count_comp; |
| hr = 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, "Got unexpected mode_count %u, expected %u.\n", mode_count, mode_count_comp); |
| |
| for (i = 0; i < mode_count; i++) |
| { |
| ok(modes[i].Height && modes[i].Width, "Proper mode was expected\n"); |
| } |
| |
| mode_count += 5; |
| hr = 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, "Got unexpected mode_count %u, expected %u.\n", mode_count, mode_count_comp); |
| |
| if (mode_count_comp) |
| { |
| mode_count = mode_count_comp - 1; |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, |
| DXGI_ENUM_MODES_SCALING, &mode_count, modes); |
| ok(hr == DXGI_ERROR_MORE_DATA, "Got unexpected hr %#x.\n", hr); |
| ok(mode_count == mode_count_comp - 1, "Got unexpected mode_count %u, expected %u.\n", |
| mode_count, mode_count_comp - 1); |
| } |
| else |
| { |
| skip("Not enough modes for test, skipping.\n"); |
| } |
| |
| HeapFree(GetProcessHeap(), 0, modes); |
| IDXGIOutput_Release(output); |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| static void test_find_closest_matching_mode(void) |
| { |
| DXGI_MODE_DESC *modes, mode, matching_mode; |
| unsigned int i, mode_count; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| IDXGIOutput *output; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| 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) |
| { |
| win_skip("Adapter doesn't have any outputs.\n"); |
| IDXGIAdapter_Release(adapter); |
| IDXGIDevice_Release(device); |
| return; |
| } |
| ok(SUCCEEDED(hr), "EnumOutputs failed, hr %#x.\n", hr); |
| |
| memset(&mode, 0, sizeof(mode)); |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == DXGI_ERROR_INVALID_CALL || broken(hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE), /* Win 7 testbot */ |
| "Got unexpected hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE) |
| { |
| win_skip("FindClosestMatchingMode() not supported.\n"); |
| goto done; |
| } |
| |
| memset(&mode, 0, sizeof(mode)); |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, (IUnknown *)device); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, 0, &mode_count, NULL); |
| ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr); |
| |
| modes = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*modes) * mode_count); |
| |
| hr = IDXGIOutput_GetDisplayModeList(output, DXGI_FORMAT_R8G8B8A8_UNORM, 0, &mode_count, modes); |
| ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr); |
| |
| for (i = 0; i < mode_count; ++i) |
| { |
| mode = modes[i]; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_IGNORE_SCALING); |
| |
| mode.Format = DXGI_FORMAT_UNKNOWN; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| |
| mode = modes[i]; |
| mode.Width = 0; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| |
| mode = modes[i]; |
| mode.Height = 0; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| |
| mode = modes[i]; |
| mode.Width = mode.Height = 0; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_IGNORE_SCALING | MODE_DESC_IGNORE_RESOLUTION); |
| ok(matching_mode.Width > 0 && matching_mode.Height > 0, "Got unexpected resolution %ux%u.\n", |
| matching_mode.Width, matching_mode.Height); |
| |
| mode = modes[i]; |
| mode.RefreshRate.Numerator = mode.RefreshRate.Denominator = 0; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_IGNORE_SCALING | MODE_DESC_IGNORE_REFRESH_RATE); |
| ok(matching_mode.RefreshRate.Numerator > 0 && matching_mode.RefreshRate.Denominator > 0, |
| "Got unexpected refresh rate %u / %u.\n", |
| matching_mode.RefreshRate.Numerator, matching_mode.RefreshRate.Denominator); |
| |
| mode = modes[i]; |
| mode.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_IGNORE_SCALING | MODE_DESC_IGNORE_SCANLINE_ORDERING); |
| ok(matching_mode.ScanlineOrdering, "Got unexpected scanline ordering %#x.\n", |
| matching_mode.ScanlineOrdering); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[i].Width; |
| mode.Height = modes[i].Height; |
| mode.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[i].Width - 1; |
| mode.Height = modes[i].Height - 1; |
| mode.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[i].Width + 1; |
| mode.Height = modes[i].Height + 1; |
| mode.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[i], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| } |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = mode.Height = 10; |
| mode.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| /* Find mode for the lowest resolution. */ |
| mode = modes[0]; |
| for (i = 0; i < mode_count; ++i) |
| { |
| if (mode.Width >= modes[i].Width && mode.Height >= modes[i].Height) |
| mode = modes[i]; |
| } |
| check_mode_desc(&matching_mode, &mode, MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[0].Width; |
| mode.Height = modes[0].Height; |
| mode.Format = modes[0].Format; |
| mode.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UPPER_FIELD_FIRST; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[0], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[0].Width; |
| mode.Height = modes[0].Height; |
| mode.Format = modes[0].Format; |
| mode.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_LOWER_FIELD_FIRST; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[0], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[0].Width; |
| mode.Height = modes[0].Height; |
| mode.Format = modes[0].Format; |
| mode.Scaling = DXGI_MODE_SCALING_CENTERED; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[0], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = modes[0].Width; |
| mode.Height = modes[0].Height; |
| mode.Format = modes[0].Format; |
| mode.Scaling = DXGI_MODE_SCALING_STRETCHED; |
| hr = IDXGIOutput_FindClosestMatchingMode(output, &mode, &matching_mode, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_mode_desc(&matching_mode, &modes[0], MODE_DESC_CHECK_RESOLUTION & MODE_DESC_CHECK_FORMAT); |
| |
| HeapFree(GetProcessHeap(), 0, modes); |
| |
| done: |
| IDXGIOutput_Release(output); |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| struct refresh_rates |
| { |
| UINT numerator; |
| UINT denominator; |
| BOOL numerator_should_pass; |
| BOOL denominator_should_pass; |
| }; |
| |
| static void test_create_swapchain(void) |
| { |
| struct swapchain_fullscreen_state initial_state, expected_state; |
| unsigned int i, expected_width, expected_height; |
| DXGI_SWAP_CHAIN_DESC creation_desc, result_desc; |
| IDXGIDevice *device, *bgra_device; |
| ULONG refcount, expected_refcount; |
| IUnknown *obj, *obj2, *parent; |
| RECT *expected_client_rect; |
| IDXGISwapChain *swapchain; |
| IDXGISurface1 *surface; |
| IDXGIAdapter *adapter; |
| IDXGIFactory *factory; |
| IDXGIOutput *target; |
| BOOL fullscreen; |
| HRESULT hr; |
| |
| const struct refresh_rates refresh_list[] = |
| { |
| {60, 60, FALSE, FALSE}, |
| {60, 0, TRUE, FALSE}, |
| {60, 1, TRUE, TRUE}, |
| { 0, 60, TRUE, FALSE}, |
| { 0, 0, TRUE, FALSE}, |
| }; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| |
| creation_desc.OutputWindow = 0; |
| creation_desc.BufferDesc.Width = 800; |
| creation_desc.BufferDesc.Height = 600; |
| creation_desc.BufferDesc.RefreshRate.Numerator = 60; |
| creation_desc.BufferDesc.RefreshRate.Denominator = 60; |
| creation_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| creation_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| creation_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| creation_desc.SampleDesc.Count = 1; |
| creation_desc.SampleDesc.Quality = 0; |
| creation_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| creation_desc.BufferCount = 1; |
| creation_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, 0, 0, 0, 0, 0, 0, 0, 0); |
| creation_desc.Windowed = TRUE; |
| creation_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| creation_desc.Flags = 0; |
| |
| memset(&initial_state, 0, sizeof(initial_state)); |
| capture_fullscreen_state(&initial_state.fullscreen_state, creation_desc.OutputWindow); |
| |
| hr = IDXGIDevice_QueryInterface(device, &IID_IUnknown, (void **)&obj); |
| ok(SUCCEEDED(hr), "IDXGIDevice does not implement IUnknown.\n"); |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| expected_refcount = get_refcount((IUnknown *)adapter); |
| refcount = get_refcount((IUnknown *)factory); |
| ok(refcount == 2, "Got unexpected refcount %u.\n", refcount); |
| refcount = get_refcount((IUnknown *)device); |
| ok(refcount == 2, "Got unexpected refcount %u.\n", refcount); |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| refcount = get_refcount((IUnknown *)adapter); |
| ok(refcount == expected_refcount, "Got refcount %u, expected %u.\n", refcount, expected_refcount); |
| refcount = get_refcount((IUnknown *)factory); |
| todo_wine ok(refcount == 4, "Got unexpected refcount %u.\n", refcount); |
| refcount = get_refcount((IUnknown *)device); |
| ok(refcount == 3, "Got unexpected refcount %u.\n", refcount); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, NULL); |
| ok(hr == E_INVALIDARG, "GetDesc unexpectedly returned %#x.\n", hr); |
| |
| hr = IDXGISwapChain_GetParent(swapchain, &IID_IUnknown, (void **)&parent); |
| ok(SUCCEEDED(hr), "GetParent failed %#x.\n", hr); |
| ok(parent == (IUnknown *)factory, "Got unexpected parent interface pointer %p.\n", parent); |
| refcount = IUnknown_Release(parent); |
| todo_wine ok(refcount == 4, "Got unexpected refcount %u.\n", refcount); |
| |
| hr = IDXGISwapChain_GetParent(swapchain, &IID_IDXGIFactory, (void **)&parent); |
| ok(SUCCEEDED(hr), "GetParent failed %#x.\n", hr); |
| ok(parent == (IUnknown *)factory, "Got unexpected parent interface pointer %p.\n", parent); |
| refcount = IUnknown_Release(parent); |
| todo_wine ok(refcount == 4, "Got unexpected refcount %u.\n", refcount); |
| |
| IDXGISwapChain_Release(swapchain); |
| |
| refcount = get_refcount((IUnknown *)factory); |
| ok(refcount == 2, "Got unexpected refcount %u.\n", refcount); |
| |
| for (i = 0; i < sizeof(refresh_list) / sizeof(*refresh_list); ++i) |
| { |
| creation_desc.BufferDesc.RefreshRate.Numerator = refresh_list[i].numerator; |
| creation_desc.BufferDesc.RefreshRate.Denominator = refresh_list[i].denominator; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "Test %u: CreateSwapChain failed, hr %#x.\n", i, hr); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "Test %u: GetDesc failed, hr %#x.\n", i, hr); |
| |
| ok(result_desc.Windowed == creation_desc.Windowed, "Test %u: Got unexpected windowed %#x.\n", |
| i, result_desc.Windowed); |
| |
| todo_wine_if (!refresh_list[i].numerator_should_pass) |
| ok(result_desc.BufferDesc.RefreshRate.Numerator == refresh_list[i].numerator, |
| "Numerator %u is %u.\n", i, result_desc.BufferDesc.RefreshRate.Numerator); |
| |
| todo_wine_if (!refresh_list[i].denominator_should_pass) |
| ok(result_desc.BufferDesc.RefreshRate.Denominator == refresh_list[i].denominator, |
| "Denominator %u is %u.\n", i, result_desc.BufferDesc.RefreshRate.Denominator); |
| |
| fullscreen = 0xdeadbeef; |
| target = (void *)0xdeadbeef; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, &target); |
| ok(hr == S_OK, "Test %u: GetFullscreenState failed, hr %#x.\n", i, hr); |
| ok(!fullscreen, "Test %u: Got unexpected fullscreen %#x.\n", i, fullscreen); |
| ok(!target, "Test %u: Got unexpected target %p.\n", i, target); |
| |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, NULL); |
| ok(hr == S_OK, "Test %u: Got unexpected hr %#x.\n", i, hr); |
| fullscreen = 0xdeadbeef; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, NULL); |
| ok(hr == S_OK, "Test %u: Got unexpected hr %#x.\n", i, hr); |
| ok(!fullscreen, "Test %u: Got unexpected fullscreen %#x.\n", i, fullscreen); |
| target = (void *)0xdeadbeef; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, &target); |
| ok(hr == S_OK, "Test %u: Got unexpected hr %#x.\n", i, hr); |
| ok(!target, "Test %u: Got unexpected target %p.\n", i, target); |
| |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| IDXGISwapChain_Release(swapchain); |
| } |
| |
| check_window_fullscreen_state(creation_desc.OutputWindow, &initial_state.fullscreen_state); |
| |
| /* Test GDI-compatible swapchain */ |
| bgra_device = create_device(D3D10_CREATE_DEVICE_BGRA_SUPPORT); |
| ok(!!bgra_device, "Failed to create BGRA capable device.\n"); |
| |
| hr = IDXGIDevice_QueryInterface(bgra_device, &IID_IUnknown, (void **)&obj2); |
| ok(SUCCEEDED(hr), "IDXGIDevice does not implement IUnknown.\n"); |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj2, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_IDXGISurface1, (void **)&surface); |
| if (SUCCEEDED(hr)) |
| { |
| HDC hdc; |
| |
| hr = IDXGISurface1_GetDC(surface, FALSE, &hdc); |
| ok(FAILED(hr), "Expected GetDC() to fail, %#x\n", hr); |
| |
| IDXGISurface1_Release(surface); |
| IDXGISwapChain_Release(swapchain); |
| |
| creation_desc.BufferDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; |
| creation_desc.Flags = DXGI_SWAP_CHAIN_FLAG_GDI_COMPATIBLE; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj2, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| creation_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| creation_desc.Flags = 0; |
| |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_IDXGISurface1, (void **)&surface); |
| ok(SUCCEEDED(hr), "Failed to get front buffer, hr %#x.\n", hr); |
| |
| hr = IDXGISurface1_GetDC(surface, FALSE, &hdc); |
| ok(SUCCEEDED(hr), "Expected GetDC() to succeed, %#x\n", hr); |
| IDXGISurface1_ReleaseDC(surface, NULL); |
| |
| IDXGISurface1_Release(surface); |
| IDXGISwapChain_Release(swapchain); |
| } |
| else |
| { |
| win_skip("IDXGISurface1 is not supported, skipping GetDC() tests.\n"); |
| IDXGISwapChain_Release(swapchain); |
| } |
| IUnknown_Release(obj2); |
| IDXGIDevice_Release(bgra_device); |
| |
| creation_desc.Windowed = FALSE; |
| |
| for (i = 0; i < sizeof(refresh_list) / sizeof(*refresh_list); ++i) |
| { |
| creation_desc.BufferDesc.RefreshRate.Numerator = refresh_list[i].numerator; |
| creation_desc.BufferDesc.RefreshRate.Denominator = refresh_list[i].denominator; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "Test %u: CreateSwapChain failed, hr %#x.\n", i, hr); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "Test %u: GetDesc failed, hr %#x.\n", i, hr); |
| |
| /* When numerator is non-zero and denominator is zero, the windowed mode is used. |
| * Additionally, some versions of WARP seem to always fail to change fullscreen state. */ |
| if (result_desc.Windowed != creation_desc.Windowed) |
| trace("Test %u: Failed to change fullscreen state.\n", i); |
| |
| todo_wine_if (!refresh_list[i].numerator_should_pass) |
| ok(result_desc.BufferDesc.RefreshRate.Numerator == refresh_list[i].numerator, |
| "Numerator %u is %u.\n", i, result_desc.BufferDesc.RefreshRate.Numerator); |
| |
| todo_wine_if (!refresh_list[i].denominator_should_pass) |
| ok(result_desc.BufferDesc.RefreshRate.Denominator == refresh_list[i].denominator, |
| "Denominator %u is %u.\n", i, result_desc.BufferDesc.RefreshRate.Denominator); |
| |
| fullscreen = FALSE; |
| target = NULL; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, &target); |
| ok(hr == S_OK, "Test %u: GetFullscreenState failed, hr %#x.\n", i, hr); |
| ok(fullscreen == !result_desc.Windowed, "Test %u: Got fullscreen %#x, expected %#x.\n", |
| i, fullscreen, result_desc.Windowed); |
| ok(result_desc.Windowed ? !target : !!target, "Test %u: Got unexpected target %p.\n", i, target); |
| if (!result_desc.Windowed) |
| { |
| IDXGIOutput *containing_output; |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &containing_output); |
| ok(SUCCEEDED(hr), "Test %u: GetContainingOutput failed, hr %#x.\n", i, hr); |
| ok(containing_output == target, "Test %u: Got unexpected containing output pointer %p.\n", |
| i, containing_output); |
| IDXGIOutput_Release(containing_output); |
| |
| ok(output_belongs_to_adapter(target, adapter), |
| "Test %u: Output %p doesn't belong to adapter %p.\n", |
| i, target, adapter); |
| IDXGIOutput_Release(target); |
| |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, NULL); |
| ok(hr == S_OK, "Test %u: Got unexpected hr %#x.\n", i, hr); |
| fullscreen = FALSE; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, NULL); |
| ok(hr == S_OK, "Test %u: Got unexpected hr %#x.\n", i, hr); |
| ok(fullscreen, "Test %u: Got unexpected fullscreen %#x.\n", i, fullscreen); |
| target = NULL; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, &target); |
| ok(hr == S_OK, "Test %u: Got unexpected hr %#x.\n", i, hr); |
| ok(!!target, "Test %u: Got unexpected target %p.\n", i, target); |
| IDXGIOutput_Release(target); |
| } |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "Test %u: SetFullscreenState failed, hr %#x.\n", i, hr); |
| |
| fullscreen = 0xdeadbeef; |
| target = (void *)0xdeadbeef; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, &target); |
| ok(hr == S_OK, "Test %u: GetFullscreenState failed, hr %#x.\n", i, hr); |
| ok(!fullscreen, "Test %u: Got unexpected fullscreen %#x.\n", i, fullscreen); |
| ok(!target, "Test %u: Got unexpected target %p.\n", i, target); |
| |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| IDXGISwapChain_Release(swapchain); |
| } |
| |
| check_window_fullscreen_state(creation_desc.OutputWindow, &initial_state.fullscreen_state); |
| |
| /* Test swapchain creation with backbuffer width and height equal to 0. */ |
| expected_state = initial_state; |
| expected_client_rect = &expected_state.fullscreen_state.client_rect; |
| |
| /* Windowed */ |
| expected_width = expected_client_rect->right; |
| expected_height = expected_client_rect->bottom; |
| |
| creation_desc.BufferDesc.Width = 0; |
| creation_desc.BufferDesc.Height = 0; |
| creation_desc.Windowed = TRUE; |
| creation_desc.Flags = 0; |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| ok(result_desc.BufferDesc.Width == expected_width, "Got width %u, expected %u.\n", |
| result_desc.BufferDesc.Width, expected_width); |
| ok(result_desc.BufferDesc.Height == expected_height, "Got height %u, expected %u.\n", |
| result_desc.BufferDesc.Height, expected_height); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| IDXGISwapChain_Release(swapchain); |
| |
| DestroyWindow(creation_desc.OutputWindow); |
| creation_desc.OutputWindow = CreateWindowA("static", "dxgi_test", |
| WS_CAPTION | WS_SYSMENU | WS_THICKFRAME | WS_MINIMIZEBOX | WS_MAXIMIZEBOX, |
| 0, 0, 222, 222, 0, 0, 0, 0); |
| SetRect(&expected_state.fullscreen_state.window_rect, 0, 0, 222, 222); |
| GetClientRect(creation_desc.OutputWindow, expected_client_rect); |
| expected_width = expected_client_rect->right; |
| expected_height = expected_client_rect->bottom; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| ok(result_desc.BufferDesc.Width == expected_width, "Got width %u, expected %u.\n", |
| result_desc.BufferDesc.Width, expected_width); |
| ok(result_desc.BufferDesc.Height == expected_height, "Got height %u, expected %u.\n", |
| result_desc.BufferDesc.Height, expected_height); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| IDXGISwapChain_Release(swapchain); |
| |
| DestroyWindow(creation_desc.OutputWindow); |
| creation_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, 0, 0, 0, 0, 0, 0, 0, 0); |
| check_window_fullscreen_state(creation_desc.OutputWindow, &initial_state.fullscreen_state); |
| |
| /* Fullscreen */ |
| creation_desc.Windowed = FALSE; |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &expected_state.target); |
| ok(SUCCEEDED(hr) || broken(hr == DXGI_ERROR_UNSUPPORTED) /* Win 7 testbot */, |
| "GetContainingOutput failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| IDXGISwapChain_Release(swapchain); |
| if (hr == DXGI_ERROR_UNSUPPORTED) |
| { |
| win_skip("GetContainingOutput() not supported.\n"); |
| goto done; |
| } |
| if (result_desc.Windowed) |
| { |
| win_skip("Fullscreen not supported.\n"); |
| IDXGIOutput_Release(expected_state.target); |
| goto done; |
| } |
| |
| creation_desc.BufferDesc.Width = 0; |
| creation_desc.BufferDesc.Height = 0; |
| creation_desc.Windowed = FALSE; |
| creation_desc.Flags = 0; |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &creation_desc, &initial_state.fullscreen_state.monitor_rect, 0, 0, expected_state.target); |
| expected_width = expected_client_rect->right - expected_client_rect->left; |
| expected_height = expected_client_rect->bottom - expected_client_rect->top; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| todo_wine ok(result_desc.BufferDesc.Width == expected_width, "Got width %u, expected %u.\n", |
| result_desc.BufferDesc.Width, expected_width); |
| todo_wine ok(result_desc.BufferDesc.Height == expected_height, "Got height %u, expected %u.\n", |
| result_desc.BufferDesc.Height, expected_height); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| IDXGISwapChain_Release(swapchain); |
| |
| /* Fullscreen and DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH */ |
| creation_desc.BufferDesc.Width = 0; |
| creation_desc.BufferDesc.Height = 0; |
| creation_desc.Windowed = FALSE; |
| creation_desc.Flags = DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH; |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &creation_desc, &initial_state.fullscreen_state.monitor_rect, 0, 0, expected_state.target); |
| expected_width = expected_client_rect->right - expected_client_rect->left; |
| expected_height = expected_client_rect->bottom - expected_client_rect->top; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &creation_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| todo_wine ok(result_desc.BufferDesc.Width == expected_width, "Got width %u, expected %u.\n", |
| result_desc.BufferDesc.Width, expected_width); |
| todo_wine ok(result_desc.BufferDesc.Height == expected_height, "Got height %u, expected %u.\n", |
| result_desc.BufferDesc.Height, expected_height); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| IDXGISwapChain_Release(swapchain); |
| |
| IDXGIOutput_Release(expected_state.target); |
| |
| done: |
| IUnknown_Release(obj); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIAdapter_Release(adapter); |
| ok(!refcount, "Adapter has %u references left.\n", refcount); |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "Factory has %u references left.\n", refcount); |
| check_window_fullscreen_state(creation_desc.OutputWindow, &initial_state.fullscreen_state); |
| DestroyWindow(creation_desc.OutputWindow); |
| } |
| |
| static void test_get_containing_output(void) |
| { |
| unsigned int output_count, output_idx; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| IDXGIOutput *output, *output2; |
| DXGI_OUTPUT_DESC output_desc; |
| MONITORINFOEXW monitor_info; |
| IDXGISwapChain *swapchain; |
| IDXGIFactory *factory; |
| IDXGIAdapter *adapter; |
| POINT points[4 * 16]; |
| IDXGIDevice *device; |
| unsigned int i, j; |
| HMONITOR monitor; |
| ULONG refcount; |
| HRESULT hr; |
| BOOL ret; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| swapchain_desc.BufferDesc.Width = 100; |
| swapchain_desc.BufferDesc.Height = 100; |
| swapchain_desc.BufferDesc.RefreshRate.Numerator = 60; |
| swapchain_desc.BufferDesc.RefreshRate.Denominator = 60; |
| swapchain_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| swapchain_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| swapchain_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| swapchain_desc.SampleDesc.Count = 1; |
| swapchain_desc.SampleDesc.Quality = 0; |
| swapchain_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| swapchain_desc.BufferCount = 1; |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", |
| WS_OVERLAPPEDWINDOW | WS_VISIBLE, 0, 0, 100, 100, 0, 0, 0, 0); |
| swapchain_desc.Windowed = TRUE; |
| swapchain_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| swapchain_desc.Flags = 0; |
| |
| output_count = 0; |
| while (IDXGIAdapter_EnumOutputs(adapter, output_count, &output) != DXGI_ERROR_NOT_FOUND) |
| { |
| ok(SUCCEEDED(hr), "Failed to enumerate output %u, hr %#x.\n", output_count, hr); |
| IDXGIOutput_Release(output); |
| ++output_count; |
| } |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| monitor = MonitorFromWindow(swapchain_desc.OutputWindow, 0); |
| ok(!!monitor, "MonitorFromWindow failed.\n"); |
| |
| monitor_info.cbSize = sizeof(monitor_info); |
| ret = GetMonitorInfoW(monitor, (MONITORINFO *)&monitor_info); |
| ok(ret, "Failed to get monitor info.\n"); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &output); |
| ok(SUCCEEDED(hr) || broken(hr == DXGI_ERROR_UNSUPPORTED) /* Win 7 testbot */, |
| "GetContainingOutput failed, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_UNSUPPORTED) |
| { |
| win_skip("GetContainingOutput() not supported.\n"); |
| IDXGISwapChain_Release(swapchain); |
| goto done; |
| } |
| |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &output2); |
| ok(SUCCEEDED(hr), "GetContainingOutput failed, hr %#x.\n", hr); |
| ok(output != output2, "Got unexpected output pointers %p, %p.\n", output, output2); |
| check_output_equal(output, output2); |
| |
| refcount = IDXGIOutput_Release(output); |
| ok(!refcount, "IDXGIOutput has %u references left.\n", refcount); |
| refcount = IDXGIOutput_Release(output2); |
| ok(!refcount, "IDXGIOutput has %u references left.\n", refcount); |
| |
| ok(!lstrcmpW(output_desc.DeviceName, monitor_info.szDevice), |
| "Got unexpected device name %s, expected %s.\n", |
| wine_dbgstr_w(output_desc.DeviceName), wine_dbgstr_w(monitor_info.szDevice)); |
| ok(EqualRect(&output_desc.DesktopCoordinates, &monitor_info.rcMonitor), |
| "Got unexpected desktop coordinates %s, expected %s.\n", |
| wine_dbgstr_rect(&output_desc.DesktopCoordinates), |
| wine_dbgstr_rect(&monitor_info.rcMonitor)); |
| |
| output_idx = 0; |
| while ((hr = IDXGIAdapter_EnumOutputs(adapter, output_idx, &output)) != DXGI_ERROR_NOT_FOUND) |
| { |
| ok(SUCCEEDED(hr), "Failed to enumerate output %u, hr %#x.\n", output_idx, hr); |
| |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| /* Move the OutputWindow to the current output. */ |
| ret = SetWindowPos(swapchain_desc.OutputWindow, 0, |
| output_desc.DesktopCoordinates.left, output_desc.DesktopCoordinates.top, |
| 0, 0, SWP_NOSIZE | SWP_NOZORDER); |
| ok(ret, "SetWindowPos failed.\n"); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &output2); |
| ok(SUCCEEDED(hr), "GetContainingOutput failed, hr %#x.\n", hr); |
| |
| check_output_equal(output, output2); |
| |
| refcount = IDXGIOutput_Release(output2); |
| ok(!refcount, "IDXGIOutput has %u references left.\n", refcount); |
| refcount = IDXGIOutput_Release(output); |
| ok(!refcount, "IDXGIOutput has %u references left.\n", refcount); |
| ++output_idx; |
| |
| /* Move the OutputWindow around the corners of the current output desktop coordinates. */ |
| for (i = 0; i < 4; ++i) |
| { |
| static const POINT offsets[] = |
| { |
| { 0, 0}, |
| {-49, 0}, {-50, 0}, {-51, 0}, |
| { 0, -49}, { 0, -50}, { 0, -51}, |
| {-49, -49}, {-50, -49}, {-51, -49}, |
| {-49, -50}, {-50, -50}, {-51, -50}, |
| {-49, -51}, {-50, -51}, {-51, -51}, |
| }; |
| unsigned int x, y; |
| |
| switch (i) |
| { |
| case 0: |
| x = output_desc.DesktopCoordinates.left; |
| y = output_desc.DesktopCoordinates.top; |
| break; |
| case 1: |
| x = output_desc.DesktopCoordinates.right; |
| y = output_desc.DesktopCoordinates.top; |
| break; |
| case 2: |
| x = output_desc.DesktopCoordinates.right; |
| y = output_desc.DesktopCoordinates.bottom; |
| break; |
| case 3: |
| x = output_desc.DesktopCoordinates.left; |
| y = output_desc.DesktopCoordinates.bottom; |
| break; |
| } |
| |
| for (j = 0; j < sizeof(offsets) / sizeof(*offsets); ++j) |
| { |
| unsigned int idx = (sizeof(offsets) / sizeof(*offsets)) * i + j; |
| assert(idx < sizeof(points) / sizeof(*points)); |
| points[idx].x = x + offsets[j].x; |
| points[idx].y = y + offsets[j].y; |
| } |
| } |
| |
| for (i = 0; i < sizeof(points) / sizeof(*points); ++i) |
| { |
| ret = SetWindowPos(swapchain_desc.OutputWindow, 0, points[i].x, points[i].y, |
| 0, 0, SWP_NOSIZE | SWP_NOZORDER); |
| ok(ret, "SetWindowPos failed.\n"); |
| |
| monitor = MonitorFromWindow(swapchain_desc.OutputWindow, MONITOR_DEFAULTTONEAREST); |
| ok(!!monitor, "MonitorFromWindow failed.\n"); |
| |
| monitor_info.cbSize = sizeof(monitor_info); |
| ret = GetMonitorInfoW(monitor, (MONITORINFO *)&monitor_info); |
| ok(ret, "Failed to get monitor info.\n"); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &output); |
| ok(SUCCEEDED(hr), "GetContainingOutput failed, hr %#x.\n", hr); |
| ok(!!output, "Got unexpected containing output %p.\n", output); |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| refcount = IDXGIOutput_Release(output); |
| ok(!refcount, "IDXGIOutput has %u references left.\n", refcount); |
| |
| ok(!lstrcmpW(output_desc.DeviceName, monitor_info.szDevice), |
| "Got unexpected device name %s, expected %s.\n", |
| wine_dbgstr_w(output_desc.DeviceName), wine_dbgstr_w(monitor_info.szDevice)); |
| ok(EqualRect(&output_desc.DesktopCoordinates, &monitor_info.rcMonitor), |
| "Got unexpected desktop coordinates %s, expected %s.\n", |
| wine_dbgstr_rect(&output_desc.DesktopCoordinates), |
| wine_dbgstr_rect(&monitor_info.rcMonitor)); |
| } |
| } |
| |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| |
| done: |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIAdapter_Release(adapter); |
| ok(!refcount, "Adapter has %u references left.\n", refcount); |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "Factory has %u references left.\n", refcount); |
| DestroyWindow(swapchain_desc.OutputWindow); |
| } |
| |
| static void test_swapchain_fullscreen_state(IDXGISwapChain *swapchain, |
| IDXGIAdapter *adapter, const struct swapchain_fullscreen_state *initial_state) |
| { |
| MONITORINFOEXW monitor_info, *output_monitor_info; |
| struct swapchain_fullscreen_state expected_state; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| DXGI_OUTPUT_DESC output_desc; |
| unsigned int i, output_count; |
| IDXGIOutput *output; |
| HRESULT hr; |
| BOOL ret; |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| check_swapchain_fullscreen_state(swapchain, initial_state); |
| |
| expected_state = *initial_state; |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &swapchain_desc, &initial_state->fullscreen_state.monitor_rect, 800, 600, NULL); |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &expected_state.target); |
| ok(SUCCEEDED(hr), "GetContainingOutput failed, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, initial_state); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, initial_state); |
| |
| IDXGIOutput_Release(expected_state.target); |
| expected_state.target = NULL; |
| |
| output_count = 0; |
| while (IDXGIAdapter_EnumOutputs(adapter, output_count, &output) != DXGI_ERROR_NOT_FOUND) |
| { |
| IDXGIOutput_Release(output); |
| ++output_count; |
| } |
| |
| output_monitor_info = HeapAlloc(GetProcessHeap(), 0, output_count * sizeof(*output_monitor_info)); |
| ok(!!output_monitor_info, "Failed to allocate memory.\n"); |
| for (i = 0; i < output_count; ++i) |
| { |
| hr = IDXGIAdapter_EnumOutputs(adapter, i, &output); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| output_monitor_info[i].cbSize = sizeof(*output_monitor_info); |
| ret = GetMonitorInfoW(output_desc.Monitor, (MONITORINFO *)&output_monitor_info[i]); |
| ok(ret, "Failed to get monitor info.\n"); |
| |
| IDXGIOutput_Release(output); |
| } |
| |
| for (i = 0; i < output_count; ++i) |
| { |
| RECT orig_monitor_rect = output_monitor_info[i].rcMonitor; |
| IDXGIOutput *target; |
| BOOL fullscreen; |
| |
| hr = IDXGIAdapter_EnumOutputs(adapter, i, &output); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| expected_state = *initial_state; |
| expected_state.target = output; |
| expected_state.fullscreen_state.monitor = output_desc.Monitor; |
| expected_state.fullscreen_state.monitor_rect = orig_monitor_rect; |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &swapchain_desc, &orig_monitor_rect, 800, 600, NULL); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, output); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| target = NULL; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, &target); |
| ok(SUCCEEDED(hr), "GetFullscreenState failed, hr %#x.\n", hr); |
| ok(target == output, "Got target pointer %p, expected %p.\n", target, output); |
| IDXGIOutput_Release(target); |
| fullscreen = FALSE; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, NULL); |
| ok(SUCCEEDED(hr), "GetFullscreenState failed, hr %#x.\n", hr); |
| ok(fullscreen, "Got unexpected fullscreen %#x.\n", hr); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, output); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, output); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, initial_state); |
| |
| fullscreen = TRUE; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, &fullscreen, NULL); |
| ok(SUCCEEDED(hr), "GetFullscreenState failed, hr %#x.\n", hr); |
| ok(!fullscreen, "Got unexpected fullscreen %#x.\n", hr); |
| |
| check_swapchain_fullscreen_state(swapchain, initial_state); |
| monitor_info.cbSize = sizeof(monitor_info); |
| ret = GetMonitorInfoW(output_desc.Monitor, (MONITORINFO *)&monitor_info); |
| ok(ret, "Failed to get monitor info.\n"); |
| ok(EqualRect(&monitor_info.rcMonitor, &orig_monitor_rect), "Got monitor rect %s, expected %s.\n", |
| wine_dbgstr_rect(&monitor_info.rcMonitor), wine_dbgstr_rect(&orig_monitor_rect)); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, output); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| |
| IDXGIOutput_Release(output); |
| } |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, initial_state); |
| |
| for (i = 0; i < output_count; ++i) |
| { |
| hr = IDXGIAdapter_EnumOutputs(adapter, i, &output); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| monitor_info.cbSize = sizeof(monitor_info); |
| ret = GetMonitorInfoW(output_desc.Monitor, (MONITORINFO *)&monitor_info); |
| ok(ret, "Failed to get monitor info.\n"); |
| |
| ok(EqualRect(&monitor_info.rcMonitor, &output_monitor_info[i].rcMonitor), |
| "Got monitor rect %s, expected %s.\n", |
| wine_dbgstr_rect(&monitor_info.rcMonitor), |
| wine_dbgstr_rect(&output_monitor_info[i].rcMonitor)); |
| |
| IDXGIOutput_Release(output); |
| } |
| |
| HeapFree(GetProcessHeap(), 0, output_monitor_info); |
| } |
| |
| static void test_set_fullscreen(void) |
| { |
| struct swapchain_fullscreen_state initial_state; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| IDXGISwapChain *swapchain; |
| IDXGIFactory *factory; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| swapchain_desc.BufferDesc.Width = 800; |
| swapchain_desc.BufferDesc.Height = 600; |
| swapchain_desc.BufferDesc.RefreshRate.Numerator = 60; |
| swapchain_desc.BufferDesc.RefreshRate.Denominator = 60; |
| swapchain_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| swapchain_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| swapchain_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| swapchain_desc.SampleDesc.Count = 1; |
| swapchain_desc.SampleDesc.Quality = 0; |
| swapchain_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| swapchain_desc.BufferCount = 1; |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, 0, 0, 400, 200, 0, 0, 0, 0); |
| swapchain_desc.Windowed = TRUE; |
| swapchain_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| swapchain_desc.Flags = 0; |
| |
| memset(&initial_state, 0, sizeof(initial_state)); |
| capture_fullscreen_state(&initial_state.fullscreen_state, swapchain_desc.OutputWindow); |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, NULL); |
| ok(SUCCEEDED(hr) || hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE || |
| broken(hr == DXGI_ERROR_UNSUPPORTED), /* Win 7 testbot */ |
| "SetFullscreenState failed, hr %#x.\n", hr); |
| if (FAILED(hr)) |
| { |
| skip("Could not change fullscreen state.\n"); |
| goto done; |
| } |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| |
| DestroyWindow(swapchain_desc.OutputWindow); |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, 0, 0, 400, 200, 0, 0, 0, 0); |
| check_window_fullscreen_state(swapchain_desc.OutputWindow, &initial_state.fullscreen_state); |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| test_swapchain_fullscreen_state(swapchain, adapter, &initial_state); |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| |
| DestroyWindow(swapchain_desc.OutputWindow); |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, 0, 0, 400, 200, 0, 0, 0, 0); |
| check_window_fullscreen_state(swapchain_desc.OutputWindow, &initial_state.fullscreen_state); |
| swapchain_desc.Flags = DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH; |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| test_swapchain_fullscreen_state(swapchain, adapter, &initial_state); |
| |
| done: |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| check_window_fullscreen_state(swapchain_desc.OutputWindow, &initial_state.fullscreen_state); |
| DestroyWindow(swapchain_desc.OutputWindow); |
| |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "Factory has %u references left.\n", refcount); |
| } |
| |
| static void test_default_fullscreen_target_output(void) |
| { |
| IDXGIOutput *output, *containing_output, *target; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| DXGI_OUTPUT_DESC output_desc; |
| IDXGISwapChain *swapchain; |
| unsigned int output_idx; |
| IDXGIFactory *factory; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| ULONG refcount; |
| HRESULT hr; |
| BOOL ret; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| swapchain_desc.BufferDesc.Width = 100; |
| swapchain_desc.BufferDesc.Height = 100; |
| swapchain_desc.BufferDesc.RefreshRate.Numerator = 60; |
| swapchain_desc.BufferDesc.RefreshRate.Denominator = 60; |
| swapchain_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| swapchain_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| swapchain_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| swapchain_desc.SampleDesc.Count = 1; |
| swapchain_desc.SampleDesc.Quality = 0; |
| swapchain_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| swapchain_desc.BufferCount = 1; |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", |
| WS_OVERLAPPEDWINDOW | WS_VISIBLE, 0, 0, 100, 100, 0, 0, 0, 0); |
| swapchain_desc.Windowed = TRUE; |
| swapchain_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| swapchain_desc.Flags = 0; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| output_idx = 0; |
| while ((hr = IDXGIAdapter_EnumOutputs(adapter, output_idx, &output)) != DXGI_ERROR_NOT_FOUND) |
| { |
| ok(SUCCEEDED(hr), "Failed to enumerate output %u, hr %#x.\n", output_idx, hr); |
| |
| hr = IDXGIOutput_GetDesc(output, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| /* Move the OutputWindow to the current output. */ |
| ret = SetWindowPos(swapchain_desc.OutputWindow, 0, |
| output_desc.DesktopCoordinates.left, output_desc.DesktopCoordinates.top, |
| 0, 0, SWP_NOSIZE | SWP_NOZORDER); |
| ok(ret, "SetWindowPos failed.\n"); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &containing_output); |
| ok(SUCCEEDED(hr) || broken(hr == DXGI_ERROR_UNSUPPORTED) /* Win 7 testbot */, |
| "GetContainingOutput failed, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_UNSUPPORTED) |
| { |
| win_skip("GetContainingOutput() not supported.\n"); |
| IDXGIOutput_Release(output); |
| goto done; |
| } |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, NULL); |
| ok(SUCCEEDED(hr) || hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE, |
| "SetFullscreenState failed, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE) |
| { |
| skip("Could not change fullscreen state.\n"); |
| IDXGIOutput_Release(containing_output); |
| IDXGIOutput_Release(output); |
| goto done; |
| } |
| |
| target = NULL; |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, &target); |
| ok(SUCCEEDED(hr), "GetFullscreenState failed, hr %#x.\n", hr); |
| ok(target != containing_output, "Got unexpected output pointers %p, %p.\n", |
| target, containing_output); |
| check_output_equal(target, containing_output); |
| |
| refcount = IDXGIOutput_Release(containing_output); |
| ok(!refcount, "IDXGIOutput has %u references left.\n", refcount); |
| |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &containing_output); |
| ok(SUCCEEDED(hr), "GetContainingOutput failed, hr %#x.\n", hr); |
| ok(containing_output == target, "Got unexpected containing output %p, expected %p.\n", |
| containing_output, target); |
| refcount = IDXGIOutput_Release(containing_output); |
| ok(refcount >= 2, "Got unexpected refcount %u.\n", refcount); |
| refcount = IDXGIOutput_Release(target); |
| ok(refcount >= 1, "Got unexpected refcount %u.\n", refcount); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| |
| IDXGIOutput_Release(output); |
| ++output_idx; |
| } |
| |
| done: |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIAdapter_Release(adapter); |
| ok(!refcount, "Adapter has %u references left.\n", refcount); |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "Factory has %u references left.\n", refcount); |
| DestroyWindow(swapchain_desc.OutputWindow); |
| } |
| |
| static void test_windowed_resize_target(IDXGISwapChain *swapchain, HWND window, |
| struct swapchain_fullscreen_state *state) |
| { |
| struct swapchain_fullscreen_state expected_state; |
| struct fullscreen_state *e; |
| DXGI_MODE_DESC mode; |
| RECT window_rect; |
| unsigned int i; |
| HRESULT hr; |
| BOOL ret; |
| |
| static const struct |
| { |
| unsigned int width, height; |
| } |
| sizes[] = |
| { |
| {200, 200}, |
| {400, 200}, |
| {400, 400}, |
| {600, 800}, |
| {1000, 600}, |
| {1600, 100}, |
| {2000, 1000}, |
| }; |
| |
| check_swapchain_fullscreen_state(swapchain, state); |
| expected_state = *state; |
| e = &expected_state.fullscreen_state; |
| |
| for (i = 0; i < sizeof(sizes) / sizeof(*sizes); ++i) |
| { |
| SetRect(&e->client_rect, 0, 0, sizes[i].width, sizes[i].height); |
| e->window_rect = e->client_rect; |
| ret = AdjustWindowRectEx(&e->window_rect, GetWindowLongW(window, GWL_STYLE), |
| FALSE, GetWindowLongW(window, GWL_EXSTYLE)); |
| ok(ret, "AdjustWindowRectEx failed.\n"); |
| if (GetMenu(window)) |
| e->client_rect.bottom -= GetSystemMetrics(SM_CYMENU); |
| SetRect(&e->window_rect, 0, 0, |
| e->window_rect.right - e->window_rect.left, |
| e->window_rect.bottom - e->window_rect.top); |
| GetWindowRect(window, &window_rect); |
| OffsetRect(&e->window_rect, window_rect.left, window_rect.top); |
| if (e->window_rect.right >= e->monitor_rect.right |
| || e->window_rect.bottom >= e->monitor_rect.bottom) |
| { |
| skip("Test %u: Window %s does not fit on screen %s.\n", |
| i, wine_dbgstr_rect(&e->window_rect), wine_dbgstr_rect(&e->monitor_rect)); |
| continue; |
| } |
| |
| memset(&mode, 0, sizeof(mode)); |
| mode.Width = sizes[i].width; |
| mode.Height = sizes[i].height; |
| hr = IDXGISwapChain_ResizeTarget(swapchain, &mode); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| } |
| |
| ret = MoveWindow(window, 0, 0, 0, 0, TRUE); |
| ok(ret, "MoveWindow failed.\n"); |
| GetWindowRect(window, &e->window_rect); |
| GetClientRect(window, &e->client_rect); |
| ret = MoveWindow(window, 0, 0, 200, 200, TRUE); |
| |
| memset(&mode, 0, sizeof(mode)); |
| hr = IDXGISwapChain_ResizeTarget(swapchain, &mode); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| GetWindowRect(window, &e->window_rect); |
| GetClientRect(window, &e->client_rect); |
| *state = expected_state; |
| } |
| |
| static void test_fullscreen_resize_target(IDXGISwapChain *swapchain, |
| const struct swapchain_fullscreen_state *initial_state) |
| { |
| struct swapchain_fullscreen_state expected_state; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| DXGI_OUTPUT_DESC output_desc; |
| unsigned int i, mode_count; |
| DXGI_MODE_DESC *modes; |
| IDXGIOutput *target; |
| HRESULT hr; |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_GetFullscreenState(swapchain, NULL, &target); |
| ok(SUCCEEDED(hr), "GetFullscreenState failed, hr %#x.\n", hr); |
| |
| hr = IDXGIOutput_GetDisplayModeList(target, DXGI_FORMAT_R8G8B8A8_UNORM, 0, &mode_count, NULL); |
| ok(SUCCEEDED(hr) || broken(hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE), /* Win 7 testbot */ |
| "Failed to list modes, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE) |
| { |
| win_skip("GetDisplayModeList() not supported.\n"); |
| IDXGIOutput_Release(target); |
| return; |
| } |
| |
| modes = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*modes) * mode_count); |
| ok(!!modes, "Failed to allocate memory.\n"); |
| |
| hr = IDXGIOutput_GetDisplayModeList(target, DXGI_FORMAT_R8G8B8A8_UNORM, 0, &mode_count, modes); |
| ok(SUCCEEDED(hr), "Failed to list modes, hr %#x.\n", hr); |
| |
| expected_state = *initial_state; |
| for (i = 0; i < min(mode_count, 20); ++i) |
| { |
| /* FIXME: Modes with scaling aren't fully tested. */ |
| if (!(swapchain_desc.Flags & DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH) |
| && modes[i].Scaling != DXGI_MODE_SCALING_UNSPECIFIED) |
| continue; |
| |
| hr = IDXGIOutput_GetDesc(target, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &swapchain_desc, &output_desc.DesktopCoordinates, modes[i].Width, modes[i].Height, NULL); |
| |
| hr = IDXGISwapChain_ResizeTarget(swapchain, &modes[i]); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| hr = IDXGIOutput_GetDesc(target, &output_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| ok(EqualRect(&output_desc.DesktopCoordinates, &expected_state.fullscreen_state.monitor_rect), |
| "Got desktop coordinates %s, expected %s.\n", |
| wine_dbgstr_rect(&output_desc.DesktopCoordinates), |
| wine_dbgstr_rect(&expected_state.fullscreen_state.monitor_rect)); |
| } |
| |
| HeapFree(GetProcessHeap(), 0, modes); |
| IDXGIOutput_Release(target); |
| } |
| |
| static void test_resize_target(void) |
| { |
| struct swapchain_fullscreen_state initial_state, expected_state; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| IDXGISwapChain *swapchain; |
| IDXGIFactory *factory; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| unsigned int i; |
| ULONG refcount; |
| HRESULT hr; |
| |
| static const struct |
| { |
| POINT origin; |
| BOOL fullscreen; |
| BOOL menu; |
| unsigned int flags; |
| } |
| tests[] = |
| { |
| {{ 0, 0}, TRUE, FALSE, 0}, |
| {{10, 10}, TRUE, FALSE, 0}, |
| {{ 0, 0}, TRUE, FALSE, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH}, |
| {{10, 10}, TRUE, FALSE, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH}, |
| {{ 0, 0}, FALSE, FALSE, 0}, |
| {{ 0, 0}, FALSE, FALSE, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH}, |
| {{10, 10}, FALSE, FALSE, 0}, |
| {{10, 10}, FALSE, FALSE, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH}, |
| {{ 0, 0}, FALSE, TRUE, 0}, |
| {{ 0, 0}, FALSE, TRUE, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH}, |
| {{10, 10}, FALSE, TRUE, 0}, |
| {{10, 10}, FALSE, TRUE, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH}, |
| }; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| swapchain_desc.BufferDesc.Width = 800; |
| swapchain_desc.BufferDesc.Height = 600; |
| swapchain_desc.BufferDesc.RefreshRate.Numerator = 60; |
| swapchain_desc.BufferDesc.RefreshRate.Denominator = 1; |
| swapchain_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| swapchain_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| swapchain_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| swapchain_desc.SampleDesc.Count = 1; |
| swapchain_desc.SampleDesc.Quality = 0; |
| swapchain_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| swapchain_desc.BufferCount = 1; |
| swapchain_desc.Windowed = TRUE; |
| swapchain_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| swapchain_desc.Flags = 0; |
| |
| for (i = 0; i < sizeof(tests) / sizeof(*tests); ++i) |
| { |
| swapchain_desc.Flags = tests[i].flags; |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, |
| tests[i].origin.x, tests[i].origin.y, 400, 200, 0, 0, 0, 0); |
| if (tests[i].menu) |
| { |
| HMENU menu_bar = CreateMenu(); |
| HMENU menu = CreateMenu(); |
| AppendMenuA(menu_bar, MF_POPUP, (UINT_PTR)menu, "Menu"); |
| SetMenu(swapchain_desc.OutputWindow, menu_bar); |
| } |
| |
| memset(&initial_state, 0, sizeof(initial_state)); |
| capture_fullscreen_state(&initial_state.fullscreen_state, swapchain_desc.OutputWindow); |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| |
| expected_state = initial_state; |
| if (tests[i].fullscreen) |
| { |
| expected_state.fullscreen = TRUE; |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &swapchain_desc, &initial_state.fullscreen_state.monitor_rect, 800, 600, NULL); |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &expected_state.target); |
| ok(SUCCEEDED(hr) || broken(hr == DXGI_ERROR_UNSUPPORTED) /* Win 7 testbot */, |
| "GetContainingOutput failed, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_UNSUPPORTED) |
| { |
| win_skip("GetContainingOutput() not supported.\n"); |
| IDXGISwapChain_Release(swapchain); |
| DestroyWindow(swapchain_desc.OutputWindow); |
| continue; |
| } |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, NULL); |
| ok(SUCCEEDED(hr) || hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE, |
| "SetFullscreenState failed, hr %#x.\n", hr); |
| if (hr == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE) |
| { |
| skip("Could not change fullscreen state.\n"); |
| IDXGIOutput_Release(expected_state.target); |
| IDXGISwapChain_Release(swapchain); |
| DestroyWindow(swapchain_desc.OutputWindow); |
| continue; |
| } |
| } |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| hr = IDXGISwapChain_ResizeTarget(swapchain, NULL); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| |
| if (tests[i].fullscreen) |
| { |
| test_fullscreen_resize_target(swapchain, &expected_state); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| IDXGIOutput_Release(expected_state.target); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| expected_state = initial_state; |
| } |
| else |
| { |
| test_windowed_resize_target(swapchain, swapchain_desc.OutputWindow, &expected_state); |
| |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| } |
| |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| check_window_fullscreen_state(swapchain_desc.OutputWindow, &expected_state.fullscreen_state); |
| DestroyWindow(swapchain_desc.OutputWindow); |
| } |
| |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "Factory has %u references left.\n", refcount); |
| } |
| |
| static void test_inexact_modes(void) |
| { |
| struct swapchain_fullscreen_state initial_state, expected_state; |
| DXGI_SWAP_CHAIN_DESC swapchain_desc, result_desc; |
| IDXGIOutput *output = NULL; |
| IDXGISwapChain *swapchain; |
| IDXGIFactory *factory; |
| IDXGIAdapter *adapter; |
| IDXGIDevice *device; |
| unsigned int i; |
| ULONG refcount; |
| HRESULT hr; |
| |
| static const struct |
| { |
| unsigned int width, height; |
| } |
| sizes[] = |
| { |
| {101, 101}, |
| {203, 204}, |
| {799, 601}, |
| }; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| swapchain_desc.BufferDesc.Width = 800; |
| swapchain_desc.BufferDesc.Height = 600; |
| swapchain_desc.BufferDesc.RefreshRate.Numerator = 60; |
| swapchain_desc.BufferDesc.RefreshRate.Denominator = 1; |
| swapchain_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| swapchain_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| swapchain_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| swapchain_desc.SampleDesc.Count = 1; |
| swapchain_desc.SampleDesc.Quality = 0; |
| swapchain_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| swapchain_desc.BufferCount = 1; |
| swapchain_desc.OutputWindow = CreateWindowA("static", "dxgi_test", 0, 0, 0, 400, 200, 0, 0, 0, 0); |
| swapchain_desc.Windowed = FALSE; |
| swapchain_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| swapchain_desc.Flags = DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH; |
| |
| memset(&initial_state, 0, sizeof(initial_state)); |
| capture_fullscreen_state(&initial_state.fullscreen_state, swapchain_desc.OutputWindow); |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetContainingOutput(swapchain, &output); |
| ok(SUCCEEDED(hr) || broken(hr == DXGI_ERROR_UNSUPPORTED) /* Win 7 testbot */, |
| "GetContainingOutput failed, hr %#x.\n", hr); |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| if (hr == DXGI_ERROR_UNSUPPORTED) |
| { |
| win_skip("GetContainingOutput() not supported.\n"); |
| goto done; |
| } |
| if (result_desc.Windowed) |
| { |
| win_skip("Fullscreen not supported.\n"); |
| goto done; |
| } |
| |
| check_window_fullscreen_state(swapchain_desc.OutputWindow, &initial_state.fullscreen_state); |
| |
| for (i = 0; i < sizeof(sizes) / sizeof(*sizes); ++i) |
| { |
| /* Test CreateSwapChain(). */ |
| swapchain_desc.BufferDesc.Width = sizes[i].width; |
| swapchain_desc.BufferDesc.Height = sizes[i].height; |
| swapchain_desc.Windowed = FALSE; |
| |
| expected_state = initial_state; |
| compute_expected_swapchain_fullscreen_state_after_fullscreen_change(&expected_state, |
| &swapchain_desc, &initial_state.fullscreen_state.monitor_rect, |
| sizes[i].width, sizes[i].height, output); |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| ok(result_desc.BufferDesc.Width == sizes[i].width, "Got width %u, expected %u.\n", |
| result_desc.BufferDesc.Width, sizes[i].width); |
| ok(result_desc.BufferDesc.Height == sizes[i].height, "Got height %u, expected %u.\n", |
| result_desc.BufferDesc.Height, sizes[i].height); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| |
| /* Test SetFullscreenState(). */ |
| swapchain_desc.BufferDesc.Width = sizes[i].width; |
| swapchain_desc.BufferDesc.Height = sizes[i].height; |
| swapchain_desc.Windowed = TRUE; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, output); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| ok(result_desc.BufferDesc.Width == sizes[i].width, "Got width %u, expected %u.\n", |
| result_desc.BufferDesc.Width, sizes[i].width); |
| ok(result_desc.BufferDesc.Height == sizes[i].height, "Got height %u, expected %u.\n", |
| result_desc.BufferDesc.Height, sizes[i].height); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| |
| /* Test ResizeTarget(). */ |
| swapchain_desc.BufferDesc.Width = 800; |
| swapchain_desc.BufferDesc.Height = 600; |
| swapchain_desc.Windowed = TRUE; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "CreateSwapChain failed, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, TRUE, output); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| |
| swapchain_desc.BufferDesc.Width = sizes[i].width; |
| swapchain_desc.BufferDesc.Height = sizes[i].height; |
| hr = IDXGISwapChain_ResizeTarget(swapchain, &swapchain_desc.BufferDesc); |
| ok(SUCCEEDED(hr), "ResizeTarget failed, hr %#x.\n", hr); |
| |
| check_swapchain_fullscreen_state(swapchain, &expected_state); |
| hr = IDXGISwapChain_GetDesc(swapchain, &result_desc); |
| ok(SUCCEEDED(hr), "GetDesc failed, hr %#x.\n", hr); |
| ok(result_desc.BufferDesc.Width == 800, "Got width %u.\n", result_desc.BufferDesc.Width); |
| ok(result_desc.BufferDesc.Height == 600, "Got height %u.\n", result_desc.BufferDesc.Height); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| check_swapchain_fullscreen_state(swapchain, &initial_state); |
| |
| refcount = IDXGISwapChain_Release(swapchain); |
| ok(!refcount, "IDXGISwapChain has %u references left.\n", refcount); |
| } |
| |
| done: |
| if (output) |
| IDXGIOutput_Release(output); |
| IDXGIAdapter_Release(adapter); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "Factory has %u references left.\n", refcount); |
| } |
| |
| static void test_create_factory(void) |
| { |
| IDXGIFactory1 *factory; |
| IUnknown *iface; |
| HRESULT hr; |
| |
| iface = (void *)0xdeadbeef; |
| hr = CreateDXGIFactory(&IID_IDXGIDevice, (void **)&iface); |
| ok(hr == E_NOINTERFACE, "Got unexpected hr %#x.\n", hr); |
| ok(!iface, "Got unexpected iface %p.\n", iface); |
| |
| hr = CreateDXGIFactory(&IID_IUnknown, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IUnknown, hr %#x.\n", hr); |
| IUnknown_Release(iface); |
| |
| hr = CreateDXGIFactory(&IID_IDXGIObject, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IDXGIObject, hr %#x.\n", hr); |
| IUnknown_Release(iface); |
| |
| factory = (void *)0xdeadbeef; |
| hr = CreateDXGIFactory(&IID_IDXGIFactory, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IDXGIFactory, hr %#x.\n", hr); |
| hr = IUnknown_QueryInterface(iface, &IID_IDXGIFactory1, (void **)&factory); |
| ok(hr == E_NOINTERFACE, "Got unexpected hr %#x.\n", hr); |
| ok(!factory, "Got unexpected factory %p.\n", factory); |
| IUnknown_Release(iface); |
| |
| iface = (void *)0xdeadbeef; |
| hr = CreateDXGIFactory(&IID_IDXGIFactory1, (void **)&iface); |
| ok(hr == E_NOINTERFACE, "Got unexpected hr %#x.\n", hr); |
| ok(!iface, "Got unexpected iface %p.\n", iface); |
| |
| if (!pCreateDXGIFactory1) |
| { |
| win_skip("CreateDXGIFactory1 not available, skipping tests.\n"); |
| return; |
| } |
| |
| iface = (void *)0xdeadbeef; |
| hr = pCreateDXGIFactory1(&IID_IDXGIDevice, (void **)&iface); |
| ok(hr == E_NOINTERFACE, "Got unexpected hr %#x.\n", hr); |
| ok(!iface, "Got unexpected iface %p.\n", iface); |
| |
| hr = pCreateDXGIFactory1(&IID_IUnknown, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IUnknown, hr %#x.\n", hr); |
| IUnknown_Release(iface); |
| |
| hr = pCreateDXGIFactory1(&IID_IDXGIObject, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IDXGIObject, hr %#x.\n", hr); |
| IUnknown_Release(iface); |
| |
| hr = pCreateDXGIFactory1(&IID_IDXGIFactory, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IDXGIFactory, hr %#x.\n", hr); |
| hr = IUnknown_QueryInterface(iface, &IID_IDXGIFactory1, (void **)&factory); |
| ok(SUCCEEDED(hr), "Failed to query IDXGIFactory1 interface, hr %#x.\n", hr); |
| IDXGIFactory1_Release(factory); |
| IUnknown_Release(iface); |
| |
| hr = pCreateDXGIFactory1(&IID_IDXGIFactory1, (void **)&iface); |
| ok(SUCCEEDED(hr), "Failed to create factory with IID_IDXGIFactory1, hr %#x.\n", hr); |
| IUnknown_Release(iface); |
| } |
| |
| static void test_private_data(void) |
| { |
| ULONG refcount, expected_refcount; |
| IDXGIDevice *device; |
| HRESULT hr; |
| IDXGIDevice *test_object; |
| IUnknown *ptr; |
| static const DWORD data[] = {1, 2, 3, 4}; |
| UINT size; |
| static const GUID dxgi_private_data_test_guid = |
| { |
| 0xfdb37466, |
| 0x428f, |
| 0x4edf, |
| {0xa3, 0x7f, 0x9b, 0x1d, 0xf4, 0x88, 0xc5, 0xfc} |
| }; |
| static const GUID dxgi_private_data_test_guid2 = |
| { |
| 0x2e5afac2, |
| 0x87b5, |
| 0x4c10, |
| {0x9b, 0x4b, 0x89, 0xd7, 0xd1, 0x12, 0xe7, 0x2b} |
| }; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| |
| test_object = create_device(0); |
| |
| /* SetPrivateData with a pointer of NULL has the purpose of FreePrivateData in previous |
| * d3d versions. A successful clear returns S_OK. A redundant clear S_FALSE. Setting a |
| * NULL interface is not considered a clear but as setting an interface pointer that |
| * happens to be NULL. */ |
| hr = IDXGIDevice_SetPrivateData(device, &dxgi_private_data_test_guid, 0, NULL); |
| ok(hr == S_FALSE, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice_SetPrivateData(device, &dxgi_private_data_test_guid, ~0U, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice_SetPrivateData(device, &dxgi_private_data_test_guid, ~0U, NULL); |
| ok(hr == S_FALSE, "Got unexpected hr %#x.\n", hr); |
| |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| size = sizeof(ptr) * 2; |
| ptr = (IUnknown *)0xdeadbeef; |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid, &size, &ptr); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| ok(!ptr, "Got unexpected pointer %p.\n", ptr); |
| ok(size == sizeof(IUnknown *), "Got unexpected size %u.\n", size); |
| |
| refcount = get_refcount((IUnknown *)test_object); |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, |
| (IUnknown *)test_object); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| expected_refcount = refcount + 1; |
| refcount = get_refcount((IUnknown *)test_object); |
| ok(refcount == expected_refcount, "Got unexpected refcount %u, expected %u.\n", refcount, expected_refcount); |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, |
| (IUnknown *)test_object); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| refcount = get_refcount((IUnknown *)test_object); |
| ok(refcount == expected_refcount, "Got unexpected refcount %u, expected %u.\n", refcount, expected_refcount); |
| |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| expected_refcount--; |
| refcount = get_refcount((IUnknown *)test_object); |
| ok(refcount == expected_refcount, "Got unexpected refcount %u, expected %u.\n", refcount, expected_refcount); |
| |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, |
| (IUnknown *)test_object); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| size = sizeof(data); |
| hr = IDXGIDevice_SetPrivateData(device, &dxgi_private_data_test_guid, size, data); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| refcount = get_refcount((IUnknown *)test_object); |
| ok(refcount == expected_refcount, "Got unexpected refcount %u, expected %u.\n", refcount, expected_refcount); |
| hr = IDXGIDevice_SetPrivateData(device, &dxgi_private_data_test_guid, 42, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice_SetPrivateData(device, &dxgi_private_data_test_guid, 42, NULL); |
| ok(hr == S_FALSE, "Got unexpected hr %#x.\n", hr); |
| |
| hr = IDXGIDevice_SetPrivateDataInterface(device, &dxgi_private_data_test_guid, |
| (IUnknown *)test_object); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| expected_refcount++; |
| size = 2 * sizeof(ptr); |
| ptr = NULL; |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid, &size, &ptr); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| ok(size == sizeof(test_object), "Got unexpected size %u.\n", size); |
| expected_refcount++; |
| refcount = get_refcount((IUnknown *)test_object); |
| ok(refcount == expected_refcount, "Got unexpected refcount %u, expected %u.\n", refcount, expected_refcount); |
| if (ptr) |
| IUnknown_Release(ptr); |
| expected_refcount--; |
| |
| ptr = (IUnknown *)0xdeadbeef; |
| size = 1; |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid, &size, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| ok(size == sizeof(device), "Got unexpected size %u.\n", size); |
| size = 2 * sizeof(ptr); |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid, &size, NULL); |
| ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| ok(size == sizeof(device), "Got unexpected size %u.\n", size); |
| refcount = get_refcount((IUnknown *)test_object); |
| ok(refcount == expected_refcount, "Got unexpected refcount %u, expected %u.\n", refcount, expected_refcount); |
| |
| size = 1; |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid, &size, &ptr); |
| ok(hr == DXGI_ERROR_MORE_DATA, "Got unexpected hr %#x.\n", hr); |
| ok(size == sizeof(device), "Got unexpected size %u.\n", size); |
| ok(ptr == (IUnknown *)0xdeadbeef, "Got unexpected pointer %p.\n", ptr); |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid2, NULL, NULL); |
| ok(hr == E_INVALIDARG, "Got unexpected hr %#x.\n", hr); |
| size = 0xdeadbabe; |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid2, &size, &ptr); |
| ok(hr == DXGI_ERROR_NOT_FOUND, "Got unexpected hr %#x.\n", hr); |
| ok(size == 0, "Got unexpected size %u.\n", size); |
| ok(ptr == (IUnknown *)0xdeadbeef, "Got unexpected pointer %p.\n", ptr); |
| hr = IDXGIDevice_GetPrivateData(device, &dxgi_private_data_test_guid, NULL, &ptr); |
| ok(hr == E_INVALIDARG, "Got unexpected hr %#x.\n", hr); |
| ok(ptr == (IUnknown *)0xdeadbeef, "Got unexpected pointer %p.\n", ptr); |
| |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| refcount = IDXGIDevice_Release(test_object); |
| ok(!refcount, "Test object has %u references left.\n", refcount); |
| } |
| |
| static void test_swapchain_resize(void) |
| { |
| DXGI_SWAP_CHAIN_DESC swapchain_desc; |
| D3D10_TEXTURE2D_DESC texture_desc; |
| DXGI_SURFACE_DESC surface_desc; |
| IDXGISwapChain *swapchain; |
| ID3D10Texture2D *texture; |
| IDXGISurface *surface; |
| IDXGIAdapter *adapter; |
| IDXGIFactory *factory; |
| IDXGIDevice *device; |
| RECT client_rect, r; |
| ULONG refcount; |
| HWND window; |
| HRESULT hr; |
| BOOL ret; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| window = CreateWindowA("static", "dxgi_test", WS_OVERLAPPEDWINDOW | WS_VISIBLE, |
| 0, 0, 640, 480, NULL, NULL, NULL, NULL); |
| ret = GetClientRect(window, &client_rect); |
| ok(ret, "Failed to get client rect.\n"); |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "Failed to get adapter, hr %#x.\n", hr); |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "Failed to get factory, hr %#x.\n", hr); |
| IDXGIAdapter_Release(adapter); |
| |
| swapchain_desc.BufferDesc.Width = 640; |
| swapchain_desc.BufferDesc.Height = 480; |
| swapchain_desc.BufferDesc.RefreshRate.Numerator = 60; |
| swapchain_desc.BufferDesc.RefreshRate.Denominator = 1; |
| swapchain_desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| swapchain_desc.BufferDesc.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED; |
| swapchain_desc.BufferDesc.Scaling = DXGI_MODE_SCALING_UNSPECIFIED; |
| swapchain_desc.SampleDesc.Count = 1; |
| swapchain_desc.SampleDesc.Quality = 0; |
| swapchain_desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| swapchain_desc.BufferCount = 1; |
| swapchain_desc.OutputWindow = window; |
| swapchain_desc.Windowed = TRUE; |
| swapchain_desc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; |
| swapchain_desc.Flags = 0; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, (IUnknown *)device, &swapchain_desc, &swapchain); |
| ok(SUCCEEDED(hr), "Failed to create swapchain, hr %#x.\n", hr); |
| IDXGIFactory_Release(factory); |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_IDXGISurface, (void **)&surface); |
| ok(SUCCEEDED(hr), "Failed to get buffer, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_ID3D10Texture2D, (void **)&texture); |
| ok(SUCCEEDED(hr), "Failed to get buffer, hr %#x.\n", hr); |
| |
| ret = GetClientRect(window, &r); |
| ok(ret, "Failed to get client rect.\n"); |
| ok(EqualRect(&r, &client_rect), "Got unexpected rect %s, expected %s.\n", |
| wine_dbgstr_rect(&r), wine_dbgstr_rect(&client_rect)); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok(SUCCEEDED(hr), "Failed to get swapchain desc, hr %#x.\n", hr); |
| ok(swapchain_desc.BufferDesc.Width == 640, |
| "Got unexpected BufferDesc.Width %u.\n", swapchain_desc.BufferDesc.Width); |
| ok(swapchain_desc.BufferDesc.Height == 480, |
| "Got unexpected bufferDesc.Height %u.\n", swapchain_desc.BufferDesc.Height); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Numerator == 60, |
| "Got unexpected BufferDesc.RefreshRate.Numerator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Numerator); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Denominator == 1, |
| "Got unexpected BufferDesc.RefreshRate.Denominator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Denominator); |
| ok(swapchain_desc.BufferDesc.Format == DXGI_FORMAT_R8G8B8A8_UNORM, |
| "Got unexpected BufferDesc.Format %#x.\n", swapchain_desc.BufferDesc.Format); |
| ok(swapchain_desc.BufferDesc.ScanlineOrdering == DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED, |
| "Got unexpected BufferDesc.ScanlineOrdering %#x.\n", swapchain_desc.BufferDesc.ScanlineOrdering); |
| ok(swapchain_desc.BufferDesc.Scaling == DXGI_MODE_SCALING_UNSPECIFIED, |
| "Got unexpected BufferDesc.Scaling %#x.\n", swapchain_desc.BufferDesc.Scaling); |
| ok(swapchain_desc.SampleDesc.Count == 1, |
| "Got unexpected SampleDesc.Count %u.\n", swapchain_desc.SampleDesc.Count); |
| ok(!swapchain_desc.SampleDesc.Quality, |
| "Got unexpected SampleDesc.Quality %u.\n", swapchain_desc.SampleDesc.Quality); |
| ok(swapchain_desc.BufferUsage == DXGI_USAGE_RENDER_TARGET_OUTPUT, |
| "Got unexpected BufferUsage %#x.\n", swapchain_desc.BufferUsage); |
| ok(swapchain_desc.BufferCount == 1, |
| "Got unexpected BufferCount %u.\n", swapchain_desc.BufferCount); |
| ok(swapchain_desc.OutputWindow == window, |
| "Got unexpected OutputWindow %p, expected %p.\n", swapchain_desc.OutputWindow, window); |
| ok(swapchain_desc.Windowed, |
| "Got unexpected Windowed %#x.\n", swapchain_desc.Windowed); |
| ok(swapchain_desc.SwapEffect == DXGI_SWAP_EFFECT_DISCARD, |
| "Got unexpected SwapEffect %#x.\n", swapchain_desc.SwapEffect); |
| ok(!swapchain_desc.Flags, |
| "Got unexpected Flags %#x.\n", swapchain_desc.Flags); |
| |
| hr = IDXGISurface_GetDesc(surface, &surface_desc); |
| ok(SUCCEEDED(hr), "Failed to get surface desc, hr %#x.\n", hr); |
| ok(surface_desc.Width == 640, "Got unexpected Width %u.\n", surface_desc.Width); |
| ok(surface_desc.Height == 480, "Got unexpected Height %u.\n", surface_desc.Height); |
| ok(surface_desc.Format == DXGI_FORMAT_R8G8B8A8_UNORM, "Got unexpected Format %#x.\n", surface_desc.Format); |
| ok(surface_desc.SampleDesc.Count == 1, "Got unexpected SampleDesc.Count %u.\n", surface_desc.SampleDesc.Count); |
| ok(!surface_desc.SampleDesc.Quality, "Got unexpected SampleDesc.Quality %u.\n", surface_desc.SampleDesc.Quality); |
| |
| ID3D10Texture2D_GetDesc(texture, &texture_desc); |
| ok(texture_desc.Width == 640, "Got unexpected Width %u.\n", texture_desc.Width); |
| ok(texture_desc.Height == 480, "Got unexpected Height %u.\n", texture_desc.Height); |
| ok(texture_desc.MipLevels == 1, "Got unexpected MipLevels %u.\n", texture_desc.MipLevels); |
| ok(texture_desc.ArraySize == 1, "Got unexpected ArraySize %u.\n", texture_desc.ArraySize); |
| ok(texture_desc.Format == DXGI_FORMAT_R8G8B8A8_UNORM, "Got unexpected Format %#x.\n", texture_desc.Format); |
| ok(texture_desc.SampleDesc.Count == 1, "Got unexpected SampleDesc.Count %u.\n", texture_desc.SampleDesc.Count); |
| ok(!texture_desc.SampleDesc.Quality, "Got unexpected SampleDesc.Quality %u.\n", texture_desc.SampleDesc.Quality); |
| ok(texture_desc.Usage == D3D10_USAGE_DEFAULT, "Got unexpected Usage %#x.\n", texture_desc.Usage); |
| ok(texture_desc.BindFlags == D3D10_BIND_RENDER_TARGET, "Got unexpected BindFlags %#x.\n", texture_desc.BindFlags); |
| ok(!texture_desc.CPUAccessFlags, "Got unexpected CPUAccessFlags %#x.\n", texture_desc.CPUAccessFlags); |
| ok(!texture_desc.MiscFlags, "Got unexpected MiscFlags %#x.\n", texture_desc.MiscFlags); |
| |
| hr = IDXGISwapChain_ResizeBuffers(swapchain, 1, 320, 240, DXGI_FORMAT_R8G8B8A8_UNORM_SRGB, 0); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| |
| ret = GetClientRect(window, &r); |
| ok(ret, "Failed to get client rect.\n"); |
| ok(EqualRect(&r, &client_rect), "Got unexpected rect %s, expected %s.\n", |
| wine_dbgstr_rect(&r), wine_dbgstr_rect(&client_rect)); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok(SUCCEEDED(hr), "Failed to get swapchain desc, hr %#x.\n", hr); |
| ok(swapchain_desc.BufferDesc.Width == 640, |
| "Got unexpected BufferDesc.Width %u.\n", swapchain_desc.BufferDesc.Width); |
| ok(swapchain_desc.BufferDesc.Height == 480, |
| "Got unexpected bufferDesc.Height %u.\n", swapchain_desc.BufferDesc.Height); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Numerator == 60, |
| "Got unexpected BufferDesc.RefreshRate.Numerator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Numerator); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Denominator == 1, |
| "Got unexpected BufferDesc.RefreshRate.Denominator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Denominator); |
| ok(swapchain_desc.BufferDesc.Format == DXGI_FORMAT_R8G8B8A8_UNORM, |
| "Got unexpected BufferDesc.Format %#x.\n", swapchain_desc.BufferDesc.Format); |
| ok(swapchain_desc.BufferDesc.ScanlineOrdering == DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED, |
| "Got unexpected BufferDesc.ScanlineOrdering %#x.\n", swapchain_desc.BufferDesc.ScanlineOrdering); |
| ok(swapchain_desc.BufferDesc.Scaling == DXGI_MODE_SCALING_UNSPECIFIED, |
| "Got unexpected BufferDesc.Scaling %#x.\n", swapchain_desc.BufferDesc.Scaling); |
| ok(swapchain_desc.SampleDesc.Count == 1, |
| "Got unexpected SampleDesc.Count %u.\n", swapchain_desc.SampleDesc.Count); |
| ok(!swapchain_desc.SampleDesc.Quality, |
| "Got unexpected SampleDesc.Quality %u.\n", swapchain_desc.SampleDesc.Quality); |
| ok(swapchain_desc.BufferUsage == DXGI_USAGE_RENDER_TARGET_OUTPUT, |
| "Got unexpected BufferUsage %#x.\n", swapchain_desc.BufferUsage); |
| ok(swapchain_desc.BufferCount == 1, |
| "Got unexpected BufferCount %u.\n", swapchain_desc.BufferCount); |
| ok(swapchain_desc.OutputWindow == window, |
| "Got unexpected OutputWindow %p, expected %p.\n", swapchain_desc.OutputWindow, window); |
| ok(swapchain_desc.Windowed, |
| "Got unexpected Windowed %#x.\n", swapchain_desc.Windowed); |
| ok(swapchain_desc.SwapEffect == DXGI_SWAP_EFFECT_DISCARD, |
| "Got unexpected SwapEffect %#x.\n", swapchain_desc.SwapEffect); |
| ok(!swapchain_desc.Flags, |
| "Got unexpected Flags %#x.\n", swapchain_desc.Flags); |
| |
| hr = IDXGISurface_GetDesc(surface, &surface_desc); |
| ok(SUCCEEDED(hr), "Failed to get surface desc, hr %#x.\n", hr); |
| ok(surface_desc.Width == 640, "Got unexpected Width %u.\n", surface_desc.Width); |
| ok(surface_desc.Height == 480, "Got unexpected Height %u.\n", surface_desc.Height); |
| ok(surface_desc.Format == DXGI_FORMAT_R8G8B8A8_UNORM, "Got unexpected Format %#x.\n", surface_desc.Format); |
| ok(surface_desc.SampleDesc.Count == 1, "Got unexpected SampleDesc.Count %u.\n", surface_desc.SampleDesc.Count); |
| ok(!surface_desc.SampleDesc.Quality, "Got unexpected SampleDesc.Quality %u.\n", surface_desc.SampleDesc.Quality); |
| |
| ID3D10Texture2D_GetDesc(texture, &texture_desc); |
| ok(texture_desc.Width == 640, "Got unexpected Width %u.\n", texture_desc.Width); |
| ok(texture_desc.Height == 480, "Got unexpected Height %u.\n", texture_desc.Height); |
| ok(texture_desc.MipLevels == 1, "Got unexpected MipLevels %u.\n", texture_desc.MipLevels); |
| ok(texture_desc.ArraySize == 1, "Got unexpected ArraySize %u.\n", texture_desc.ArraySize); |
| ok(texture_desc.Format == DXGI_FORMAT_R8G8B8A8_UNORM, "Got unexpected Format %#x.\n", texture_desc.Format); |
| ok(texture_desc.SampleDesc.Count == 1, "Got unexpected SampleDesc.Count %u.\n", texture_desc.SampleDesc.Count); |
| ok(!texture_desc.SampleDesc.Quality, "Got unexpected SampleDesc.Quality %u.\n", texture_desc.SampleDesc.Quality); |
| ok(texture_desc.Usage == D3D10_USAGE_DEFAULT, "Got unexpected Usage %#x.\n", texture_desc.Usage); |
| ok(texture_desc.BindFlags == D3D10_BIND_RENDER_TARGET, "Got unexpected BindFlags %#x.\n", texture_desc.BindFlags); |
| ok(!texture_desc.CPUAccessFlags, "Got unexpected CPUAccessFlags %#x.\n", texture_desc.CPUAccessFlags); |
| ok(!texture_desc.MiscFlags, "Got unexpected MiscFlags %#x.\n", texture_desc.MiscFlags); |
| |
| ID3D10Texture2D_Release(texture); |
| IDXGISurface_Release(surface); |
| hr = IDXGISwapChain_ResizeBuffers(swapchain, 1, 320, 240, DXGI_FORMAT_R8G8B8A8_UNORM_SRGB, 0); |
| ok(SUCCEEDED(hr), "Failed to resize buffers, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_IDXGISurface, (void **)&surface); |
| ok(SUCCEEDED(hr), "Failed to get buffer, hr %#x.\n", hr); |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_ID3D10Texture2D, (void **)&texture); |
| ok(SUCCEEDED(hr), "Failed to get buffer, hr %#x.\n", hr); |
| |
| ret = GetClientRect(window, &r); |
| ok(ret, "Failed to get client rect.\n"); |
| ok(EqualRect(&r, &client_rect), "Got unexpected rect %s, expected %s.\n", |
| wine_dbgstr_rect(&r), wine_dbgstr_rect(&client_rect)); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok(SUCCEEDED(hr), "Failed to get swapchain desc, hr %#x.\n", hr); |
| ok(swapchain_desc.BufferDesc.Width == 320, |
| "Got unexpected BufferDesc.Width %u.\n", swapchain_desc.BufferDesc.Width); |
| ok(swapchain_desc.BufferDesc.Height == 240, |
| "Got unexpected bufferDesc.Height %u.\n", swapchain_desc.BufferDesc.Height); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Numerator == 60, |
| "Got unexpected BufferDesc.RefreshRate.Numerator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Numerator); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Denominator == 1, |
| "Got unexpected BufferDesc.RefreshRate.Denominator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Denominator); |
| ok(swapchain_desc.BufferDesc.Format == DXGI_FORMAT_R8G8B8A8_UNORM_SRGB, |
| "Got unexpected BufferDesc.Format %#x.\n", swapchain_desc.BufferDesc.Format); |
| ok(swapchain_desc.BufferDesc.ScanlineOrdering == DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED, |
| "Got unexpected BufferDesc.ScanlineOrdering %#x.\n", swapchain_desc.BufferDesc.ScanlineOrdering); |
| ok(swapchain_desc.BufferDesc.Scaling == DXGI_MODE_SCALING_UNSPECIFIED, |
| "Got unexpected BufferDesc.Scaling %#x.\n", swapchain_desc.BufferDesc.Scaling); |
| ok(swapchain_desc.SampleDesc.Count == 1, |
| "Got unexpected SampleDesc.Count %u.\n", swapchain_desc.SampleDesc.Count); |
| ok(!swapchain_desc.SampleDesc.Quality, |
| "Got unexpected SampleDesc.Quality %u.\n", swapchain_desc.SampleDesc.Quality); |
| ok(swapchain_desc.BufferUsage == DXGI_USAGE_RENDER_TARGET_OUTPUT, |
| "Got unexpected BufferUsage %#x.\n", swapchain_desc.BufferUsage); |
| ok(swapchain_desc.BufferCount == 1, |
| "Got unexpected BufferCount %u.\n", swapchain_desc.BufferCount); |
| ok(swapchain_desc.OutputWindow == window, |
| "Got unexpected OutputWindow %p, expected %p.\n", swapchain_desc.OutputWindow, window); |
| ok(swapchain_desc.Windowed, |
| "Got unexpected Windowed %#x.\n", swapchain_desc.Windowed); |
| ok(swapchain_desc.SwapEffect == DXGI_SWAP_EFFECT_DISCARD, |
| "Got unexpected SwapEffect %#x.\n", swapchain_desc.SwapEffect); |
| ok(!swapchain_desc.Flags, |
| "Got unexpected Flags %#x.\n", swapchain_desc.Flags); |
| |
| hr = IDXGISurface_GetDesc(surface, &surface_desc); |
| ok(SUCCEEDED(hr), "Failed to get surface desc, hr %#x.\n", hr); |
| ok(surface_desc.Width == 320, "Got unexpected Width %u.\n", surface_desc.Width); |
| ok(surface_desc.Height == 240, "Got unexpected Height %u.\n", surface_desc.Height); |
| ok(surface_desc.Format == DXGI_FORMAT_R8G8B8A8_UNORM_SRGB, "Got unexpected Format %#x.\n", surface_desc.Format); |
| ok(surface_desc.SampleDesc.Count == 1, "Got unexpected SampleDesc.Count %u.\n", surface_desc.SampleDesc.Count); |
| ok(!surface_desc.SampleDesc.Quality, "Got unexpected SampleDesc.Quality %u.\n", surface_desc.SampleDesc.Quality); |
| |
| ID3D10Texture2D_GetDesc(texture, &texture_desc); |
| ok(texture_desc.Width == 320, "Got unexpected Width %u.\n", texture_desc.Width); |
| ok(texture_desc.Height == 240, "Got unexpected Height %u.\n", texture_desc.Height); |
| ok(texture_desc.MipLevels == 1, "Got unexpected MipLevels %u.\n", texture_desc.MipLevels); |
| ok(texture_desc.ArraySize == 1, "Got unexpected ArraySize %u.\n", texture_desc.ArraySize); |
| ok(texture_desc.Format == DXGI_FORMAT_R8G8B8A8_UNORM_SRGB, "Got unexpected Format %#x.\n", texture_desc.Format); |
| ok(texture_desc.SampleDesc.Count == 1, "Got unexpected SampleDesc.Count %u.\n", texture_desc.SampleDesc.Count); |
| ok(!texture_desc.SampleDesc.Quality, "Got unexpected SampleDesc.Quality %u.\n", texture_desc.SampleDesc.Quality); |
| ok(texture_desc.Usage == D3D10_USAGE_DEFAULT, "Got unexpected Usage %#x.\n", texture_desc.Usage); |
| ok(texture_desc.BindFlags == D3D10_BIND_RENDER_TARGET, "Got unexpected BindFlags %#x.\n", texture_desc.BindFlags); |
| ok(!texture_desc.CPUAccessFlags, "Got unexpected CPUAccessFlags %#x.\n", texture_desc.CPUAccessFlags); |
| ok(!texture_desc.MiscFlags, "Got unexpected MiscFlags %#x.\n", texture_desc.MiscFlags); |
| |
| ID3D10Texture2D_Release(texture); |
| IDXGISurface_Release(surface); |
| |
| hr = IDXGISwapChain_ResizeBuffers(swapchain, 0, 0, 0, DXGI_FORMAT_UNKNOWN, 0); |
| ok(SUCCEEDED(hr), "Failed to resize buffers, hr %#x.\n", hr); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &swapchain_desc); |
| ok(SUCCEEDED(hr), "Failed to get swapchain desc, hr %#x.\n", hr); |
| ok(swapchain_desc.BufferDesc.Width == client_rect.right - client_rect.left, |
| "Got unexpected BufferDesc.Width %u, expected %u.\n", |
| swapchain_desc.BufferDesc.Width, client_rect.right - client_rect.left); |
| ok(swapchain_desc.BufferDesc.Height == client_rect.bottom - client_rect.top, |
| "Got unexpected bufferDesc.Height %u, expected %u.\n", |
| swapchain_desc.BufferDesc.Height, client_rect.bottom - client_rect.top); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Numerator == 60, |
| "Got unexpected BufferDesc.RefreshRate.Numerator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Numerator); |
| ok(swapchain_desc.BufferDesc.RefreshRate.Denominator == 1, |
| "Got unexpected BufferDesc.RefreshRate.Denominator %u.\n", |
| swapchain_desc.BufferDesc.RefreshRate.Denominator); |
| ok(swapchain_desc.BufferDesc.Format == DXGI_FORMAT_R8G8B8A8_UNORM_SRGB, |
| "Got unexpected BufferDesc.Format %#x.\n", swapchain_desc.BufferDesc.Format); |
| ok(swapchain_desc.BufferDesc.ScanlineOrdering == DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED, |
| "Got unexpected BufferDesc.ScanlineOrdering %#x.\n", swapchain_desc.BufferDesc.ScanlineOrdering); |
| ok(swapchain_desc.BufferDesc.Scaling == DXGI_MODE_SCALING_UNSPECIFIED, |
| "Got unexpected BufferDesc.Scaling %#x.\n", swapchain_desc.BufferDesc.Scaling); |
| ok(swapchain_desc.SampleDesc.Count == 1, |
| "Got unexpected SampleDesc.Count %u.\n", swapchain_desc.SampleDesc.Count); |
| ok(!swapchain_desc.SampleDesc.Quality, |
| "Got unexpected SampleDesc.Quality %u.\n", swapchain_desc.SampleDesc.Quality); |
| ok(swapchain_desc.BufferUsage == DXGI_USAGE_RENDER_TARGET_OUTPUT, |
| "Got unexpected BufferUsage %#x.\n", swapchain_desc.BufferUsage); |
| ok(swapchain_desc.BufferCount == 1, |
| "Got unexpected BufferCount %u.\n", swapchain_desc.BufferCount); |
| ok(swapchain_desc.OutputWindow == window, |
| "Got unexpected OutputWindow %p, expected %p.\n", swapchain_desc.OutputWindow, window); |
| ok(swapchain_desc.Windowed, |
| "Got unexpected Windowed %#x.\n", swapchain_desc.Windowed); |
| ok(swapchain_desc.SwapEffect == DXGI_SWAP_EFFECT_DISCARD, |
| "Got unexpected SwapEffect %#x.\n", swapchain_desc.SwapEffect); |
| ok(!swapchain_desc.Flags, |
| "Got unexpected Flags %#x.\n", swapchain_desc.Flags); |
| |
| IDXGISwapChain_Release(swapchain); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| DestroyWindow(window); |
| } |
| |
| static void test_swapchain_parameters(void) |
| { |
| IDXGISwapChain *swapchain; |
| IUnknown *obj; |
| IDXGIAdapter *adapter; |
| IDXGIFactory *factory; |
| IDXGIDevice *device; |
| IDXGIResource *resource; |
| DXGI_SWAP_CHAIN_DESC desc; |
| HRESULT hr; |
| unsigned int i, j; |
| ULONG refcount; |
| DXGI_USAGE usage, expected_usage, broken_usage; |
| HWND window; |
| static const struct |
| { |
| BOOL windowed; |
| UINT buffer_count; |
| DXGI_SWAP_EFFECT swap_effect; |
| HRESULT hr, vista_hr; |
| UINT highest_accessible_buffer; |
| } |
| tests[] = |
| { |
| {TRUE, 0, DXGI_SWAP_EFFECT_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 1, DXGI_SWAP_EFFECT_DISCARD, S_OK, S_OK, 0}, |
| {TRUE, 2, DXGI_SWAP_EFFECT_DISCARD, S_OK, S_OK, 0}, |
| {TRUE, 0, DXGI_SWAP_EFFECT_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 1, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 0}, |
| {TRUE, 2, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 1}, |
| {TRUE, 3, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 2}, |
| {TRUE, 0, 2 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 1, 2 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 2, 2 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 0, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 1, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 2, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, S_OK, DXGI_ERROR_INVALID_CALL, 1}, |
| {TRUE, 3, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, S_OK, DXGI_ERROR_INVALID_CALL, 2}, |
| {TRUE, 0, DXGI_SWAP_EFFECT_FLIP_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 1, DXGI_SWAP_EFFECT_FLIP_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 2, DXGI_SWAP_EFFECT_FLIP_DISCARD, S_OK, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 0, 5 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 1, 5 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 2, 5 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 16, DXGI_SWAP_EFFECT_DISCARD, S_OK, S_OK, 0}, |
| {TRUE, 16, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 15}, |
| {TRUE, 16, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, S_OK, DXGI_ERROR_INVALID_CALL, 15}, |
| {TRUE, 16, DXGI_SWAP_EFFECT_FLIP_DISCARD, S_OK, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 17, DXGI_SWAP_EFFECT_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 17, DXGI_SWAP_EFFECT_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 17, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {TRUE, 17, DXGI_SWAP_EFFECT_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| |
| {FALSE, 0, DXGI_SWAP_EFFECT_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 1, DXGI_SWAP_EFFECT_DISCARD, S_OK, S_OK, 0}, |
| {FALSE, 2, DXGI_SWAP_EFFECT_DISCARD, S_OK, S_OK, 0}, |
| {FALSE, 0, DXGI_SWAP_EFFECT_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 1, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 0}, |
| {FALSE, 2, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 1}, |
| {FALSE, 3, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 2}, |
| {FALSE, 0, 2 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 1, 2 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 2, 2 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 0, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 1, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 2, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, S_OK, DXGI_ERROR_INVALID_CALL, 1}, |
| {FALSE, 3, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, S_OK, DXGI_ERROR_INVALID_CALL, 2}, |
| {FALSE, 0, DXGI_SWAP_EFFECT_FLIP_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 1, DXGI_SWAP_EFFECT_FLIP_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 2, DXGI_SWAP_EFFECT_FLIP_DISCARD, S_OK, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 0, 5 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 1, 5 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 2, 5 /* undefined */, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 16, DXGI_SWAP_EFFECT_DISCARD, S_OK, S_OK, 0}, |
| {FALSE, 16, DXGI_SWAP_EFFECT_SEQUENTIAL, S_OK, S_OK, 15}, |
| {FALSE, 16, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, S_OK, DXGI_ERROR_INVALID_CALL, 15}, |
| /* The following test fails on Nvidia with E_OUTOFMEMORY and leaks device references in the |
| * process. Disable it for now. |
| {FALSE, 16, DXGI_SWAP_EFFECT_FLIP_DISCARD, S_OK, DXGI_ERROR_INVALID_CALL, 0}, |
| */ |
| {FALSE, 17, DXGI_SWAP_EFFECT_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 17, DXGI_SWAP_EFFECT_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 17, DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| {FALSE, 17, DXGI_SWAP_EFFECT_FLIP_DISCARD, DXGI_ERROR_INVALID_CALL, DXGI_ERROR_INVALID_CALL, 0}, |
| }; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device, skipping tests.\n"); |
| return; |
| } |
| window = CreateWindowA("static", "dxgi_test", WS_OVERLAPPEDWINDOW | WS_VISIBLE, |
| 0, 0, 640, 480, 0, 0, 0, 0); |
| |
| hr = IDXGIDevice_QueryInterface(device, &IID_IUnknown, (void **)&obj); |
| ok(SUCCEEDED(hr), "IDXGIDevice does not implement IUnknown\n"); |
| |
| hr = IDXGIDevice_GetAdapter(device, &adapter); |
| ok(SUCCEEDED(hr), "GetAdapter failed, hr %#x.\n", hr); |
| |
| hr = IDXGIAdapter_GetParent(adapter, &IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "GetParent failed, hr %#x.\n", hr); |
| |
| for (i = 0; i < sizeof(tests) / sizeof(*tests); ++i) |
| { |
| memset(&desc, 0, sizeof(desc)); |
| desc.BufferDesc.Width = registry_mode.dmPelsWidth; |
| desc.BufferDesc.Height = registry_mode.dmPelsHeight; |
| desc.BufferDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM; |
| desc.SampleDesc.Count = 1; |
| desc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; |
| desc.OutputWindow = window; |
| |
| desc.Windowed = tests[i].windowed; |
| desc.BufferCount = tests[i].buffer_count; |
| desc.SwapEffect = tests[i].swap_effect; |
| |
| hr = IDXGIFactory_CreateSwapChain(factory, obj, &desc, &swapchain); |
| ok(hr == tests[i].hr || broken(hr == tests[i].vista_hr) |
| || (SUCCEEDED(tests[i].hr) && hr == DXGI_STATUS_OCCLUDED), |
| "Got unexpected hr %#x, test %u.\n", hr, i); |
| if (FAILED(hr)) |
| continue; |
| |
| hr = IDXGISwapChain_GetBuffer(swapchain, 0, &IID_IDXGIResource, (void **)&resource); |
| todo_wine ok(SUCCEEDED(hr), "GetBuffer(0) failed, hr %#x, test %u.\n", hr, i); |
| if (FAILED(hr)) |
| { |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| |
| IDXGISwapChain_Release(swapchain); |
| continue; |
| } |
| |
| expected_usage = DXGI_USAGE_RENDER_TARGET_OUTPUT | DXGI_USAGE_BACK_BUFFER; |
| if (tests[i].swap_effect == DXGI_SWAP_EFFECT_DISCARD) |
| expected_usage |= DXGI_USAGE_DISCARD_ON_PRESENT; |
| hr = IDXGIResource_GetUsage(resource, &usage); |
| ok(SUCCEEDED(hr), "Failed to get resource usage, hr %#x, test %u.\n", hr, i); |
| ok(usage == expected_usage, "Got usage %x, expected %x, test %u.\n", usage, expected_usage, i); |
| |
| IDXGIResource_Release(resource); |
| |
| hr = IDXGISwapChain_GetDesc(swapchain, &desc); |
| ok(SUCCEEDED(hr), "Failed to get swapchain desc, hr %#x.\n", hr); |
| |
| for (j = 1; j <= tests[i].highest_accessible_buffer; j++) |
| { |
| hr = IDXGISwapChain_GetBuffer(swapchain, j, &IID_IDXGIResource, (void **)&resource); |
| ok(SUCCEEDED(hr), "GetBuffer(%u) failed, hr %#x, test %u.\n", hr, i, j); |
| |
| /* Buffers > 0 are supposed to be read only. This is the case except that in |
| * fullscreen mode on Windows <= 8 the last backbuffer (BufferCount - 1) is |
| * writable. This is not the case if an unsupported refresh rate is passed |
| * for some reason, probably because the invalid refresh rate triggers a |
| * kinda-sorta windowed mode. |
| * |
| * On Windows 10 all buffers > 0 are read-only. Mark the earlier behavior |
| * broken. |
| * |
| * This last buffer acts as a shadow frontbuffer. Writing to it doesn't show |
| * the draw on the screen right away (Aero on or off doesn't matter), but |
| * Present with DXGI_PRESENT_DO_NOT_SEQUENCE will show the modifications. |
| * |
| * Note that if the application doesn't have focused creating a fullscreen |
| * swapchain returns DXGI_STATUS_OCCLUDED and we get a windowed swapchain, |
| * so use the Windowed property of the swapchain that was actually created. */ |
| expected_usage = DXGI_USAGE_RENDER_TARGET_OUTPUT | DXGI_USAGE_BACK_BUFFER | DXGI_USAGE_READ_ONLY; |
| broken_usage = DXGI_USAGE_RENDER_TARGET_OUTPUT | DXGI_USAGE_BACK_BUFFER; |
| |
| if (desc.Windowed || j < tests[i].highest_accessible_buffer) |
| broken_usage |= DXGI_USAGE_READ_ONLY; |
| |
| hr = IDXGIResource_GetUsage(resource, &usage); |
| ok(SUCCEEDED(hr), "Failed to get resource usage, hr %#x, test %u, buffer %u.\n", hr, i, j); |
| ok(usage == expected_usage || broken(usage == broken_usage), |
| "Got usage %x, expected %x, test %u, buffer %u.\n", |
| usage, expected_usage, i, j); |
| |
| IDXGIResource_Release(resource); |
| } |
| hr = IDXGISwapChain_GetBuffer(swapchain, j, &IID_IDXGIResource, (void **)&resource); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "GetBuffer(%u) returned unexpected hr %#x, test %u.\n", j, hr, i); |
| |
| hr = IDXGISwapChain_SetFullscreenState(swapchain, FALSE, NULL); |
| ok(SUCCEEDED(hr), "SetFullscreenState failed, hr %#x.\n", hr); |
| |
| IDXGISwapChain_Release(swapchain); |
| } |
| |
| IDXGIFactory_Release(factory); |
| IDXGIAdapter_Release(adapter); |
| IUnknown_Release(obj); |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| DestroyWindow(window); |
| } |
| |
| static void test_maximum_frame_latency(void) |
| { |
| IDXGIDevice1 *device1; |
| IDXGIDevice *device; |
| UINT max_latency; |
| ULONG refcount; |
| HRESULT hr; |
| |
| if (!(device = create_device(0))) |
| { |
| skip("Failed to create device.\n"); |
| return; |
| } |
| |
| if (SUCCEEDED(IDXGIDevice_QueryInterface(device, &IID_IDXGIDevice1, (void **)&device1))) |
| { |
| hr = IDXGIDevice1_GetMaximumFrameLatency(device1, &max_latency); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| ok(max_latency == DEFAULT_FRAME_LATENCY, "Got unexpected maximum frame latency %u.\n", max_latency); |
| |
| hr = IDXGIDevice1_SetMaximumFrameLatency(device1, MAX_FRAME_LATENCY); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice1_GetMaximumFrameLatency(device1, &max_latency); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| todo_wine ok(max_latency == MAX_FRAME_LATENCY, "Got unexpected maximum frame latency %u.\n", max_latency); |
| |
| hr = IDXGIDevice1_SetMaximumFrameLatency(device1, MAX_FRAME_LATENCY + 1); |
| ok(hr == DXGI_ERROR_INVALID_CALL, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice1_GetMaximumFrameLatency(device1, &max_latency); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| todo_wine ok(max_latency == MAX_FRAME_LATENCY, "Got unexpected maximum frame latency %u.\n", max_latency); |
| |
| hr = IDXGIDevice1_SetMaximumFrameLatency(device1, 0); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| hr = IDXGIDevice1_GetMaximumFrameLatency(device1, &max_latency); |
| todo_wine ok(hr == S_OK, "Got unexpected hr %#x.\n", hr); |
| /* 0 does not reset to the default frame latency on all Windows versions. */ |
| ok(max_latency == DEFAULT_FRAME_LATENCY || broken(!max_latency), |
| "Got unexpected maximum frame latency %u.\n", max_latency); |
| |
| IDXGIDevice1_Release(device1); |
| } |
| else |
| { |
| win_skip("IDXGIDevice1 is not implemented.\n"); |
| } |
| |
| refcount = IDXGIDevice_Release(device); |
| ok(!refcount, "Device has %u references left.\n", refcount); |
| } |
| |
| static void test_output_desc(void) |
| { |
| IDXGIAdapter *adapter, *adapter2; |
| IDXGIOutput *output, *output2; |
| DXGI_OUTPUT_DESC desc; |
| IDXGIFactory *factory; |
| unsigned int i, j; |
| ULONG refcount; |
| HRESULT hr; |
| |
| hr = CreateDXGIFactory(&IID_IDXGIFactory, (void **)&factory); |
| ok(SUCCEEDED(hr), "Failed to create DXGI factory, hr %#x.\n", hr); |
| |
| for (i = 0; ; ++i) |
| { |
| hr = IDXGIFactory_EnumAdapters(factory, i, &adapter); |
| if (hr == DXGI_ERROR_NOT_FOUND) |
| break; |
| ok(SUCCEEDED(hr), "Failed to enumerate adapter %u, hr %#x.\n", i, hr); |
| |
| hr = IDXGIFactory_EnumAdapters(factory, i, &adapter2); |
| ok(SUCCEEDED(hr), "Failed to enumerate adapter %u, hr %#x.\n", i, hr); |
| ok(adapter != adapter2, "Expected to get new instance of IDXGIAdapter, %p == %p.\n", adapter, adapter2); |
| refcount = get_refcount((IUnknown *)adapter); |
| ok(refcount == 1, "Get unexpected refcount %u for adapter %u.\n", refcount, i); |
| IDXGIAdapter_Release(adapter2); |
| |
| refcount = get_refcount((IUnknown *)factory); |
| ok(refcount == 2, "Get unexpected refcount %u.\n", refcount); |
| refcount = get_refcount((IUnknown *)adapter); |
| ok(refcount == 1, "Get unexpected refcount %u for adapter %u.\n", refcount, i); |
| |
| for (j = 0; ; ++j) |
| { |
| MONITORINFOEXW monitor_info; |
| BOOL ret; |
| |
| hr = IDXGIAdapter_EnumOutputs(adapter, j, &output); |
| if (hr == DXGI_ERROR_NOT_FOUND) |
| break; |
| ok(SUCCEEDED(hr), "Failed to enumerate output %u on adapter %u, hr %#x.\n", j, i, hr); |
| |
| hr = IDXGIAdapter_EnumOutputs(adapter, j, &output2); |
| ok(SUCCEEDED(hr), "Failed to enumerate output %u on adapter %u, hr %#x.\n", j, i, hr); |
| ok(output != output2, "Expected to get new instance of IDXGIOutput, %p == %p.\n", output, output2); |
| refcount = get_refcount((IUnknown *)output); |
| ok(refcount == 1, "Get unexpected refcount %u for output %u, adapter %u.\n", refcount, j, i); |
| IDXGIOutput_Release(output2); |
| |
| refcount = get_refcount((IUnknown *)factory); |
| ok(refcount == 2, "Get unexpected refcount %u.\n", refcount); |
| refcount = get_refcount((IUnknown *)adapter); |
| ok(refcount == 2, "Get unexpected refcount %u for adapter %u.\n", refcount, i); |
| refcount = get_refcount((IUnknown *)output); |
| ok(refcount == 1, "Get unexpected refcount %u for output %u, adapter %u.\n", refcount, j, i); |
| |
| hr = IDXGIOutput_GetDesc(output, NULL); |
| ok(hr == E_INVALIDARG, "Got unexpected hr %#x for output %u on adapter %u.\n", hr, j, i); |
| hr = IDXGIOutput_GetDesc(output, &desc); |
| ok(SUCCEEDED(hr), "Failed to get desc for output %u on adapter %u, hr %#x.\n", j, i, hr); |
| |
| monitor_info.cbSize = sizeof(monitor_info); |
| ret = GetMonitorInfoW(desc.Monitor, (MONITORINFO *)&monitor_info); |
| ok(ret, "Failed to get monitor info.\n"); |
| ok(!lstrcmpW(desc.DeviceName, monitor_info.szDevice), "Got unexpected device name %s, expected %s.\n", |
| wine_dbgstr_w(desc.DeviceName), wine_dbgstr_w(monitor_info.szDevice)); |
| ok(EqualRect(&desc.DesktopCoordinates, &monitor_info.rcMonitor), |
| "Got unexpected desktop coordinates %s, expected %s.\n", |
| wine_dbgstr_rect(&desc.DesktopCoordinates), |
| wine_dbgstr_rect(&monitor_info.rcMonitor)); |
| |
| IDXGIOutput_Release(output); |
| refcount = get_refcount((IUnknown *)adapter); |
| ok(refcount == 1, "Get unexpected refcount %u for adapter %u.\n", refcount, i); |
| } |
| |
| IDXGIAdapter_Release(adapter); |
| refcount = get_refcount((IUnknown *)factory); |
| ok(refcount == 1, "Get unexpected refcount %u.\n", refcount); |
| } |
| |
| refcount = IDXGIFactory_Release(factory); |
| ok(!refcount, "IDXGIFactory has %u references left.\n", refcount); |
| } |
| |
| START_TEST(device) |
| { |
| pCreateDXGIFactory1 = (void *)GetProcAddress(GetModuleHandleA("dxgi.dll"), "CreateDXGIFactory1"); |
| |
| registry_mode.dmSize = sizeof(registry_mode); |
| ok(EnumDisplaySettingsW(NULL, ENUM_REGISTRY_SETTINGS, ®istry_mode), "Failed to get display mode.\n"); |
| |
| test_adapter_desc(); |
| test_check_interface_support(); |
| test_create_surface(); |
| test_parents(); |
| test_output(); |
| test_find_closest_matching_mode(); |
| test_create_swapchain(); |
| test_get_containing_output(); |
| test_set_fullscreen(); |
| test_default_fullscreen_target_output(); |
| test_resize_target(); |
| test_inexact_modes(); |
| test_create_factory(); |
| test_private_data(); |
| test_swapchain_resize(); |
| test_swapchain_parameters(); |
| test_maximum_frame_latency(); |
| test_output_desc(); |
| } |