blob: 4a7344bb2f1de3429926f1a2c72b98b7a7900fed [file] [log] [blame]
José Fonsecafc550c52012-11-18 11:57:56 +00001/**************************************************************************
2 *
3 * Copyright 2012 Jose Fonseca
4 * All Rights Reserved.
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sub license,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice (including the next
14 * paragraph) shall be included in all copies or substantial portions of the
15 * Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
20 * AUTHORS,
21 * AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
22 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
23 * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
24 * SOFTWARE.
25 *
26 **************************************************************************/
27
28
29/*
30 * Auxiliary functions to compute the size of array/blob arguments.
31 */
32
Jose Fonseca9653f952015-05-19 16:32:43 +010033#pragma once
José Fonsecafc550c52012-11-18 11:57:56 +000034
35
36/* We purposedly don't include any D3D header, so that this header can be used
37 * with all D3D versions. */
38
39#include <assert.h>
40
41#include "os.hpp"
42
43
44static inline size_t
45_vertexCount(D3DPRIMITIVETYPE PrimitiveType, UINT PrimitiveCount)
46{
47 switch (PrimitiveType) {
48 case D3DPT_POINTLIST:
49 return PrimitiveCount;
50 case D3DPT_LINELIST:
51 return PrimitiveCount*2;
52 case D3DPT_LINESTRIP:
53 return PrimitiveCount + 1;
54 case D3DPT_TRIANGLELIST:
55 return PrimitiveCount * 3;
56 case D3DPT_TRIANGLESTRIP:
57 return PrimitiveCount + 2;
58 case D3DPT_TRIANGLEFAN:
José Fonseca2b36b812013-05-03 10:47:39 +010059 return PrimitiveCount + 2;
José Fonsecafc550c52012-11-18 11:57:56 +000060 default:
61 os::log("apitrace: warning: %s: unknown D3DPRIMITIVETYPE %u\n", __FUNCTION__, PrimitiveType);
62 return 0;
63 }
64}
65
66
67static inline size_t
68_vertexDataSize(D3DPRIMITIVETYPE PrimitiveType, UINT PrimitiveCount, UINT VertexStride) {
69 return _vertexCount(PrimitiveType, PrimitiveCount) * VertexStride;
70}
71
72
73static inline size_t
74_indexDataSize(D3DPRIMITIVETYPE PrimitiveType, UINT PrimitiveCount, D3DFORMAT IndexDataFormat) {
75 UINT IndexStride;
76 switch (IndexDataFormat) {
77 case D3DFMT_INDEX16:
78 IndexStride = 2;
79 break;
80 case D3DFMT_INDEX32:
81 IndexStride = 4;
82 break;
83 default:
84 os::log("apitrace: warning: %s: unexpected index D3DFORMAT %u\n", __FUNCTION__, IndexDataFormat);
85 return 0;
86 }
87 return _vertexCount(PrimitiveType, PrimitiveCount) * IndexStride;
88}
89
90
91/*
92 * Return the number of tokens for a given shader.
93 */
94static inline size_t
95_shaderSize(const DWORD *pFunction)
96{
97 DWORD dwLength = 0;
98
99 while (true) {
100 DWORD dwToken = pFunction[dwLength++];
101
102 switch (dwToken & D3DSI_OPCODE_MASK) {
103 case D3DSIO_COMMENT:
104 dwLength += (dwToken & D3DSI_COMMENTSIZE_MASK) >> D3DSI_COMMENTSIZE_SHIFT;
105 break;
106
107 case D3DSIO_END:
José Fonsecaaacd5672013-06-26 21:13:13 +0100108 if (dwToken == D3DSIO_END) {
109 return dwLength * sizeof *pFunction;
José Fonsecafc550c52012-11-18 11:57:56 +0000110 }
José Fonsecaaacd5672013-06-26 21:13:13 +0100111 break;
José Fonsecafc550c52012-11-18 11:57:56 +0000112 }
113 }
114}
115
116
117static inline size_t
118_getLockSize(D3DFORMAT Format, UINT Width, UINT Height, INT RowPitch, UINT Depth = 1, INT SlicePitch = 0) {
119 if (Width == 0 || Height == 0 || Depth == 0) {
120 return 0;
121 }
122
123 if (RowPitch < 0) {
124 os::log("apitrace: warning: %s: negative row pitch %i\n", __FUNCTION__, RowPitch);
125 return 0;
126 }
127
128 if (SlicePitch < 0) {
129 os::log("apitrace: warning: %s: negative slice pitch %i\n", __FUNCTION__, SlicePitch);
130 return 0;
131 }
132
133 switch ((DWORD)Format) {
134 case D3DFMT_DXT1:
135 case D3DFMT_DXT2:
136 case D3DFMT_DXT3:
137 case D3DFMT_DXT4:
138 case D3DFMT_DXT5:
139 Width = (Width + 3) / 4;
140 Height = (Height + 3) / 4;
141 break;
142
143#if DIRECT3D_VERSION >= 0x900
144 case D3DFMT_ATI1N:
145 case D3DFMT_ATI2N:
146 /*
147 * Because these are unsupported formats, RowPitch is not set to the
148 * number of bytes between row of blocks, but instead in such way that
149 * Height * RowPitch will match the expected size.
150 */
151 break;
152#endif /* DIRECT3D_VERSION >= 0x900 */
153
154 case D3DFMT_UYVY:
155 case D3DFMT_YUY2:
156#if DIRECT3D_VERSION >= 0x900
157 case D3DFMT_R8G8_B8G8:
158 case D3DFMT_G8R8_G8B8:
159#endif /* DIRECT3D_VERSION >= 0x900 */
160 Width = (Width + 1) / 2;
161 break;
162
163#if DIRECT3D_VERSION >= 0x900
164 case D3DFMT_NV12:
José Fonsecad5422a42013-02-06 11:36:59 +0000165 case D3DFMT_YV12:
José Fonsecafc550c52012-11-18 11:57:56 +0000166 return (Height + ((Height + 1) / 2)) * RowPitch;
167
168 case D3DFMT_NULL:
169 return 0;
170#endif /* DIRECT3D_VERSION >= 0x900 */
171
172 default:
173 break;
174 }
175
176 (void)Width;
177
178 size_t size = Height * RowPitch;
179
180 if (Depth > 1) {
181 size += (Depth - 1) * SlicePitch;
182 }
183
184 return size;
185}
186
187