wined3d: Take scanline ordering into account in the mode setting code.
[wine] / dlls / dxgi / output.c
1 /*
2  * Copyright 2009 Henri Verbeet for CodeWeavers
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  */
18
19 #include "config.h"
20 #include "wine/port.h"
21
22 #include "dxgi_private.h"
23
24 WINE_DEFAULT_DEBUG_CHANNEL(dxgi);
25
26 static inline struct dxgi_output *impl_from_IDXGIOutput(IDXGIOutput *iface)
27 {
28     return CONTAINING_RECORD(iface, struct dxgi_output, IDXGIOutput_iface);
29 }
30
31 /* IUnknown methods */
32
33 static HRESULT STDMETHODCALLTYPE dxgi_output_QueryInterface(IDXGIOutput *iface, REFIID riid, void **object)
34 {
35     TRACE("iface %p, riid %s, object %p.\n", iface, debugstr_guid(riid), object);
36
37     if (IsEqualGUID(riid, &IID_IDXGIOutput)
38             || IsEqualGUID(riid, &IID_IDXGIObject)
39             || IsEqualGUID(riid, &IID_IUnknown))
40     {
41         IUnknown_AddRef(iface);
42         *object = iface;
43         return S_OK;
44     }
45
46     WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(riid));
47
48     *object = NULL;
49     return E_NOINTERFACE;
50 }
51
52 static ULONG STDMETHODCALLTYPE dxgi_output_AddRef(IDXGIOutput *iface)
53 {
54     struct dxgi_output *This = impl_from_IDXGIOutput(iface);
55     ULONG refcount = InterlockedIncrement(&This->refcount);
56
57     TRACE("%p increasing refcount to %u.\n", This, refcount);
58
59     return refcount;
60 }
61
62 static ULONG STDMETHODCALLTYPE dxgi_output_Release(IDXGIOutput *iface)
63 {
64     struct dxgi_output *This = impl_from_IDXGIOutput(iface);
65     ULONG refcount = InterlockedDecrement(&This->refcount);
66
67     TRACE("%p decreasing refcount to %u.\n", This, refcount);
68
69     if (!refcount)
70     {
71         HeapFree(GetProcessHeap(), 0, This);
72     }
73
74     return refcount;
75 }
76
77 /* IDXGIObject methods */
78
79 static HRESULT STDMETHODCALLTYPE dxgi_output_SetPrivateData(IDXGIOutput *iface,
80         REFGUID guid, UINT data_size, const void *data)
81 {
82     FIXME("iface %p, guid %s, data_size %u, data %p stub!\n", iface, debugstr_guid(guid), data_size, data);
83
84     return E_NOTIMPL;
85 }
86
87 static HRESULT STDMETHODCALLTYPE dxgi_output_SetPrivateDataInterface(IDXGIOutput *iface,
88         REFGUID guid, const IUnknown *object)
89 {
90     FIXME("iface %p, guid %s, object %p stub!\n", iface, debugstr_guid(guid), object);
91
92     return E_NOTIMPL;
93 }
94
95 static HRESULT STDMETHODCALLTYPE dxgi_output_GetPrivateData(IDXGIOutput *iface,
96         REFGUID guid, UINT *data_size, void *data)
97 {
98     FIXME("iface %p, guid %s, data_size %p, data %p stub!\n", iface, debugstr_guid(guid), data_size, data);
99
100     return E_NOTIMPL;
101 }
102
103 static HRESULT STDMETHODCALLTYPE dxgi_output_GetParent(IDXGIOutput *iface,
104         REFIID riid, void **parent)
105 {
106     struct dxgi_output *This = impl_from_IDXGIOutput(iface);
107
108     TRACE("iface %p, riid %s, parent %p.\n", iface, debugstr_guid(riid), parent);
109
110     return IDXGIAdapter_QueryInterface((IDXGIAdapter *)This->adapter, riid, parent);
111 }
112
113 /* IDXGIOutput methods */
114
115 static HRESULT STDMETHODCALLTYPE dxgi_output_GetDesc(IDXGIOutput *iface, DXGI_OUTPUT_DESC *desc)
116 {
117     FIXME("iface %p, desc %p stub!\n", iface, desc);
118
119     return E_NOTIMPL;
120 }
121
122 static HRESULT STDMETHODCALLTYPE dxgi_output_GetDisplayModeList(IDXGIOutput *iface,
123         DXGI_FORMAT format, UINT flags, UINT *mode_count, DXGI_MODE_DESC *desc)
124 {
125     struct dxgi_output *This = impl_from_IDXGIOutput(iface);
126     enum wined3d_format_id wined3d_format;
127     struct wined3d *wined3d;
128     UINT i;
129     UINT max_count;
130
131     FIXME("iface %p, format %s, flags %#x, mode_count %p, desc %p partial stub!\n",
132             iface, debug_dxgi_format(format), flags, mode_count, desc);
133
134     if (!mode_count)
135     {
136         return S_OK;
137     }
138
139     if (format == DXGI_FORMAT_UNKNOWN)
140     {
141         *mode_count = 0;
142         return S_OK;
143     }
144
145     wined3d = IWineDXGIFactory_get_wined3d(This->adapter->parent);
146     wined3d_format = wined3dformat_from_dxgi_format(format);
147
148     EnterCriticalSection(&dxgi_cs);
149     max_count = wined3d_get_adapter_mode_count(wined3d, This->adapter->ordinal, wined3d_format);
150
151     if (!desc)
152     {
153         wined3d_decref(wined3d);
154         LeaveCriticalSection(&dxgi_cs);
155         *mode_count = max_count;
156         return S_OK;
157     }
158
159     *mode_count = min(*mode_count,max_count);
160
161     for (i = 0; i < *mode_count; ++i)
162     {
163         struct wined3d_display_mode mode;
164         HRESULT hr;
165
166         hr = wined3d_enum_adapter_modes(wined3d, This->adapter->ordinal, wined3d_format, i, &mode);
167         if (FAILED(hr))
168         {
169             WARN("EnumAdapterModes failed, hr %#x.\n", hr);
170             wined3d_decref(wined3d);
171             LeaveCriticalSection(&dxgi_cs);
172             return hr;
173         }
174
175         desc[i].Width = mode.width;
176         desc[i].Height = mode.height;
177         desc[i].RefreshRate.Numerator = mode.refresh_rate;
178         desc[i].RefreshRate.Denominator = 1;
179         desc[i].Format = format;
180         desc[i].ScanlineOrdering = mode.scanline_ordering;
181         desc[i].Scaling = DXGI_MODE_SCALING_UNSPECIFIED; /* FIXME */
182     }
183     wined3d_decref(wined3d);
184     LeaveCriticalSection(&dxgi_cs);
185
186     return S_OK;
187 }
188
189 static HRESULT STDMETHODCALLTYPE dxgi_output_FindClosestMatchingMode(IDXGIOutput *iface,
190         const DXGI_MODE_DESC *mode, DXGI_MODE_DESC *closest_match, IUnknown *device)
191 {
192     FIXME("iface %p, mode %p, closest_match %p, device %p stub!\n", iface, mode, closest_match, device);
193
194     return E_NOTIMPL;
195 }
196
197 static HRESULT STDMETHODCALLTYPE dxgi_output_WaitForVBlank(IDXGIOutput *iface)
198 {
199     FIXME("iface %p stub!\n", iface);
200
201     return E_NOTIMPL;
202 }
203
204 static HRESULT STDMETHODCALLTYPE dxgi_output_TakeOwnership(IDXGIOutput *iface, IUnknown *device, BOOL exclusive)
205 {
206     FIXME("iface %p, device %p, exclusive %d stub!\n", iface, device, exclusive);
207
208     return E_NOTIMPL;
209 }
210
211 static void STDMETHODCALLTYPE dxgi_output_ReleaseOwnership(IDXGIOutput *iface)
212 {
213     FIXME("iface %p stub!\n", iface);
214 }
215
216 static HRESULT STDMETHODCALLTYPE dxgi_output_GetGammaControlCapabilities(IDXGIOutput *iface,
217         DXGI_GAMMA_CONTROL_CAPABILITIES *gamma_caps)
218 {
219     FIXME("iface %p, gamma_caps %p stub!\n", iface, gamma_caps);
220
221     return E_NOTIMPL;
222 }
223
224 static HRESULT STDMETHODCALLTYPE dxgi_output_SetGammaControl(IDXGIOutput *iface,
225         const DXGI_GAMMA_CONTROL *gamma_control)
226 {
227     FIXME("iface %p, gamma_control %p stub!\n", iface, gamma_control);
228
229     return E_NOTIMPL;
230 }
231
232 static HRESULT STDMETHODCALLTYPE dxgi_output_GetGammaControl(IDXGIOutput *iface, DXGI_GAMMA_CONTROL *gamma_control)
233 {
234     FIXME("iface %p, gamma_control %p stub!\n", iface, gamma_control);
235
236     return E_NOTIMPL;
237 }
238
239 static HRESULT STDMETHODCALLTYPE dxgi_output_SetDisplaySurface(IDXGIOutput *iface, IDXGISurface *surface)
240 {
241     FIXME("iface %p, surface %p stub!\n", iface, surface);
242
243     return E_NOTIMPL;
244 }
245
246 static HRESULT STDMETHODCALLTYPE dxgi_output_GetDisplaySurfaceData(IDXGIOutput *iface, IDXGISurface *surface)
247 {
248     FIXME("iface %p, surface %p stub!\n", iface, surface);
249
250     return E_NOTIMPL;
251 }
252
253 static HRESULT STDMETHODCALLTYPE dxgi_output_GetFrameStatistics(IDXGIOutput *iface, DXGI_FRAME_STATISTICS *stats)
254 {
255     FIXME("iface %p, stats %p stub!\n", iface, stats);
256
257     return E_NOTIMPL;
258 }
259
260 static const struct IDXGIOutputVtbl dxgi_output_vtbl =
261 {
262     dxgi_output_QueryInterface,
263     dxgi_output_AddRef,
264     dxgi_output_Release,
265     /* IDXGIObject methods */
266     dxgi_output_SetPrivateData,
267     dxgi_output_SetPrivateDataInterface,
268     dxgi_output_GetPrivateData,
269     dxgi_output_GetParent,
270     /* IDXGIOutput methods */
271     dxgi_output_GetDesc,
272     dxgi_output_GetDisplayModeList,
273     dxgi_output_FindClosestMatchingMode,
274     dxgi_output_WaitForVBlank,
275     dxgi_output_TakeOwnership,
276     dxgi_output_ReleaseOwnership,
277     dxgi_output_GetGammaControlCapabilities,
278     dxgi_output_SetGammaControl,
279     dxgi_output_GetGammaControl,
280     dxgi_output_SetDisplaySurface,
281     dxgi_output_GetDisplaySurfaceData,
282     dxgi_output_GetFrameStatistics,
283 };
284
285 void dxgi_output_init(struct dxgi_output *output, struct dxgi_adapter *adapter)
286 {
287     output->IDXGIOutput_iface.lpVtbl = &dxgi_output_vtbl;
288     output->refcount = 1;
289     output->adapter = adapter;
290 }