blob: 6a51a4d4e3c5360128398e6cf184dab4ecbf13a2 [file] [log] [blame]
José Fonseca610942b2012-11-08 10:46:03 +00001##########################################################################
2#
3# Copyright 2011 Jose Fonseca
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
27"""D3D retracer generator."""
28
29
José Fonseca6f810332012-11-11 10:05:09 +000030import sys
José Fonseca610942b2012-11-08 10:46:03 +000031from dllretrace import DllRetracer as Retracer
José Fonseca75cbb8c2013-02-12 16:19:23 +000032import specs.dxgi
José Fonseca6f810332012-11-11 10:05:09 +000033from specs.stdapi import API
34from specs.dxgi import dxgi
35from specs.d3d10 import d3d10
36from specs.d3d10_1 import d3d10_1
37from specs.d3d11 import d3d11
José Fonseca610942b2012-11-08 10:46:03 +000038
39
40class D3DRetracer(Retracer):
41
José Fonsecae3814852012-11-11 09:45:06 +000042 def retraceApi(self, api):
José Fonseca610942b2012-11-08 10:46:03 +000043 print '// Swizzling mapping for lock addresses'
44 print 'static std::map<void *, void *> _maps;'
45 print
José Fonseca7d042e02012-11-23 17:01:04 +000046 print r'''
47static void
48createWindow(DXGI_SWAP_CHAIN_DESC *pSwapChainDesc) {
José Fonseca75cbb8c2013-02-12 16:19:23 +000049 UINT Width = pSwapChainDesc->BufferDesc.Width;
50 UINT Height = pSwapChainDesc->BufferDesc.Height;
51 if (!Width) Width = 1024;
52 if (!Height) Height = 768;
53 pSwapChainDesc->OutputWindow = d3dretrace::createWindow(Width, Height);
José Fonseca7d042e02012-11-23 17:01:04 +000054}
55'''
José Fonseca610942b2012-11-08 10:46:03 +000056
José Fonseca73341c22012-11-24 13:04:42 +000057 self.table_name = 'd3dretrace::dxgi_callbacks'
José Fonseca610942b2012-11-08 10:46:03 +000058
José Fonsecae3814852012-11-11 09:45:06 +000059 Retracer.retraceApi(self, api)
José Fonseca610942b2012-11-08 10:46:03 +000060
José Fonseca4f49d212012-11-14 14:02:35 +000061 createDeviceFunctionNames = [
62 "D3D10CreateDevice",
63 "D3D10CreateDeviceAndSwapChain",
64 "D3D10CreateDevice1",
65 "D3D10CreateDeviceAndSwapChain1",
66 "D3D11CreateDevice",
67 "D3D11CreateDeviceAndSwapChain",
68 ]
José Fonseca610942b2012-11-08 10:46:03 +000069
José Fonseca4f49d212012-11-14 14:02:35 +000070 def invokeFunction(self, function):
71 if function.name in self.createDeviceFunctionNames:
72 # create windows as neccessary
73 if 'pSwapChainDesc' in function.argNames():
José Fonseca7d042e02012-11-23 17:01:04 +000074 print r' createWindow(pSwapChainDesc);'
José Fonseca4f49d212012-11-14 14:02:35 +000075
José Fonseca4f49d212012-11-14 14:02:35 +000076 # Compensate for the fact we don't trace DXGI object creation
77 if function.name.startswith('D3D11CreateDevice'):
78 print r' if (DriverType == D3D_DRIVER_TYPE_UNKNOWN && !pAdapter) {'
79 print r' DriverType = D3D_DRIVER_TYPE_HARDWARE;'
80 print r' }'
José Fonsecaea799192012-11-13 21:37:24 +000081
José Fonseca42b89fc2012-11-27 12:19:45 +000082 if function.name.startswith('D3D10CreateDevice'):
José Fonseca7bcc96c2012-12-05 19:28:22 +000083 # Toggle debugging
84 print r' Flags &= ~D3D10_CREATE_DEVICE_DEBUG;'
85 print r' if (retrace::debug) {'
86 print r' if (LoadLibraryA("d3d10sdklayers")) {'
87 print r' Flags |= D3D10_CREATE_DEVICE_DEBUG;'
88 print r' }'
89 print r' }'
90
91 # Force driver
José Fonseca42b89fc2012-11-27 12:19:45 +000092 self.forceDriver('D3D10_DRIVER_TYPE')
José Fonseca7bcc96c2012-12-05 19:28:22 +000093
José Fonseca42b89fc2012-11-27 12:19:45 +000094 if function.name.startswith('D3D11CreateDevice'):
José Fonseca7bcc96c2012-12-05 19:28:22 +000095 # Toggle debugging
96 print r' Flags &= ~D3D11_CREATE_DEVICE_DEBUG;'
97 print r' if (retrace::debug) {'
98 print r' if (LoadLibraryA("d3d11sdklayers")) {'
99 print r' Flags |= D3D11_CREATE_DEVICE_DEBUG;'
100 print r' }'
101 print r' }'
102
103 # Force driver
José Fonseca42b89fc2012-11-27 12:19:45 +0000104 self.forceDriver('D3D_DRIVER_TYPE')
105
José Fonseca610942b2012-11-08 10:46:03 +0000106 Retracer.invokeFunction(self, function)
107
José Fonseca42b89fc2012-11-27 12:19:45 +0000108 def forceDriver(self, enum):
José Fonseca82ea89d2012-12-05 19:34:26 +0000109 # This can only work when pAdapter is NULL. For non-NULL pAdapter we
110 # need to override inside the EnumAdapters call below
111 print r' if (pAdapter == NULL) {'
112 print r' switch (retrace::driver) {'
113 print r' case retrace::DRIVER_HARDWARE:'
114 print r' DriverType = %s_HARDWARE;' % enum
115 print r' Software = NULL;'
116 print r' break;'
117 print r' case retrace::DRIVER_SOFTWARE:'
118 print r' DriverType = %s_WARP;' % enum
119 print r' Software = NULL;'
120 print r' break;'
121 print r' case retrace::DRIVER_REFERENCE:'
122 print r' DriverType = %s_REFERENCE;' % enum
123 print r' Software = NULL;'
124 print r' break;'
125 print r' case retrace::DRIVER_NULL:'
126 print r' DriverType = %s_NULL;' % enum
127 print r' Software = NULL;'
128 print r' break;'
129 print r' case retrace::DRIVER_MODULE:'
130 print r' DriverType = %s_SOFTWARE;' % enum
131 print r' Software = LoadLibraryA(retrace::driverModule);'
José Fonseca42b89fc2012-11-27 12:19:45 +0000132 print r' if (!Software) {'
José Fonseca82ea89d2012-12-05 19:34:26 +0000133 print r' retrace::warning(call) << "failed to load " << retrace::driverModule << "\n";'
José Fonseca42b89fc2012-11-27 12:19:45 +0000134 print r' }'
José Fonseca82ea89d2012-12-05 19:34:26 +0000135 print r' break;'
136 print r' default:'
137 print r' assert(0);'
138 print r' /* fall-through */'
139 print r' case retrace::DRIVER_DEFAULT:'
140 print r' if (DriverType == %s_SOFTWARE) {' % enum
141 print r' Software = LoadLibraryA("d3d10warp");'
142 print r' if (!Software) {'
143 print r' retrace::warning(call) << "failed to load d3d10warp.dll\n";'
144 print r' }'
145 print r' }'
146 print r' break;'
José Fonseca42b89fc2012-11-27 12:19:45 +0000147 print r' }'
José Fonseca82ea89d2012-12-05 19:34:26 +0000148 print r' } else {'
149 print r' Software = NULL;'
José Fonseca42b89fc2012-11-27 12:19:45 +0000150 print r' }'
151
José Fonseca610942b2012-11-08 10:46:03 +0000152 def invokeInterfaceMethod(self, interface, method):
153 # keep track of the last used device for state dumping
José Fonseca944088e2012-11-20 14:47:03 +0000154 if interface.name in ('ID3D10Device', 'ID3D10Device1'):
155 if method.name == 'Release':
156 print r' d3d10Dumper.unbindDevice(_this);'
157 else:
158 print r' d3d10Dumper.bindDevice(_this);'
José Fonseca5773beb2012-11-14 11:46:58 +0000159 if interface.name in ('ID3D11DeviceContext',):
160 if method.name == 'Release':
161 print r' d3d11Dumper.unbindDevice(_this);'
162 else:
163 print r' d3d11Dumper.bindDevice(_this);'
José Fonseca610942b2012-11-08 10:46:03 +0000164
José Fonseca75cbb8c2013-02-12 16:19:23 +0000165 if interface.name == 'IDXGIFactory' and method.name == 'QueryInterface':
166 print r' if (riid == IID_IDXGIFactoryDWM) {'
167 print r' _this->AddRef();'
168 print r' *ppvObj = new d3dretrace::CDXGIFactoryDWM(_this);'
169 print r' _result = S_OK;'
170 print r' } else {'
171 Retracer.invokeInterfaceMethod(self, interface, method)
172 print r' }'
173 return
174
José Fonseca610942b2012-11-08 10:46:03 +0000175 # create windows as neccessary
176 if method.name == 'CreateSwapChain':
José Fonseca7d042e02012-11-23 17:01:04 +0000177 print r' createWindow(pDesc);'
José Fonseca610942b2012-11-08 10:46:03 +0000178
179 # notify frame has been completed
180 if method.name == 'Present':
181 print r' retrace::frameComplete(call);'
182
183 if 'pSharedResource' in method.argNames():
184 print r' if (pSharedResource) {'
185 print r' retrace::warning(call) << "shared surfaces unsupported\n";'
186 print r' pSharedResource = NULL;'
187 print r' }'
188
José Fonseca82ea89d2012-12-05 19:34:26 +0000189 # Force driver
190 if interface.name.startswith('IDXGIFactory') and method.name == 'EnumAdapters':
191 print r' const char *szSoftware = NULL;'
192 print r' switch (retrace::driver) {'
193 print r' case retrace::DRIVER_REFERENCE:'
194 print r' case retrace::DRIVER_SOFTWARE:'
195 print r' szSoftware = "d3d10warp.dll";'
196 print r' break;'
197 print r' case retrace::DRIVER_MODULE:'
198 print r' szSoftware = retrace::driverModule;'
199 print r' break;'
200 print r' default:'
201 print r' break;'
202 print r' }'
203 print r' HMODULE hSoftware = NULL;'
204 print r' if (szSoftware) {'
205 print r' hSoftware = LoadLibraryA(szSoftware);'
206 print r' if (!hSoftware) {'
207 print r' retrace::warning(call) << "failed to load " << szSoftware << "\n";'
208 print r' }'
209 print r' }'
210 print r' if (hSoftware) {'
211 print r' _result = _this->CreateSoftwareAdapter(hSoftware, ppAdapter);'
212 print r' } else {'
213 Retracer.invokeInterfaceMethod(self, interface, method)
214 print r' }'
215 return
216
José Fonseca75cbb8c2013-02-12 16:19:23 +0000217 if interface.name.startswith('ID3D10Device') and method.name == 'OpenSharedResource':
218 print r' retrace::warning(call) << "replacing shared resource with checker pattern\n";'
219 print r' D3D10_TEXTURE2D_DESC Desc;'
220 print r' memset(&Desc, 0, sizeof Desc);'
221 print r' Desc.Width = 8;'
222 print r' Desc.Height = 8;'
223 print r' Desc.MipLevels = 1;'
224 print r' Desc.ArraySize = 1;'
225 print r' Desc.Format = DXGI_FORMAT_B8G8R8A8_UNORM;'
226 print r' Desc.SampleDesc.Count = 1;'
227 print r' Desc.SampleDesc.Quality = 0;'
228 print r' Desc.Usage = D3D10_USAGE_DEFAULT;'
229 print r' Desc.BindFlags = D3D10_BIND_SHADER_RESOURCE | D3D10_BIND_RENDER_TARGET;'
230 print r' Desc.CPUAccessFlags = 0x0;'
231 print r' Desc.MiscFlags = 0 /* D3D10_RESOURCE_MISC_SHARED */;'
232 print r'''
233 const DWORD Checker[8][8] = {
234 { 0, ~0, 0, ~0, 0, ~0, 0, ~0, },
235 {~0, 0, ~0, 0, ~0, 0, ~0, 0, },
236 { 0, ~0, 0, ~0, 0, ~0, 0, ~0, },
237 {~0, 0, ~0, 0, ~0, 0, ~0, 0, },
238 { 0, ~0, 0, ~0, 0, ~0, 0, ~0, },
239 {~0, 0, ~0, 0, ~0, 0, ~0, 0, },
240 { 0, ~0, 0, ~0, 0, ~0, 0, ~0, },
241 {~0, 0, ~0, 0, ~0, 0, ~0, 0, }
242 };
243 const D3D10_SUBRESOURCE_DATA InitialData = {Checker, sizeof Checker[0], sizeof Checker};
244 '''
245 print r' _result = _this->CreateTexture2D(&Desc, &InitialData, (ID3D10Texture2D**)ppResource);'
246 self.checkResult(method.type)
José Fonseca999284f2013-02-19 13:29:26 +0000247 return
248
249 if method.name == 'Map':
250 # Reset _DO_NOT_WAIT flags. Otherwise they may fail, and we have no
251 # way to cope with it (other than retry).
252 mapFlagsArg = method.getArgByName('MapFlags')
253 for flag in mapFlagsArg.type.values:
254 if flag.endswith('_MAP_FLAG_DO_NOT_WAIT'):
255 print r' MapFlags &= ~%s;' % flag
256
257 Retracer.invokeInterfaceMethod(self, interface, method)
José Fonseca610942b2012-11-08 10:46:03 +0000258
259 # process events after presents
260 if method.name == 'Present':
261 print r' d3dretrace::processEvents();'
262
José Fonseca610942b2012-11-08 10:46:03 +0000263 if method.name == 'Map':
264 print ' VOID *_pbData = NULL;'
265 print ' size_t _MappedSize = 0;'
266 print ' _getMapInfo(_this, %s, _pbData, _MappedSize);' % ', '.join(method.argNames())
José Fonseca4f49d212012-11-14 14:02:35 +0000267 print ' if (_MappedSize) {'
268 print ' _maps[_this] = _pbData;'
269 print ' } else {'
270 print ' return;'
271 print ' }'
José Fonseca610942b2012-11-08 10:46:03 +0000272
273 if method.name == 'Unmap':
274 print ' VOID *_pbData = 0;'
275 print ' _pbData = _maps[_this];'
276 print ' if (_pbData) {'
277 print ' retrace::delRegionByPointer(_pbData);'
José Fonseca4f49d212012-11-14 14:02:35 +0000278 print ' _maps[_this] = 0;'
José Fonseca610942b2012-11-08 10:46:03 +0000279 print ' }'
José Fonseca6f810332012-11-11 10:05:09 +0000280
José Fonseca1d4fd142012-11-28 15:06:22 +0000281 # Attach shader byte code for lookup
282 if 'pShaderBytecode' in method.argNames():
283 ppShader = method.args[-1]
284 assert ppShader.output
285 print r' if (retrace::dumpingState && SUCCEEDED(_result)) {'
286 print r' (*%s)->SetPrivateData(d3dstate::GUID_D3DSTATE, BytecodeLength, pShaderBytecode);' % ppShader.name
287 print r' }'
288
José Fonseca6f810332012-11-11 10:05:09 +0000289
290def main():
José Fonseca75cbb8c2013-02-12 16:19:23 +0000291 print r'#define INITGUID'
292 print
José Fonseca73341c22012-11-24 13:04:42 +0000293 print r'#include <string.h>'
José Fonseca6f810332012-11-11 10:05:09 +0000294 print
295 print r'#include <iostream>'
296 print
297 print r'#include "d3dretrace.hpp"'
298 print
299
300 moduleNames = sys.argv[1:]
301
302 api = API()
José Fonseca944088e2012-11-20 14:47:03 +0000303
José Fonseca6f810332012-11-11 10:05:09 +0000304 if moduleNames:
José Fonseca75cbb8c2013-02-12 16:19:23 +0000305 print r'#include "d3dretrace_dxgi.hpp"'
José Fonseca6f810332012-11-11 10:05:09 +0000306 api.addModule(dxgi)
José Fonseca944088e2012-11-20 14:47:03 +0000307
José Fonseca6f810332012-11-11 10:05:09 +0000308 if 'd3d10' in moduleNames:
309 if 'd3d10_1' in moduleNames:
310 print r'#include "d3d10_1imports.hpp"'
José Fonseca6f810332012-11-11 10:05:09 +0000311 api.addModule(d3d10_1)
312 else:
313 print r'#include "d3d10imports.hpp"'
314 print r'#include "d3d10size.hpp"'
315 api.addModule(d3d10)
José Fonseca944088e2012-11-20 14:47:03 +0000316 print
317 print '''static d3dretrace::D3DDumper<ID3D10Device> d3d10Dumper;'''
318 print
319
José Fonseca6f810332012-11-11 10:05:09 +0000320 if 'd3d11' in moduleNames:
321 print r'#include "d3d11imports.hpp"'
322 if 'd3d11_1' in moduleNames:
323 print '#include <d3d11_1.h>'
324 import specs.d3d11_1
325 print r'#include "d3d11size.hpp"'
José Fonseca5773beb2012-11-14 11:46:58 +0000326 print r'#include "d3dstate.hpp"'
José Fonseca6f810332012-11-11 10:05:09 +0000327 api.addModule(d3d11)
José Fonseca5773beb2012-11-14 11:46:58 +0000328
329 print
330 print '''static d3dretrace::D3DDumper<ID3D11DeviceContext> d3d11Dumper;'''
331 print
José Fonseca6f810332012-11-11 10:05:09 +0000332
333 retracer = D3DRetracer()
334 retracer.retraceApi(api)
335
336
337if __name__ == '__main__':
338 main()