José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 1 | ########################################################################## |
| 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é Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 30 | import sys |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 31 | from dllretrace import DllRetracer as Retracer |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 32 | from specs.stdapi import API |
| 33 | from specs.dxgi import dxgi |
| 34 | from specs.d3d10 import d3d10 |
| 35 | from specs.d3d10_1 import d3d10_1 |
| 36 | from specs.d3d11 import d3d11 |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 37 | |
| 38 | |
| 39 | class D3DRetracer(Retracer): |
| 40 | |
José Fonseca | e381485 | 2012-11-11 09:45:06 +0000 | [diff] [blame] | 41 | def retraceApi(self, api): |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 42 | print '// Swizzling mapping for lock addresses' |
| 43 | print 'static std::map<void *, void *> _maps;' |
| 44 | print |
José Fonseca | 7d042e0 | 2012-11-23 17:01:04 +0000 | [diff] [blame] | 45 | print r''' |
| 46 | static void |
| 47 | createWindow(DXGI_SWAP_CHAIN_DESC *pSwapChainDesc) { |
| 48 | if (pSwapChainDesc->Windowed) { |
| 49 | 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); |
| 54 | } |
| 55 | } |
| 56 | ''' |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 57 | |
José Fonseca | 73341c2 | 2012-11-24 13:04:42 +0000 | [diff] [blame] | 58 | self.table_name = 'd3dretrace::dxgi_callbacks' |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 59 | |
José Fonseca | e381485 | 2012-11-11 09:45:06 +0000 | [diff] [blame] | 60 | Retracer.retraceApi(self, api) |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 61 | |
José Fonseca | 4f49d21 | 2012-11-14 14:02:35 +0000 | [diff] [blame] | 62 | createDeviceFunctionNames = [ |
| 63 | "D3D10CreateDevice", |
| 64 | "D3D10CreateDeviceAndSwapChain", |
| 65 | "D3D10CreateDevice1", |
| 66 | "D3D10CreateDeviceAndSwapChain1", |
| 67 | "D3D11CreateDevice", |
| 68 | "D3D11CreateDeviceAndSwapChain", |
| 69 | ] |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 70 | |
José Fonseca | 4f49d21 | 2012-11-14 14:02:35 +0000 | [diff] [blame] | 71 | def invokeFunction(self, function): |
| 72 | if function.name in self.createDeviceFunctionNames: |
| 73 | # create windows as neccessary |
| 74 | if 'pSwapChainDesc' in function.argNames(): |
José Fonseca | 7d042e0 | 2012-11-23 17:01:04 +0000 | [diff] [blame] | 75 | print r' createWindow(pSwapChainDesc);' |
José Fonseca | 4f49d21 | 2012-11-14 14:02:35 +0000 | [diff] [blame] | 76 | |
| 77 | # Compensate for the fact we don't trace the software renderer |
| 78 | # module LoadLibrary call |
| 79 | if 'Software' in function.argNames(): |
| 80 | print r' if (Software) {' |
| 81 | print r' retrace::warning(call) << "using WARP for software device\n";' |
| 82 | print r' Software = LoadLibraryA("d3d10warp");' |
| 83 | print r' }' |
| 84 | |
| 85 | # Compensate for the fact we don't trace DXGI object creation |
| 86 | if function.name.startswith('D3D11CreateDevice'): |
| 87 | print r' if (DriverType == D3D_DRIVER_TYPE_UNKNOWN && !pAdapter) {' |
| 88 | print r' DriverType = D3D_DRIVER_TYPE_HARDWARE;' |
| 89 | print r' }' |
José Fonseca | ea79919 | 2012-11-13 21:37:24 +0000 | [diff] [blame] | 90 | |
José Fonseca | 42b89fc | 2012-11-27 12:19:45 +0000 | [diff] [blame] | 91 | if function.name.startswith('D3D10CreateDevice'): |
| 92 | self.forceDriver('D3D10_DRIVER_TYPE') |
| 93 | if function.name.startswith('D3D11CreateDevice'): |
| 94 | self.forceDriver('D3D_DRIVER_TYPE') |
| 95 | |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 96 | Retracer.invokeFunction(self, function) |
| 97 | |
José Fonseca | 42b89fc | 2012-11-27 12:19:45 +0000 | [diff] [blame] | 98 | def forceDriver(self, enum): |
| 99 | print r' switch (retrace::driver) {' |
| 100 | print r' case retrace::DRIVER_HARDWARE:' |
| 101 | print r' DriverType = %s_HARDWARE;' % enum |
| 102 | print r' Software = NULL;' |
| 103 | print r' break;' |
| 104 | print r' case retrace::DRIVER_SOFTWARE:' |
| 105 | print r' pAdapter = NULL;' |
| 106 | print r' DriverType = %s_WARP;' % enum |
| 107 | print r' Software = NULL;' |
| 108 | print r' break;' |
| 109 | print r' case retrace::DRIVER_REFERENCE:' |
| 110 | print r' pAdapter = NULL;' |
| 111 | print r' DriverType = %s_REFERENCE;' % enum |
| 112 | print r' Software = NULL;' |
| 113 | print r' break;' |
| 114 | print r' case retrace::DRIVER_NULL:' |
| 115 | print r' pAdapter = NULL;' |
| 116 | print r' DriverType = %s_NULL;' % enum |
| 117 | print r' Software = NULL;' |
| 118 | print r' break;' |
| 119 | print r' case retrace::DRIVER_MODULE:' |
| 120 | print r' pAdapter = NULL;' |
| 121 | print r' DriverType = %s_SOFTWARE;' % enum |
| 122 | print r' Software = LoadLibraryA(retrace::driverModule);' |
| 123 | print r' if (!Software) {' |
| 124 | print r' retrace::warning(call) << "failed to load " << retrace::driverModule << "\n";' |
| 125 | print r' }' |
| 126 | print r' break;' |
| 127 | print r' default:' |
| 128 | print r' assert(0);' |
| 129 | print r' /* fall-through */' |
| 130 | print r' case retrace::DRIVER_DEFAULT:' |
| 131 | print r' if (DriverType == %s_SOFTWARE) {' % enum |
| 132 | print r' Software = LoadLibraryA("d3d10warp");' |
| 133 | print r' if (!Software) {' |
| 134 | print r' retrace::warning(call) << "failed to load d3d10warp.dll\n";' |
| 135 | print r' }' |
| 136 | print r' }' |
| 137 | print r' break;' |
| 138 | print r' }' |
| 139 | |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 140 | def invokeInterfaceMethod(self, interface, method): |
| 141 | # keep track of the last used device for state dumping |
José Fonseca | 944088e | 2012-11-20 14:47:03 +0000 | [diff] [blame] | 142 | if interface.name in ('ID3D10Device', 'ID3D10Device1'): |
| 143 | if method.name == 'Release': |
| 144 | print r' d3d10Dumper.unbindDevice(_this);' |
| 145 | else: |
| 146 | print r' d3d10Dumper.bindDevice(_this);' |
José Fonseca | 5773beb | 2012-11-14 11:46:58 +0000 | [diff] [blame] | 147 | if interface.name in ('ID3D11DeviceContext',): |
| 148 | if method.name == 'Release': |
| 149 | print r' d3d11Dumper.unbindDevice(_this);' |
| 150 | else: |
| 151 | print r' d3d11Dumper.bindDevice(_this);' |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 152 | |
| 153 | # create windows as neccessary |
| 154 | if method.name == 'CreateSwapChain': |
José Fonseca | 7d042e0 | 2012-11-23 17:01:04 +0000 | [diff] [blame] | 155 | print r' createWindow(pDesc);' |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 156 | |
| 157 | # notify frame has been completed |
| 158 | if method.name == 'Present': |
| 159 | print r' retrace::frameComplete(call);' |
| 160 | |
| 161 | if 'pSharedResource' in method.argNames(): |
| 162 | print r' if (pSharedResource) {' |
| 163 | print r' retrace::warning(call) << "shared surfaces unsupported\n";' |
| 164 | print r' pSharedResource = NULL;' |
| 165 | print r' }' |
| 166 | |
| 167 | Retracer.invokeInterfaceMethod(self, interface, method) |
| 168 | |
| 169 | # process events after presents |
| 170 | if method.name == 'Present': |
| 171 | print r' d3dretrace::processEvents();' |
| 172 | |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 173 | if method.name == 'Map': |
| 174 | print ' VOID *_pbData = NULL;' |
| 175 | print ' size_t _MappedSize = 0;' |
| 176 | print ' _getMapInfo(_this, %s, _pbData, _MappedSize);' % ', '.join(method.argNames()) |
José Fonseca | 4f49d21 | 2012-11-14 14:02:35 +0000 | [diff] [blame] | 177 | print ' if (_MappedSize) {' |
| 178 | print ' _maps[_this] = _pbData;' |
| 179 | print ' } else {' |
| 180 | print ' return;' |
| 181 | print ' }' |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 182 | |
| 183 | if method.name == 'Unmap': |
| 184 | print ' VOID *_pbData = 0;' |
| 185 | print ' _pbData = _maps[_this];' |
| 186 | print ' if (_pbData) {' |
| 187 | print ' retrace::delRegionByPointer(_pbData);' |
José Fonseca | 4f49d21 | 2012-11-14 14:02:35 +0000 | [diff] [blame] | 188 | print ' _maps[_this] = 0;' |
José Fonseca | 610942b | 2012-11-08 10:46:03 +0000 | [diff] [blame] | 189 | print ' }' |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 190 | |
José Fonseca | 1d4fd14 | 2012-11-28 15:06:22 +0000 | [diff] [blame^] | 191 | # Attach shader byte code for lookup |
| 192 | if 'pShaderBytecode' in method.argNames(): |
| 193 | ppShader = method.args[-1] |
| 194 | assert ppShader.output |
| 195 | print r' if (retrace::dumpingState && SUCCEEDED(_result)) {' |
| 196 | print r' (*%s)->SetPrivateData(d3dstate::GUID_D3DSTATE, BytecodeLength, pShaderBytecode);' % ppShader.name |
| 197 | print r' }' |
| 198 | |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 199 | |
| 200 | def main(): |
José Fonseca | 73341c2 | 2012-11-24 13:04:42 +0000 | [diff] [blame] | 201 | print r'#include <string.h>' |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 202 | print |
| 203 | print r'#include <iostream>' |
| 204 | print |
| 205 | print r'#include "d3dretrace.hpp"' |
| 206 | print |
| 207 | |
| 208 | moduleNames = sys.argv[1:] |
| 209 | |
| 210 | api = API() |
José Fonseca | 944088e | 2012-11-20 14:47:03 +0000 | [diff] [blame] | 211 | |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 212 | if moduleNames: |
| 213 | api.addModule(dxgi) |
José Fonseca | 944088e | 2012-11-20 14:47:03 +0000 | [diff] [blame] | 214 | |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 215 | if 'd3d10' in moduleNames: |
| 216 | if 'd3d10_1' in moduleNames: |
| 217 | print r'#include "d3d10_1imports.hpp"' |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 218 | api.addModule(d3d10_1) |
| 219 | else: |
| 220 | print r'#include "d3d10imports.hpp"' |
| 221 | print r'#include "d3d10size.hpp"' |
| 222 | api.addModule(d3d10) |
José Fonseca | 944088e | 2012-11-20 14:47:03 +0000 | [diff] [blame] | 223 | print |
| 224 | print '''static d3dretrace::D3DDumper<ID3D10Device> d3d10Dumper;''' |
| 225 | print |
| 226 | |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 227 | if 'd3d11' in moduleNames: |
| 228 | print r'#include "d3d11imports.hpp"' |
| 229 | if 'd3d11_1' in moduleNames: |
| 230 | print '#include <d3d11_1.h>' |
| 231 | import specs.d3d11_1 |
| 232 | print r'#include "d3d11size.hpp"' |
José Fonseca | 5773beb | 2012-11-14 11:46:58 +0000 | [diff] [blame] | 233 | print r'#include "d3dstate.hpp"' |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 234 | api.addModule(d3d11) |
José Fonseca | 5773beb | 2012-11-14 11:46:58 +0000 | [diff] [blame] | 235 | |
| 236 | print |
| 237 | print '''static d3dretrace::D3DDumper<ID3D11DeviceContext> d3d11Dumper;''' |
| 238 | print |
José Fonseca | 6f81033 | 2012-11-11 10:05:09 +0000 | [diff] [blame] | 239 | |
| 240 | retracer = D3DRetracer() |
| 241 | retracer.retraceApi(api) |
| 242 | |
| 243 | |
| 244 | if __name__ == '__main__': |
| 245 | main() |