blob: ec1b2f093f42a0a14c91027ec36deb9cd5369c70 [file] [log] [blame]
Henri Verbeet9e376212008-10-21 15:06:58 +02001/*
2 * Copyright 2008 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
20#include "config.h"
21#include "wine/port.h"
22
Henri Verbeet479e3672008-11-17 12:16:04 +010023#define DXGI_INIT_GUID
Henri Verbeet9e376212008-10-21 15:06:58 +020024#include "dxgi_private.h"
25
26WINE_DEFAULT_DEBUG_CHANNEL(dxgi);
27
Henri Verbeet800fedc2008-11-14 13:57:06 +010028static CRITICAL_SECTION_DEBUG dxgi_cs_debug =
29{
30 0, 0, &dxgi_cs,
31 {&dxgi_cs_debug.ProcessLocksList,
32 &dxgi_cs_debug.ProcessLocksList},
33 0, 0, {(DWORD_PTR)(__FILE__ ": dxgi_cs")}
34};
Henri Verbeet479e3672008-11-17 12:16:04 +010035CRITICAL_SECTION dxgi_cs = {&dxgi_cs_debug, -1, 0, 0, 0, 0};
Henri Verbeet800fedc2008-11-14 13:57:06 +010036
37struct dxgi_main
38{
Henri Verbeetddbd4ab2008-11-14 13:57:06 +010039 HMODULE d3d10core;
Henri Verbeet800fedc2008-11-14 13:57:06 +010040 struct dxgi_device_layer *device_layers;
41 UINT layer_count;
42 LONG refcount;
43};
44static struct dxgi_main dxgi_main;
45
46static void dxgi_main_cleanup(void)
47{
48 EnterCriticalSection(&dxgi_cs);
49
50 HeapFree(GetProcessHeap(), 0, dxgi_main.device_layers);
51 dxgi_main.device_layers = NULL;
52 dxgi_main.layer_count = 0;
53
Henri Verbeetddbd4ab2008-11-14 13:57:06 +010054 FreeLibrary(dxgi_main.d3d10core);
55 dxgi_main.d3d10core = NULL;
56
Henri Verbeet800fedc2008-11-14 13:57:06 +010057 LeaveCriticalSection(&dxgi_cs);
58}
59
Henri Verbeet9e376212008-10-21 15:06:58 +020060BOOL WINAPI DllMain(HINSTANCE hInstDLL, DWORD fdwReason, LPVOID lpv)
61{
62 TRACE("fdwReason %u\n", fdwReason);
63
64 switch(fdwReason)
65 {
66 case DLL_PROCESS_ATTACH:
67 DisableThreadLibraryCalls(hInstDLL);
Henri Verbeet800fedc2008-11-14 13:57:06 +010068 ++dxgi_main.refcount;
69 break;
70
71 case DLL_PROCESS_DETACH:
72 if (!--dxgi_main.refcount) dxgi_main_cleanup();
Henri Verbeet9e376212008-10-21 15:06:58 +020073 break;
74 }
75
76 return TRUE;
77}
Henri Verbeet9fbd6762008-10-21 15:06:58 +020078
79HRESULT WINAPI CreateDXGIFactory(REFIID riid, void **factory)
80{
81 struct dxgi_factory *object;
82 HRESULT hr;
83
Henri Verbeet479e3672008-11-17 12:16:04 +010084 TRACE("riid %s, factory %p\n", debugstr_guid(riid), factory);
Henri Verbeet9fbd6762008-10-21 15:06:58 +020085
86 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(*object));
87 if (!object)
88 {
89 ERR("Failed to allocate DXGI factory object memory\n");
Henri Verbeet808bbeb2008-11-18 09:27:58 +010090 *factory = NULL;
Henri Verbeet9fbd6762008-10-21 15:06:58 +020091 return E_OUTOFMEMORY;
92 }
93
Henri Verbeet9e886422009-12-29 17:10:23 +010094 hr = dxgi_factory_init(object);
95 if (FAILED(hr))
Huw Davies23922142008-12-03 13:33:32 +000096 {
Henri Verbeet9e886422009-12-29 17:10:23 +010097 WARN("Failed to initialize swapchain, hr %#x.\n", hr);
98 HeapFree(GetProcessHeap(), 0, object);
99 *factory = NULL;
100 return hr;
Henri Verbeet808bbeb2008-11-18 09:27:58 +0100101 }
Henri Verbeet479e3672008-11-17 12:16:04 +0100102
Henri Verbeet9fbd6762008-10-21 15:06:58 +0200103 TRACE("Created IDXGIFactory %p\n", object);
104
105 hr = IDXGIFactory_QueryInterface((IDXGIFactory *)object, riid, factory);
106 IDXGIFactory_Release((IDXGIFactory *)object);
107
108 return hr;
109}
Henri Verbeet56692e92008-11-12 15:43:46 +0100110
Henri Verbeetddbd4ab2008-11-14 13:57:06 +0100111static BOOL get_layer(enum dxgi_device_layer_id id, struct dxgi_device_layer *layer)
112{
113 UINT i;
114
115 EnterCriticalSection(&dxgi_cs);
116
117 for (i = 0; i < dxgi_main.layer_count; ++i)
118 {
119 if (dxgi_main.device_layers[i].id == id)
120 {
121 *layer = dxgi_main.device_layers[i];
122 LeaveCriticalSection(&dxgi_cs);
123 return TRUE;
124 }
125 }
126
127 LeaveCriticalSection(&dxgi_cs);
128 return FALSE;
129}
130
131static HRESULT register_d3d10core_layers(HMODULE d3d10core)
132{
133 EnterCriticalSection(&dxgi_cs);
134
135 if (!dxgi_main.d3d10core)
136 {
137 HRESULT hr;
138 HRESULT (WINAPI *d3d10core_register_layers)(void);
139 HMODULE mod;
140 BOOL ret;
141
142 ret = GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS, (LPCSTR)d3d10core, &mod);
143 if (!ret)
144 {
145 LeaveCriticalSection(&dxgi_cs);
146 return E_FAIL;
147 }
148
149 d3d10core_register_layers = (HRESULT (WINAPI *)(void))GetProcAddress(mod, "D3D10CoreRegisterLayers");
150 hr = d3d10core_register_layers();
151 if (FAILED(hr))
152 {
153 ERR("Failed to register d3d10core layers, returning %#x\n", hr);
154 LeaveCriticalSection(&dxgi_cs);
155 return hr;
156 }
157
158 dxgi_main.d3d10core = mod;
159 }
160
161 LeaveCriticalSection(&dxgi_cs);
162
163 return S_OK;
164}
165
166HRESULT WINAPI DXGID3D10CreateDevice(HMODULE d3d10core, IDXGIFactory *factory, IDXGIAdapter *adapter,
Henri Verbeetefbd38a2009-12-07 20:20:03 +0100167 UINT flags, void *unknown0, void **device)
Henri Verbeetddbd4ab2008-11-14 13:57:06 +0100168{
169 struct layer_get_size_args get_size_args;
170 struct dxgi_device *dxgi_device;
171 struct dxgi_device_layer d3d10_layer;
Henri Verbeetddbd4ab2008-11-14 13:57:06 +0100172 UINT device_size;
173 DWORD count;
174 HRESULT hr;
175
Henri Verbeetefbd38a2009-12-07 20:20:03 +0100176 TRACE("d3d10core %p, factory %p, adapter %p, flags %#x, unknown0 %p, device %p.\n",
Henri Verbeetddbd4ab2008-11-14 13:57:06 +0100177 d3d10core, factory, adapter, flags, unknown0, device);
178
179 hr = register_d3d10core_layers(d3d10core);
180 if (FAILED(hr))
181 {
182 ERR("Failed to register d3d10core layers, returning %#x\n", hr);
183 return hr;
184 }
185
186 if (!get_layer(DXGI_DEVICE_LAYER_D3D10_DEVICE, &d3d10_layer))
187 {
188 ERR("Failed to get D3D10 device layer\n");
189 return E_FAIL;
190 }
191
192 count = 0;
193 hr = d3d10_layer.init(d3d10_layer.id, &count, NULL);
194 if (FAILED(hr))
195 {
196 WARN("Failed to initialize D3D10 device layer\n");
197 return E_FAIL;
198 }
199
200 get_size_args.unknown0 = 0;
201 get_size_args.unknown1 = 0;
202 get_size_args.unknown2 = NULL;
203 get_size_args.unknown3 = NULL;
204 get_size_args.adapter = adapter;
205 get_size_args.interface_major = 10;
206 get_size_args.interface_minor = 1;
207 get_size_args.version_build = 4;
208 get_size_args.version_revision = 6000;
209
210 device_size = d3d10_layer.get_size(d3d10_layer.id, &get_size_args, 0);
211 device_size += sizeof(*dxgi_device);
212
213 dxgi_device = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, device_size);
214 if (!dxgi_device)
215 {
216 ERR("Failed to allocate device memory\n");
217 return E_OUTOFMEMORY;
218 }
219
Henri Verbeet15b6bc62009-11-19 11:41:25 +0100220 hr = dxgi_device_init(dxgi_device, &d3d10_layer, factory, adapter);
Henri Verbeeta9662932009-01-16 10:14:24 +0100221 if (FAILED(hr))
222 {
Henri Verbeet15b6bc62009-11-19 11:41:25 +0100223 WARN("Failed to initialize device, hr %#x.\n", hr);
224 HeapFree(GetProcessHeap(), 0, dxgi_device);
225 *device = NULL;
226 return hr;
Henri Verbeeta9662932009-01-16 10:14:24 +0100227 }
228
Henri Verbeet15b6bc62009-11-19 11:41:25 +0100229 TRACE("Created device %p.\n", dxgi_device);
Michael Stefaniucf93b29b2009-01-11 00:15:40 +0100230 *device = dxgi_device;
Henri Verbeetddbd4ab2008-11-14 13:57:06 +0100231
Henri Verbeet15b6bc62009-11-19 11:41:25 +0100232 return S_OK;
Henri Verbeetddbd4ab2008-11-14 13:57:06 +0100233}
234
Henri Verbeet56692e92008-11-12 15:43:46 +0100235HRESULT WINAPI DXGID3D10RegisterLayers(const struct dxgi_device_layer *layers, UINT layer_count)
236{
Henri Verbeet800fedc2008-11-14 13:57:06 +0100237 UINT i;
238 struct dxgi_device_layer *new_layers;
Henri Verbeet56692e92008-11-12 15:43:46 +0100239
Henri Verbeet800fedc2008-11-14 13:57:06 +0100240 TRACE("layers %p, layer_count %u\n", layers, layer_count);
241
242 EnterCriticalSection(&dxgi_cs);
243
244 if (!dxgi_main.layer_count)
245 new_layers = HeapAlloc(GetProcessHeap(), 0, layer_count * sizeof(*new_layers));
246 else
247 new_layers = HeapReAlloc(GetProcessHeap(), 0, dxgi_main.device_layers,
248 (dxgi_main.layer_count + layer_count) * sizeof(*new_layers));
249
250 if (!new_layers)
251 {
252 LeaveCriticalSection(&dxgi_cs);
253 ERR("Failed to allocate layer memory\n");
254 return E_OUTOFMEMORY;
255 }
256
257 for (i = 0; i < layer_count; ++i)
258 {
259 const struct dxgi_device_layer *layer = &layers[i];
260
261 TRACE("layer %d: id %#x, init %p, get_size %p, create %p\n",
262 i, layer->id, layer->init, layer->get_size, layer->create);
263
264 new_layers[dxgi_main.layer_count + i] = *layer;
265 }
266
267 dxgi_main.device_layers = new_layers;
268 dxgi_main.layer_count += layer_count;
269
270 LeaveCriticalSection(&dxgi_cs);
271
272 return S_OK;
Henri Verbeet56692e92008-11-12 15:43:46 +0100273}