blob: 51e1009c98e79cc2a9f117eb062eb6e4f4494173 [file] [log] [blame]
José Fonsecac8695f72013-03-06 12:12:04 +00001/**************************************************************************
2 *
3 * Copyright 2013 VMware, Inc.
4 * All Rights Reserved.
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 * THE SOFTWARE.
23 *
24 **************************************************************************/
25
26
José Fonsecabbb9eb32013-04-16 19:38:00 +010027#include <assert.h>
28
José Fonseca4bdfcb92013-07-09 13:27:00 +010029#include <iostream>
30
José Fonsecabbb9eb32013-04-16 19:38:00 +010031#include "image.hpp"
José Fonseca29d4bda2014-02-28 17:50:24 +000032#include "json.hpp"
José Fonseca00dc2f12014-06-12 19:18:11 +010033#include "com_ptr.hpp"
José Fonsecabbb9eb32013-04-16 19:38:00 +010034
José Fonsecac8695f72013-03-06 12:12:04 +000035#include "dxgistate.hpp"
José Fonseca29d4bda2014-02-28 17:50:24 +000036#include "d3d10state.hpp"
José Fonseca00dc2f12014-06-12 19:18:11 +010037#include "d3d11state.hpp"
José Fonsecac8695f72013-03-06 12:12:04 +000038
39#ifdef __MINGW32__
40#define nullptr NULL
41#endif
42#include "DirectXTex.h"
43
44
45namespace d3dstate {
46
47
José Fonsecabbb9eb32013-04-16 19:38:00 +010048static DXGI_FORMAT
José Fonseca52fc2b02013-10-10 18:43:33 -070049ChooseConversionFormat(DXGI_FORMAT Format, unsigned &numChannels, image::ChannelType &channelType)
José Fonsecabbb9eb32013-04-16 19:38:00 +010050{
José Fonseca52fc2b02013-10-10 18:43:33 -070051 numChannels = 4;
José Fonseca75af8672013-09-16 12:20:40 +010052 channelType = image::TYPE_UNORM8;
53
José Fonsecabbb9eb32013-04-16 19:38:00 +010054 switch (Format) {
55
56 // Float
57 case DXGI_FORMAT_R32G32B32A32_FLOAT:
58 case DXGI_FORMAT_R32G32B32_FLOAT:
59 case DXGI_FORMAT_R16G16B16A16_FLOAT:
60 case DXGI_FORMAT_R32G32_FLOAT:
61 case DXGI_FORMAT_R11G11B10_FLOAT:
62 case DXGI_FORMAT_R16G16_FLOAT:
63 case DXGI_FORMAT_R32_FLOAT:
64 case DXGI_FORMAT_R16_FLOAT:
65 case DXGI_FORMAT_R9G9B9E5_SHAREDEXP:
66 case DXGI_FORMAT_BC6H_UF16:
67 case DXGI_FORMAT_BC6H_SF16:
José Fonseca75af8672013-09-16 12:20:40 +010068 channelType = image::TYPE_FLOAT;
69 return DXGI_FORMAT_R32G32B32A32_FLOAT;
José Fonsecabbb9eb32013-04-16 19:38:00 +010070
71 // Unsigned normalized
72 case DXGI_FORMAT_R16G16B16A16_UNORM:
73 case DXGI_FORMAT_R10G10B10A2_UNORM:
74 case DXGI_FORMAT_R8G8B8A8_UNORM:
75 case DXGI_FORMAT_R16G16_UNORM:
76 case DXGI_FORMAT_R8G8_UNORM:
77 case DXGI_FORMAT_R16_UNORM:
78 case DXGI_FORMAT_R8_UNORM:
79 case DXGI_FORMAT_A8_UNORM:
80 case DXGI_FORMAT_R1_UNORM:
81 case DXGI_FORMAT_R8G8_B8G8_UNORM:
82 case DXGI_FORMAT_G8R8_G8B8_UNORM:
83 case DXGI_FORMAT_BC1_UNORM:
84 case DXGI_FORMAT_BC2_UNORM:
85 case DXGI_FORMAT_BC3_UNORM:
86 case DXGI_FORMAT_BC4_UNORM:
87 case DXGI_FORMAT_BC5_UNORM:
88 case DXGI_FORMAT_B5G6R5_UNORM:
89 case DXGI_FORMAT_B5G5R5A1_UNORM:
90 case DXGI_FORMAT_B8G8R8A8_UNORM:
91 case DXGI_FORMAT_B8G8R8X8_UNORM:
92 case DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM:
93 case DXGI_FORMAT_BC7_UNORM:
94 return DXGI_FORMAT_R8G8B8A8_UNORM;
95
96 // SRGB
97 case DXGI_FORMAT_R8G8B8A8_UNORM_SRGB:
98 case DXGI_FORMAT_BC1_UNORM_SRGB:
99 case DXGI_FORMAT_BC2_UNORM_SRGB:
100 case DXGI_FORMAT_BC3_UNORM_SRGB:
101 case DXGI_FORMAT_B8G8R8A8_UNORM_SRGB:
102 case DXGI_FORMAT_B8G8R8X8_UNORM_SRGB:
103 case DXGI_FORMAT_BC7_UNORM_SRGB:
104 return DXGI_FORMAT_R8G8B8A8_UNORM;
105
106 // Signed normalized
107 case DXGI_FORMAT_R16G16B16A16_SNORM:
108 case DXGI_FORMAT_R8G8B8A8_SNORM:
109 case DXGI_FORMAT_R16G16_SNORM:
110 case DXGI_FORMAT_R8G8_SNORM:
111 case DXGI_FORMAT_R16_SNORM:
112 case DXGI_FORMAT_R8_SNORM:
113 case DXGI_FORMAT_BC4_SNORM:
114 case DXGI_FORMAT_BC5_SNORM:
115 return DXGI_FORMAT_R8G8B8A8_SNORM;
116
117 // Unsigned
118 case DXGI_FORMAT_R32G32B32A32_UINT:
119 case DXGI_FORMAT_R32G32B32_UINT:
120 case DXGI_FORMAT_R16G16B16A16_UINT:
121 case DXGI_FORMAT_R32G32_UINT:
122 case DXGI_FORMAT_R10G10B10A2_UINT:
123 case DXGI_FORMAT_R8G8B8A8_UINT:
124 case DXGI_FORMAT_R16G16_UINT:
125 case DXGI_FORMAT_R32_UINT:
126 case DXGI_FORMAT_R8G8_UINT:
127 case DXGI_FORMAT_R16_UINT:
128 case DXGI_FORMAT_R8_UINT:
129 return DXGI_FORMAT_R8G8B8A8_UINT;
130
131 // Signed
132 case DXGI_FORMAT_R32G32B32A32_SINT:
133 case DXGI_FORMAT_R32G32B32_SINT:
134 case DXGI_FORMAT_R16G16B16A16_SINT:
135 case DXGI_FORMAT_R32G32_SINT:
136 case DXGI_FORMAT_R8G8B8A8_SINT:
137 case DXGI_FORMAT_R16G16_SINT:
138 case DXGI_FORMAT_R32_SINT:
139 case DXGI_FORMAT_R8G8_SINT:
140 case DXGI_FORMAT_R16_SINT:
141 case DXGI_FORMAT_R8_SINT:
142 return DXGI_FORMAT_R8G8B8A8_SINT;
143
144 // Depth
145 case DXGI_FORMAT_D32_FLOAT:
146 case DXGI_FORMAT_D24_UNORM_S8_UINT:
147 case DXGI_FORMAT_D32_FLOAT_S8X24_UINT:
148 case DXGI_FORMAT_D16_UNORM:
José Fonseca52fc2b02013-10-10 18:43:33 -0700149 numChannels = 1;
150 channelType = image::TYPE_FLOAT;
151 return DXGI_FORMAT_D32_FLOAT;
José Fonsecabbb9eb32013-04-16 19:38:00 +0100152
153 // Typeless
154 case DXGI_FORMAT_UNKNOWN:
155 case DXGI_FORMAT_R32G32B32A32_TYPELESS:
156 case DXGI_FORMAT_R32G32B32_TYPELESS:
157 case DXGI_FORMAT_R16G16B16A16_TYPELESS:
158 case DXGI_FORMAT_R32G32_TYPELESS:
159 case DXGI_FORMAT_R32G8X24_TYPELESS:
160 case DXGI_FORMAT_R10G10B10A2_TYPELESS:
161 case DXGI_FORMAT_R8G8B8A8_TYPELESS:
162 case DXGI_FORMAT_R16G16_TYPELESS:
163 case DXGI_FORMAT_R32_TYPELESS:
164 case DXGI_FORMAT_R8G8_TYPELESS:
165 case DXGI_FORMAT_R16_TYPELESS:
166 case DXGI_FORMAT_R8_TYPELESS:
167 case DXGI_FORMAT_BC1_TYPELESS:
168 case DXGI_FORMAT_BC2_TYPELESS:
169 case DXGI_FORMAT_BC3_TYPELESS:
170 case DXGI_FORMAT_BC4_TYPELESS:
171 case DXGI_FORMAT_BC5_TYPELESS:
172 case DXGI_FORMAT_B8G8R8A8_TYPELESS:
173 case DXGI_FORMAT_B8G8R8X8_TYPELESS:
174 case DXGI_FORMAT_BC6H_TYPELESS:
175 case DXGI_FORMAT_BC7_TYPELESS:
176 case DXGI_FORMAT_R24G8_TYPELESS:
177 case DXGI_FORMAT_R24_UNORM_X8_TYPELESS:
178 case DXGI_FORMAT_X24_TYPELESS_G8_UINT:
179 case DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS:
180 case DXGI_FORMAT_X32_TYPELESS_G8X24_UINT:
181 return DXGI_FORMAT_UNKNOWN;
182
183 default:
184 return DXGI_FORMAT_R8G8B8A8_UNORM;
185 }
186}
187
188
José Fonsecac8695f72013-03-06 12:12:04 +0000189/**
190 * Convert between DXGI formats.
191 *
192 */
193HRESULT
194ConvertFormat(DXGI_FORMAT SrcFormat,
195 void *SrcData,
196 UINT SrcPitch,
197 DXGI_FORMAT DstFormat,
198 void *DstData,
199 UINT DstPitch,
200 UINT Width, UINT Height)
201{
202 HRESULT hr;
203
204 DirectX::Image SrcImage;
205 DirectX::Image DstImage;
206
207 SrcImage.width = Width;
208 SrcImage.height = Height;
209 SrcImage.format = SrcFormat;
210 SrcImage.rowPitch = SrcPitch;
211 SrcImage.slicePitch = Height * SrcPitch;
212 SrcImage.pixels = (uint8_t*)SrcData;
213
214 DstImage.width = Width;
215 DstImage.height = Height;
216 DstImage.format = DstFormat;
217 DstImage.rowPitch = DstPitch;
218 DstImage.slicePitch = Height * DstPitch;
219 DstImage.pixels = (uint8_t*)DstData;
220
221 DirectX::Rect rect(0, 0, Width, Height);
222
José Fonseca386d29d2013-05-20 13:35:23 +0100223 hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
224 assert(SUCCEEDED(hr));
225 if (FAILED(hr)) {
226 return hr;
227 }
228
José Fonsecac8695f72013-03-06 12:12:04 +0000229 if (SrcFormat != DstFormat) {
230 DirectX::ScratchImage ScratchImage;
231 ScratchImage.Initialize2D(DstFormat, Width, Height, 1, 1);
232
José Fonseca4bdfcb92013-07-09 13:27:00 +0100233 if (DirectX::IsCompressed(SrcFormat)) {
234 hr = DirectX::Decompress(SrcImage, DstFormat, ScratchImage);
235 } else {
236 hr = DirectX::Convert(SrcImage, DstFormat, DirectX::TEX_FILTER_DEFAULT, 0.0f, ScratchImage);
237 }
238
José Fonsecac8695f72013-03-06 12:12:04 +0000239 if (SUCCEEDED(hr)) {
240 hr = CopyRectangle(*ScratchImage.GetImage(0, 0, 0), rect, DstImage, DirectX::TEX_FILTER_DEFAULT, 0, 0);
241 }
242 } else {
243 hr = CopyRectangle(SrcImage, rect, DstImage, DirectX::TEX_FILTER_DEFAULT, 0, 0);
244 }
245
246 return hr;
247}
248
249
José Fonsecabbb9eb32013-04-16 19:38:00 +0100250image::Image *
251ConvertImage(DXGI_FORMAT SrcFormat,
252 void *SrcData,
253 UINT SrcPitch,
254 UINT Width, UINT Height)
255{
José Fonseca52fc2b02013-10-10 18:43:33 -0700256 unsigned numChannels;
José Fonseca75af8672013-09-16 12:20:40 +0100257 image::ChannelType channelType;
José Fonsecabbb9eb32013-04-16 19:38:00 +0100258
José Fonseca52fc2b02013-10-10 18:43:33 -0700259 DXGI_FORMAT DstFormat = ChooseConversionFormat(SrcFormat, numChannels, channelType);
José Fonsecabbb9eb32013-04-16 19:38:00 +0100260 if (DstFormat == DXGI_FORMAT_UNKNOWN) {
261 return NULL;
262 }
263
José Fonseca52fc2b02013-10-10 18:43:33 -0700264 image::Image *image = new image::Image(Width, Height, numChannels, false, channelType);
José Fonsecabbb9eb32013-04-16 19:38:00 +0100265 if (!image) {
266 return NULL;
267 }
268 assert(image->stride() > 0);
269
270 HRESULT hr;
271 hr = ConvertFormat(SrcFormat,
272 SrcData, SrcPitch,
273 DstFormat,
274 image->start(), image->stride(),
275 Width, Height);
276 if (FAILED(hr)) {
José Fonseca4bdfcb92013-07-09 13:27:00 +0100277 std::cerr << "warning: failed to convert from format " << SrcFormat << " to format " << DstFormat << "\n";
José Fonsecabbb9eb32013-04-16 19:38:00 +0100278 delete image;
279 image = NULL;
280 }
281
282 return image;
283}
284
José Fonseca29d4bda2014-02-28 17:50:24 +0000285
286image::Image *
287getRenderTargetImage(IDXGISwapChain *pSwapChain)
288{
289 HRESULT hr;
290
291 assert(pSwapChain);
292 if (!pSwapChain) {
293 return NULL;
294 }
295
296 DXGI_SWAP_CHAIN_DESC Desc;
297 hr = pSwapChain->GetDesc(&Desc);
298 assert(SUCCEEDED(hr));
299
300 /*
301 * There is a IDXGISurface::Map method, but swapchains are not normally mappable,
302 * and there is no way to copy into a staging resource, which effectively means there
303 * is no way to read a IDXGISwapChain using DXGI interfaces alone.
304 *
305 * We must figure out
306 * the appropriate D3D10/D3D11 interfaces, and use them instead.
307 */
308
José Fonseca00dc2f12014-06-12 19:18:11 +0100309 com_ptr<ID3D10Device> pD3D10Device;
310 hr = pSwapChain->GetDevice(IID_ID3D10Device, (void **)&pD3D10Device);
José Fonseca29d4bda2014-02-28 17:50:24 +0000311 if (SUCCEEDED(hr)) {
José Fonseca00dc2f12014-06-12 19:18:11 +0100312 ID3D10Resource *pD3D10Resource = NULL;
313 hr = pSwapChain->GetBuffer(0, IID_ID3D10Resource, (void **)&pD3D10Resource);
José Fonseca29d4bda2014-02-28 17:50:24 +0000314 assert(SUCCEEDED(hr));
315 if (FAILED(hr)) {
316 return NULL;
317 }
318
319 DXGI_FORMAT Format = Desc.BufferDesc.Format;
320
321 return getSubResourceImage(pD3D10Device, pD3D10Resource, Format, 0, 0);
322 }
323
José Fonseca00dc2f12014-06-12 19:18:11 +0100324 com_ptr<ID3D11Device> pD3D11Device;
325 hr = pSwapChain->GetDevice(IID_ID3D11Device, (void **)&pD3D11Device);
326 if (SUCCEEDED(hr)) {
327 ID3D11Resource *pD3D11Resource = NULL;
328 hr = pSwapChain->GetBuffer(0, IID_ID3D11Resource, (void **)&pD3D11Resource);
329 assert(SUCCEEDED(hr));
330 if (FAILED(hr)) {
331 return NULL;
332 }
333
334 com_ptr<ID3D11DeviceContext> pD3D11DeviceContext;
335 pD3D11Device->GetImmediateContext(&pD3D11DeviceContext);
336
337 DXGI_FORMAT Format = Desc.BufferDesc.Format;
338
339 return getSubResourceImage(pD3D11DeviceContext, pD3D11Resource, Format, 0, 0);
340 }
José Fonseca29d4bda2014-02-28 17:50:24 +0000341
342 return NULL;
343}
344
345
346void
347dumpDevice(std::ostream &os, IDXGISwapChain *pSwapChain)
348{
José Fonseca52024be2014-06-12 19:48:35 +0100349 HRESULT hr;
José Fonseca29d4bda2014-02-28 17:50:24 +0000350
José Fonseca52024be2014-06-12 19:48:35 +0100351 if (pSwapChain) {
352 com_ptr<ID3D10Device> pD3D10Device;
353 hr = pSwapChain->GetDevice(IID_ID3D10Device, (void **)&pD3D10Device);
354 if (SUCCEEDED(hr)) {
355 dumpDevice(os, pD3D10Device);
356 return;
357 }
358
José Fonseca52024be2014-06-12 19:48:35 +0100359 com_ptr<ID3D11Device> pD3D11Device;
360 hr = pSwapChain->GetDevice(IID_ID3D11Device, (void **)&pD3D11Device);
361 if (SUCCEEDED(hr)) {
362 com_ptr<ID3D11DeviceContext> pD3D11DeviceContext;
363 pD3D11Device->GetImmediateContext(&pD3D11DeviceContext);
364 dumpDevice(os, pD3D11DeviceContext);
365 return;
366 }
José Fonseca52024be2014-06-12 19:48:35 +0100367 }
368
369 // Fallback -- this should really never happen.
370 assert(0);
371 JSONWriter json(os);
José Fonseca29d4bda2014-02-28 17:50:24 +0000372 json.beginMember("framebuffer");
373 json.beginObject();
José Fonseca29d4bda2014-02-28 17:50:24 +0000374 if (pSwapChain) {
375 image::Image *image;
376 image = getRenderTargetImage(pSwapChain);
377 if (image) {
378 json.beginMember("SWAP_CHAIN");
379 json.writeImage(image, "UNKNOWN");
380 json.endMember();
José Fonseca52024be2014-06-12 19:48:35 +0100381 delete image;
José Fonseca29d4bda2014-02-28 17:50:24 +0000382 }
383 }
José Fonseca29d4bda2014-02-28 17:50:24 +0000384 json.endObject();
385 json.endMember(); // framebuffer
386}
387
388
José Fonsecac8695f72013-03-06 12:12:04 +0000389} /* namespace d3dstate */