2 * Copyright 2008 Henri Verbeet for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "wine/port.h"
23 #define DXGI_INIT_GUID
24 #include "dxgi_private.h"
26 WINE_DEFAULT_DEBUG_CHANNEL(dxgi
);
28 static CRITICAL_SECTION_DEBUG dxgi_cs_debug
=
31 {&dxgi_cs_debug
.ProcessLocksList
,
32 &dxgi_cs_debug
.ProcessLocksList
},
33 0, 0, {(DWORD_PTR
)(__FILE__
": dxgi_cs")}
35 CRITICAL_SECTION dxgi_cs
= {&dxgi_cs_debug
, -1, 0, 0, 0, 0};
40 struct dxgi_device_layer
*device_layers
;
44 static struct dxgi_main dxgi_main
;
46 static void dxgi_main_cleanup(void)
48 EnterCriticalSection(&dxgi_cs
);
50 HeapFree(GetProcessHeap(), 0, dxgi_main
.device_layers
);
51 dxgi_main
.device_layers
= NULL
;
52 dxgi_main
.layer_count
= 0;
54 FreeLibrary(dxgi_main
.d3d10core
);
55 dxgi_main
.d3d10core
= NULL
;
57 LeaveCriticalSection(&dxgi_cs
);
60 BOOL WINAPI
DllMain(HINSTANCE hInstDLL
, DWORD fdwReason
, LPVOID lpv
)
62 TRACE("fdwReason %u\n", fdwReason
);
66 case DLL_PROCESS_ATTACH
:
67 DisableThreadLibraryCalls(hInstDLL
);
71 case DLL_PROCESS_DETACH
:
72 if (!--dxgi_main
.refcount
) dxgi_main_cleanup();
79 HRESULT WINAPI
CreateDXGIFactory(REFIID riid
, void **factory
)
81 struct dxgi_factory
*object
;
85 TRACE("riid %s, factory %p\n", debugstr_guid(riid
), factory
);
87 object
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(*object
));
90 ERR("Failed to allocate DXGI factory object memory\n");
95 object
->vtbl
= &dxgi_factory_vtbl
;
98 EnterCriticalSection(&dxgi_cs
);
99 object
->wined3d
= WineDirect3DCreate(10, (IUnknown
*)object
);
102 hr
= DXGI_ERROR_UNSUPPORTED
;
103 LeaveCriticalSection(&dxgi_cs
);
107 object
->adapter_count
= IWineD3D_GetAdapterCount(object
->wined3d
);
108 LeaveCriticalSection(&dxgi_cs
);
109 object
->adapters
= HeapAlloc(GetProcessHeap(), 0, object
->adapter_count
* sizeof(*object
->adapters
));
110 if (!object
->adapters
)
112 ERR("Failed to allocate DXGI adapter array memory\n");
117 for (i
= 0; i
< object
->adapter_count
; ++i
)
119 struct dxgi_adapter
*adapter
= HeapAlloc(GetProcessHeap(), 0, sizeof(*adapter
));
123 ERR("Failed to allocate DXGI adapter memory\n");
124 for (j
= 0; j
< i
; ++j
)
126 HeapFree(GetProcessHeap(), 0, object
->adapters
[j
]);
132 adapter
->vtbl
= &dxgi_adapter_vtbl
;
133 adapter
->refcount
= 1;
134 adapter
->ordinal
= i
;
135 adapter
->parent
= (IDXGIFactory
*)object
;
136 object
->adapters
[i
] = (IDXGIAdapter
*)adapter
;
139 TRACE("Created IDXGIFactory %p\n", object
);
141 hr
= IDXGIFactory_QueryInterface((IDXGIFactory
*)object
, riid
, factory
);
142 IDXGIFactory_Release((IDXGIFactory
*)object
);
147 HeapFree(GetProcessHeap(), 0, object
->adapters
);
150 EnterCriticalSection(&dxgi_cs
);
151 IWineD3D_Release(object
->wined3d
);
152 LeaveCriticalSection(&dxgi_cs
);
154 HeapFree(GetProcessHeap(), 0, object
);
160 static BOOL
get_layer(enum dxgi_device_layer_id id
, struct dxgi_device_layer
*layer
)
164 EnterCriticalSection(&dxgi_cs
);
166 for (i
= 0; i
< dxgi_main
.layer_count
; ++i
)
168 if (dxgi_main
.device_layers
[i
].id
== id
)
170 *layer
= dxgi_main
.device_layers
[i
];
171 LeaveCriticalSection(&dxgi_cs
);
176 LeaveCriticalSection(&dxgi_cs
);
180 static HRESULT
register_d3d10core_layers(HMODULE d3d10core
)
182 EnterCriticalSection(&dxgi_cs
);
184 if (!dxgi_main
.d3d10core
)
187 HRESULT (WINAPI
*d3d10core_register_layers
)(void);
191 ret
= GetModuleHandleExA(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS
, (LPCSTR
)d3d10core
, &mod
);
194 LeaveCriticalSection(&dxgi_cs
);
198 d3d10core_register_layers
= (HRESULT (WINAPI
*)(void))GetProcAddress(mod
, "D3D10CoreRegisterLayers");
199 hr
= d3d10core_register_layers();
202 ERR("Failed to register d3d10core layers, returning %#x\n", hr
);
203 LeaveCriticalSection(&dxgi_cs
);
207 dxgi_main
.d3d10core
= mod
;
210 LeaveCriticalSection(&dxgi_cs
);
215 HRESULT WINAPI
DXGID3D10CreateDevice(HMODULE d3d10core
, IDXGIFactory
*factory
, IDXGIAdapter
*adapter
,
216 UINT flags
, DWORD unknown0
, void **device
)
218 IWineD3DDeviceParent
*wined3d_device_parent
;
219 struct layer_get_size_args get_size_args
;
220 struct dxgi_device
*dxgi_device
;
221 struct dxgi_device_layer d3d10_layer
;
222 IWineDXGIAdapter
*wine_adapter
;
223 UINT adapter_ordinal
;
230 TRACE("d3d10core %p, factory %p, adapter %p, flags %#x, unknown0 %#x, device %p\n",
231 d3d10core
, factory
, adapter
, flags
, unknown0
, device
);
233 hr
= register_d3d10core_layers(d3d10core
);
236 ERR("Failed to register d3d10core layers, returning %#x\n", hr
);
240 if (!get_layer(DXGI_DEVICE_LAYER_D3D10_DEVICE
, &d3d10_layer
))
242 ERR("Failed to get D3D10 device layer\n");
247 hr
= d3d10_layer
.init(d3d10_layer
.id
, &count
, NULL
);
250 WARN("Failed to initialize D3D10 device layer\n");
254 get_size_args
.unknown0
= 0;
255 get_size_args
.unknown1
= 0;
256 get_size_args
.unknown2
= NULL
;
257 get_size_args
.unknown3
= NULL
;
258 get_size_args
.adapter
= adapter
;
259 get_size_args
.interface_major
= 10;
260 get_size_args
.interface_minor
= 1;
261 get_size_args
.version_build
= 4;
262 get_size_args
.version_revision
= 6000;
264 device_size
= d3d10_layer
.get_size(d3d10_layer
.id
, &get_size_args
, 0);
265 device_size
+= sizeof(*dxgi_device
);
267 dxgi_device
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, device_size
);
270 ERR("Failed to allocate device memory\n");
271 return E_OUTOFMEMORY
;
274 dxgi_device
->vtbl
= &dxgi_device_vtbl
;
275 dxgi_device
->refcount
= 1;
277 layer_base
= dxgi_device
+ 1;
279 hr
= d3d10_layer
.create(d3d10_layer
.id
, &layer_base
, 0,
280 dxgi_device
, &IID_IUnknown
, (void **)&dxgi_device
->child_layer
);
283 WARN("Failed to create device, returning %#x\n", hr
);
287 hr
= IDXGIFactory_QueryInterface(factory
, &IID_IWineDXGIFactory
, (void **)&dxgi_device
->factory
);
290 WARN("This is not the factory we're looking for, returning %#x\n", hr
);
293 wined3d
= IWineDXGIFactory_get_wined3d(dxgi_device
->factory
);
295 hr
= IDXGIAdapter_QueryInterface(adapter
, &IID_IWineDXGIAdapter
, (void **)&wine_adapter
);
298 WARN("This is not the adapter we're looking for, returning %#x\n", hr
);
299 EnterCriticalSection(&dxgi_cs
);
300 IWineD3D_Release(wined3d
);
301 LeaveCriticalSection(&dxgi_cs
);
304 adapter_ordinal
= IWineDXGIAdapter_get_ordinal(wine_adapter
);
305 IWineDXGIAdapter_Release(wine_adapter
);
307 hr
= IUnknown_QueryInterface((IUnknown
*)dxgi_device
, &IID_IWineD3DDeviceParent
, (void **)&wined3d_device_parent
);
310 ERR("DXGI device should implement IWineD3DDeviceParent\n");
314 FIXME("Ignoring adapter type\n");
315 EnterCriticalSection(&dxgi_cs
);
316 hr
= IWineD3D_CreateDevice(wined3d
, adapter_ordinal
, WINED3DDEVTYPE_HAL
, NULL
, 0,
317 (IUnknown
*)dxgi_device
, wined3d_device_parent
, &dxgi_device
->wined3d_device
);
318 IWineD3DDeviceParent_Release(wined3d_device_parent
);
319 IWineD3D_Release(wined3d
);
320 LeaveCriticalSection(&dxgi_cs
);
323 WARN("Failed to create a WineD3D device, returning %#x\n", hr
);
327 *device
= dxgi_device
;
332 if (dxgi_device
->wined3d_device
)
334 EnterCriticalSection(&dxgi_cs
);
335 IWineD3DDevice_Release(dxgi_device
->wined3d_device
);
336 LeaveCriticalSection(&dxgi_cs
);
338 if (dxgi_device
->factory
) IWineDXGIFactory_Release(dxgi_device
->factory
);
339 if (dxgi_device
->child_layer
) IUnknown_Release(dxgi_device
->child_layer
);
340 HeapFree(GetProcessHeap(), 0, dxgi_device
);
345 HRESULT WINAPI
DXGID3D10RegisterLayers(const struct dxgi_device_layer
*layers
, UINT layer_count
)
348 struct dxgi_device_layer
*new_layers
;
350 TRACE("layers %p, layer_count %u\n", layers
, layer_count
);
352 EnterCriticalSection(&dxgi_cs
);
354 if (!dxgi_main
.layer_count
)
355 new_layers
= HeapAlloc(GetProcessHeap(), 0, layer_count
* sizeof(*new_layers
));
357 new_layers
= HeapReAlloc(GetProcessHeap(), 0, dxgi_main
.device_layers
,
358 (dxgi_main
.layer_count
+ layer_count
) * sizeof(*new_layers
));
362 LeaveCriticalSection(&dxgi_cs
);
363 ERR("Failed to allocate layer memory\n");
364 return E_OUTOFMEMORY
;
367 for (i
= 0; i
< layer_count
; ++i
)
369 const struct dxgi_device_layer
*layer
= &layers
[i
];
371 TRACE("layer %d: id %#x, init %p, get_size %p, create %p\n",
372 i
, layer
->id
, layer
->init
, layer
->get_size
, layer
->create
);
374 new_layers
[dxgi_main
.layer_count
+ i
] = *layer
;
377 dxgi_main
.device_layers
= new_layers
;
378 dxgi_main
.layer_count
+= layer_count
;
380 LeaveCriticalSection(&dxgi_cs
);