blob: fa411e0db0cd8caa1cdcbadac47c824f92553994 [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é Fonsecabbb9eb32013-04-16 19:38:00 +010033
José Fonsecac8695f72013-03-06 12:12:04 +000034#include "dxgistate.hpp"
José Fonseca29d4bda2014-02-28 17:50:24 +000035#include "d3d10state.hpp"
José Fonsecac8695f72013-03-06 12:12:04 +000036
37#ifdef __MINGW32__
38#define nullptr NULL
39#endif
40#include "DirectXTex.h"
41
42
43namespace d3dstate {
44
45
José Fonsecabbb9eb32013-04-16 19:38:00 +010046static DXGI_FORMAT
José Fonseca52fc2b02013-10-10 18:43:33 -070047ChooseConversionFormat(DXGI_FORMAT Format, unsigned &numChannels, image::ChannelType &channelType)
José Fonsecabbb9eb32013-04-16 19:38:00 +010048{
José Fonseca52fc2b02013-10-10 18:43:33 -070049 numChannels = 4;
José Fonseca75af8672013-09-16 12:20:40 +010050 channelType = image::TYPE_UNORM8;
51
José Fonsecabbb9eb32013-04-16 19:38:00 +010052 switch (Format) {
53
54 // Float
55 case DXGI_FORMAT_R32G32B32A32_FLOAT:
56 case DXGI_FORMAT_R32G32B32_FLOAT:
57 case DXGI_FORMAT_R16G16B16A16_FLOAT:
58 case DXGI_FORMAT_R32G32_FLOAT:
59 case DXGI_FORMAT_R11G11B10_FLOAT:
60 case DXGI_FORMAT_R16G16_FLOAT:
61 case DXGI_FORMAT_R32_FLOAT:
62 case DXGI_FORMAT_R16_FLOAT:
63 case DXGI_FORMAT_R9G9B9E5_SHAREDEXP:
64 case DXGI_FORMAT_BC6H_UF16:
65 case DXGI_FORMAT_BC6H_SF16:
José Fonseca75af8672013-09-16 12:20:40 +010066 channelType = image::TYPE_FLOAT;
67 return DXGI_FORMAT_R32G32B32A32_FLOAT;
José Fonsecabbb9eb32013-04-16 19:38:00 +010068
69 // Unsigned normalized
70 case DXGI_FORMAT_R16G16B16A16_UNORM:
71 case DXGI_FORMAT_R10G10B10A2_UNORM:
72 case DXGI_FORMAT_R8G8B8A8_UNORM:
73 case DXGI_FORMAT_R16G16_UNORM:
74 case DXGI_FORMAT_R8G8_UNORM:
75 case DXGI_FORMAT_R16_UNORM:
76 case DXGI_FORMAT_R8_UNORM:
77 case DXGI_FORMAT_A8_UNORM:
78 case DXGI_FORMAT_R1_UNORM:
79 case DXGI_FORMAT_R8G8_B8G8_UNORM:
80 case DXGI_FORMAT_G8R8_G8B8_UNORM:
81 case DXGI_FORMAT_BC1_UNORM:
82 case DXGI_FORMAT_BC2_UNORM:
83 case DXGI_FORMAT_BC3_UNORM:
84 case DXGI_FORMAT_BC4_UNORM:
85 case DXGI_FORMAT_BC5_UNORM:
86 case DXGI_FORMAT_B5G6R5_UNORM:
87 case DXGI_FORMAT_B5G5R5A1_UNORM:
88 case DXGI_FORMAT_B8G8R8A8_UNORM:
89 case DXGI_FORMAT_B8G8R8X8_UNORM:
90 case DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM:
91 case DXGI_FORMAT_BC7_UNORM:
92 return DXGI_FORMAT_R8G8B8A8_UNORM;
93
94 // SRGB
95 case DXGI_FORMAT_R8G8B8A8_UNORM_SRGB:
96 case DXGI_FORMAT_BC1_UNORM_SRGB:
97 case DXGI_FORMAT_BC2_UNORM_SRGB:
98 case DXGI_FORMAT_BC3_UNORM_SRGB:
99 case DXGI_FORMAT_B8G8R8A8_UNORM_SRGB:
100 case DXGI_FORMAT_B8G8R8X8_UNORM_SRGB:
101 case DXGI_FORMAT_BC7_UNORM_SRGB:
102 return DXGI_FORMAT_R8G8B8A8_UNORM;
103
104 // Signed normalized
105 case DXGI_FORMAT_R16G16B16A16_SNORM:
106 case DXGI_FORMAT_R8G8B8A8_SNORM:
107 case DXGI_FORMAT_R16G16_SNORM:
108 case DXGI_FORMAT_R8G8_SNORM:
109 case DXGI_FORMAT_R16_SNORM:
110 case DXGI_FORMAT_R8_SNORM:
111 case DXGI_FORMAT_BC4_SNORM:
112 case DXGI_FORMAT_BC5_SNORM:
113 return DXGI_FORMAT_R8G8B8A8_SNORM;
114
115 // Unsigned
116 case DXGI_FORMAT_R32G32B32A32_UINT:
117 case DXGI_FORMAT_R32G32B32_UINT:
118 case DXGI_FORMAT_R16G16B16A16_UINT:
119 case DXGI_FORMAT_R32G32_UINT:
120 case DXGI_FORMAT_R10G10B10A2_UINT:
121 case DXGI_FORMAT_R8G8B8A8_UINT:
122 case DXGI_FORMAT_R16G16_UINT:
123 case DXGI_FORMAT_R32_UINT:
124 case DXGI_FORMAT_R8G8_UINT:
125 case DXGI_FORMAT_R16_UINT:
126 case DXGI_FORMAT_R8_UINT:
127 return DXGI_FORMAT_R8G8B8A8_UINT;
128
129 // Signed
130 case DXGI_FORMAT_R32G32B32A32_SINT:
131 case DXGI_FORMAT_R32G32B32_SINT:
132 case DXGI_FORMAT_R16G16B16A16_SINT:
133 case DXGI_FORMAT_R32G32_SINT:
134 case DXGI_FORMAT_R8G8B8A8_SINT:
135 case DXGI_FORMAT_R16G16_SINT:
136 case DXGI_FORMAT_R32_SINT:
137 case DXGI_FORMAT_R8G8_SINT:
138 case DXGI_FORMAT_R16_SINT:
139 case DXGI_FORMAT_R8_SINT:
140 return DXGI_FORMAT_R8G8B8A8_SINT;
141
142 // Depth
143 case DXGI_FORMAT_D32_FLOAT:
144 case DXGI_FORMAT_D24_UNORM_S8_UINT:
145 case DXGI_FORMAT_D32_FLOAT_S8X24_UINT:
146 case DXGI_FORMAT_D16_UNORM:
José Fonseca52fc2b02013-10-10 18:43:33 -0700147 numChannels = 1;
148 channelType = image::TYPE_FLOAT;
149 return DXGI_FORMAT_D32_FLOAT;
José Fonsecabbb9eb32013-04-16 19:38:00 +0100150
151 // Typeless
152 case DXGI_FORMAT_UNKNOWN:
153 case DXGI_FORMAT_R32G32B32A32_TYPELESS:
154 case DXGI_FORMAT_R32G32B32_TYPELESS:
155 case DXGI_FORMAT_R16G16B16A16_TYPELESS:
156 case DXGI_FORMAT_R32G32_TYPELESS:
157 case DXGI_FORMAT_R32G8X24_TYPELESS:
158 case DXGI_FORMAT_R10G10B10A2_TYPELESS:
159 case DXGI_FORMAT_R8G8B8A8_TYPELESS:
160 case DXGI_FORMAT_R16G16_TYPELESS:
161 case DXGI_FORMAT_R32_TYPELESS:
162 case DXGI_FORMAT_R8G8_TYPELESS:
163 case DXGI_FORMAT_R16_TYPELESS:
164 case DXGI_FORMAT_R8_TYPELESS:
165 case DXGI_FORMAT_BC1_TYPELESS:
166 case DXGI_FORMAT_BC2_TYPELESS:
167 case DXGI_FORMAT_BC3_TYPELESS:
168 case DXGI_FORMAT_BC4_TYPELESS:
169 case DXGI_FORMAT_BC5_TYPELESS:
170 case DXGI_FORMAT_B8G8R8A8_TYPELESS:
171 case DXGI_FORMAT_B8G8R8X8_TYPELESS:
172 case DXGI_FORMAT_BC6H_TYPELESS:
173 case DXGI_FORMAT_BC7_TYPELESS:
174 case DXGI_FORMAT_R24G8_TYPELESS:
175 case DXGI_FORMAT_R24_UNORM_X8_TYPELESS:
176 case DXGI_FORMAT_X24_TYPELESS_G8_UINT:
177 case DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS:
178 case DXGI_FORMAT_X32_TYPELESS_G8X24_UINT:
179 return DXGI_FORMAT_UNKNOWN;
180
181 default:
182 return DXGI_FORMAT_R8G8B8A8_UNORM;
183 }
184}
185
186
José Fonsecac8695f72013-03-06 12:12:04 +0000187/**
188 * Convert between DXGI formats.
189 *
190 */
191HRESULT
192ConvertFormat(DXGI_FORMAT SrcFormat,
193 void *SrcData,
194 UINT SrcPitch,
195 DXGI_FORMAT DstFormat,
196 void *DstData,
197 UINT DstPitch,
198 UINT Width, UINT Height)
199{
200 HRESULT hr;
201
202 DirectX::Image SrcImage;
203 DirectX::Image DstImage;
204
205 SrcImage.width = Width;
206 SrcImage.height = Height;
207 SrcImage.format = SrcFormat;
208 SrcImage.rowPitch = SrcPitch;
209 SrcImage.slicePitch = Height * SrcPitch;
210 SrcImage.pixels = (uint8_t*)SrcData;
211
212 DstImage.width = Width;
213 DstImage.height = Height;
214 DstImage.format = DstFormat;
215 DstImage.rowPitch = DstPitch;
216 DstImage.slicePitch = Height * DstPitch;
217 DstImage.pixels = (uint8_t*)DstData;
218
219 DirectX::Rect rect(0, 0, Width, Height);
220
José Fonseca386d29d2013-05-20 13:35:23 +0100221 hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
222 assert(SUCCEEDED(hr));
223 if (FAILED(hr)) {
224 return hr;
225 }
226
José Fonsecac8695f72013-03-06 12:12:04 +0000227 if (SrcFormat != DstFormat) {
228 DirectX::ScratchImage ScratchImage;
229 ScratchImage.Initialize2D(DstFormat, Width, Height, 1, 1);
230
José Fonseca4bdfcb92013-07-09 13:27:00 +0100231 if (DirectX::IsCompressed(SrcFormat)) {
232 hr = DirectX::Decompress(SrcImage, DstFormat, ScratchImage);
233 } else {
234 hr = DirectX::Convert(SrcImage, DstFormat, DirectX::TEX_FILTER_DEFAULT, 0.0f, ScratchImage);
235 }
236
José Fonsecac8695f72013-03-06 12:12:04 +0000237 if (SUCCEEDED(hr)) {
238 hr = CopyRectangle(*ScratchImage.GetImage(0, 0, 0), rect, DstImage, DirectX::TEX_FILTER_DEFAULT, 0, 0);
239 }
240 } else {
241 hr = CopyRectangle(SrcImage, rect, DstImage, DirectX::TEX_FILTER_DEFAULT, 0, 0);
242 }
243
244 return hr;
245}
246
247
José Fonsecabbb9eb32013-04-16 19:38:00 +0100248image::Image *
249ConvertImage(DXGI_FORMAT SrcFormat,
250 void *SrcData,
251 UINT SrcPitch,
252 UINT Width, UINT Height)
253{
José Fonseca52fc2b02013-10-10 18:43:33 -0700254 unsigned numChannels;
José Fonseca75af8672013-09-16 12:20:40 +0100255 image::ChannelType channelType;
José Fonsecabbb9eb32013-04-16 19:38:00 +0100256
José Fonseca52fc2b02013-10-10 18:43:33 -0700257 DXGI_FORMAT DstFormat = ChooseConversionFormat(SrcFormat, numChannels, channelType);
José Fonsecabbb9eb32013-04-16 19:38:00 +0100258 if (DstFormat == DXGI_FORMAT_UNKNOWN) {
259 return NULL;
260 }
261
José Fonseca52fc2b02013-10-10 18:43:33 -0700262 image::Image *image = new image::Image(Width, Height, numChannels, false, channelType);
José Fonsecabbb9eb32013-04-16 19:38:00 +0100263 if (!image) {
264 return NULL;
265 }
266 assert(image->stride() > 0);
267
268 HRESULT hr;
269 hr = ConvertFormat(SrcFormat,
270 SrcData, SrcPitch,
271 DstFormat,
272 image->start(), image->stride(),
273 Width, Height);
274 if (FAILED(hr)) {
José Fonseca4bdfcb92013-07-09 13:27:00 +0100275 std::cerr << "warning: failed to convert from format " << SrcFormat << " to format " << DstFormat << "\n";
José Fonsecabbb9eb32013-04-16 19:38:00 +0100276 delete image;
277 image = NULL;
278 }
279
280 return image;
281}
282
José Fonseca29d4bda2014-02-28 17:50:24 +0000283
284image::Image *
285getRenderTargetImage(IDXGISwapChain *pSwapChain)
286{
287 HRESULT hr;
288
289 assert(pSwapChain);
290 if (!pSwapChain) {
291 return NULL;
292 }
293
294 DXGI_SWAP_CHAIN_DESC Desc;
295 hr = pSwapChain->GetDesc(&Desc);
296 assert(SUCCEEDED(hr));
297
298 /*
299 * There is a IDXGISurface::Map method, but swapchains are not normally mappable,
300 * and there is no way to copy into a staging resource, which effectively means there
301 * is no way to read a IDXGISwapChain using DXGI interfaces alone.
302 *
303 * We must figure out
304 * the appropriate D3D10/D3D11 interfaces, and use them instead.
305 */
306
307 ID3D10Resource *pD3D10Resource = NULL;
308 hr = pSwapChain->GetBuffer(0, IID_ID3D10Resource, (void **)&pD3D10Resource);
309 if (SUCCEEDED(hr)) {
310 ID3D10Device * pD3D10Device;
311 hr = pSwapChain->GetDevice(IID_ID3D10Device, (void **)&pD3D10Device);
312 assert(SUCCEEDED(hr));
313 if (FAILED(hr)) {
314 return NULL;
315 }
316
317 DXGI_FORMAT Format = Desc.BufferDesc.Format;
318
319 return getSubResourceImage(pD3D10Device, pD3D10Resource, Format, 0, 0);
320 }
321
322 /* FIXME: Handle D3D11 too. */
323
324 return NULL;
325}
326
327
328void
329dumpDevice(std::ostream &os, IDXGISwapChain *pSwapChain)
330{
331 JSONWriter json(os);
332
333 json.beginMember("framebuffer");
334 json.beginObject();
335
336 if (pSwapChain) {
337 image::Image *image;
338 image = getRenderTargetImage(pSwapChain);
339 if (image) {
340 json.beginMember("SWAP_CHAIN");
341 json.writeImage(image, "UNKNOWN");
342 json.endMember();
343 }
344 }
345
346 json.endObject();
347 json.endMember(); // framebuffer
348}
349
350
José Fonsecac8695f72013-03-06 12:12:04 +0000351} /* namespace d3dstate */