blob: 7bb7f0a8d8672fdaff2e6bcadbfb3d30f2426f11 [file] [log] [blame]
John Bauman89401822014-05-06 15:04:28 -04001// SwiftShader Software Renderer
2//
John Bauman66b8ab22014-05-06 15:57:45 -04003// Copyright(c) 2005-2013 TransGaming Inc.
John Bauman89401822014-05-06 15:04:28 -04004//
5// All rights reserved. No part of this software may be copied, distributed, transmitted,
6// transcribed, stored in a retrieval system, translated into any human or computer
7// language by any means, or disclosed to third parties without the explicit written
8// agreement of TransGaming Inc. Without such an agreement, no rights or licenses, express
9// or implied, including but not limited to any patent rights, are granted to you.
10//
11
12#include "VertexShader.hpp"
13
14#include "Vertex.hpp"
15#include "Debug.hpp"
16
John Bauman66b8ab22014-05-06 15:57:45 -040017#include <string.h>
18
John Bauman89401822014-05-06 15:04:28 -040019namespace sw
20{
John Bauman19bac1e2014-05-06 15:23:49 -040021 VertexShader::VertexShader(const VertexShader *vs) : Shader()
22 {
23 version = 0x0300;
24 positionRegister = Pos;
25 pointSizeRegister = -1; // No vertex point size
Alexis Hetu6743bbf2015-04-21 17:06:14 -040026 instanceIdDeclared = false;
John Bauman19bac1e2014-05-06 15:23:49 -040027
John Bauman66b8ab22014-05-06 15:57:45 -040028 for(int i = 0; i < MAX_INPUT_ATTRIBUTES; i++)
John Bauman19bac1e2014-05-06 15:23:49 -040029 {
30 input[i] = Semantic(-1, -1);
31 }
32
33 if(vs) // Make a copy
34 {
Alexis Hetu903e0252014-11-25 14:25:32 -050035 for(size_t i = 0; i < vs->getLength(); i++)
John Bauman19bac1e2014-05-06 15:23:49 -040036 {
37 append(new sw::Shader::Instruction(*vs->getInstruction(i)));
38 }
39
40 memcpy(output, vs->output, sizeof(output));
41 memcpy(input, vs->input, sizeof(input));
42 positionRegister = vs->positionRegister;
43 pointSizeRegister = vs->pointSizeRegister;
Alexis Hetudd8df682015-06-05 17:08:39 -040044 instanceIdDeclared = vs->instanceIdDeclared;
John Bauman19bac1e2014-05-06 15:23:49 -040045 usedSamplers = vs->usedSamplers;
46
John Baumand4ae8632014-05-06 16:18:33 -040047 optimize();
John Bauman19bac1e2014-05-06 15:23:49 -040048 analyze();
49 }
50 }
51
52 VertexShader::VertexShader(const unsigned long *token) : Shader()
John Bauman89401822014-05-06 15:04:28 -040053 {
54 parse(token);
John Bauman19bac1e2014-05-06 15:23:49 -040055
56 positionRegister = Pos;
57 pointSizeRegister = -1; // No vertex point size
Alexis Hetu6743bbf2015-04-21 17:06:14 -040058 instanceIdDeclared = false;
John Bauman19bac1e2014-05-06 15:23:49 -040059
John Bauman66b8ab22014-05-06 15:57:45 -040060 for(int i = 0; i < MAX_INPUT_ATTRIBUTES; i++)
John Bauman19bac1e2014-05-06 15:23:49 -040061 {
62 input[i] = Semantic(-1, -1);
63 }
64
John Baumand4ae8632014-05-06 16:18:33 -040065 optimize();
John Bauman19bac1e2014-05-06 15:23:49 -040066 analyze();
John Bauman89401822014-05-06 15:04:28 -040067 }
68
69 VertexShader::~VertexShader()
70 {
71 }
72
John Bauman89401822014-05-06 15:04:28 -040073 int VertexShader::validate(const unsigned long *const token)
74 {
75 if(!token)
76 {
77 return 0;
78 }
79
80 unsigned short version = (unsigned short)(token[0] & 0x0000FFFF);
81 unsigned char minorVersion = (unsigned char)(token[0] & 0x000000FF);
82 unsigned char majorVersion = (unsigned char)((token[0] & 0x0000FF00) >> 8);
83 ShaderType shaderType = (ShaderType)((token[0] & 0xFFFF0000) >> 16);
84
85 if(shaderType != SHADER_VERTEX || majorVersion > 3)
86 {
87 return 0;
88 }
89
90 int instructionCount = 1;
91
92 for(int i = 0; token[i] != 0x0000FFFF; i++)
93 {
94 if((token[i] & 0x0000FFFF) == 0x0000FFFE) // Comment token
95 {
96 int length = (token[i] & 0x7FFF0000) >> 16;
97
98 i += length;
99 }
100 else
101 {
John Bauman19bac1e2014-05-06 15:23:49 -0400102 Shader::Opcode opcode = (Shader::Opcode)(token[i] & 0x0000FFFF);
John Bauman89401822014-05-06 15:04:28 -0400103
104 switch(opcode)
105 {
John Bauman19bac1e2014-05-06 15:23:49 -0400106 case Shader::OPCODE_TEXCOORD:
107 case Shader::OPCODE_TEXKILL:
108 case Shader::OPCODE_TEX:
109 case Shader::OPCODE_TEXBEM:
110 case Shader::OPCODE_TEXBEML:
111 case Shader::OPCODE_TEXREG2AR:
112 case Shader::OPCODE_TEXREG2GB:
113 case Shader::OPCODE_TEXM3X2PAD:
114 case Shader::OPCODE_TEXM3X2TEX:
115 case Shader::OPCODE_TEXM3X3PAD:
116 case Shader::OPCODE_TEXM3X3TEX:
117 case Shader::OPCODE_RESERVED0:
118 case Shader::OPCODE_TEXM3X3SPEC:
119 case Shader::OPCODE_TEXM3X3VSPEC:
120 case Shader::OPCODE_TEXREG2RGB:
121 case Shader::OPCODE_TEXDP3TEX:
122 case Shader::OPCODE_TEXM3X2DEPTH:
123 case Shader::OPCODE_TEXDP3:
124 case Shader::OPCODE_TEXM3X3:
125 case Shader::OPCODE_TEXDEPTH:
126 case Shader::OPCODE_CMP0:
127 case Shader::OPCODE_BEM:
128 case Shader::OPCODE_DP2ADD:
129 case Shader::OPCODE_DFDX:
130 case Shader::OPCODE_DFDY:
131 case Shader::OPCODE_TEXLDD:
John Bauman89401822014-05-06 15:04:28 -0400132 return 0; // Unsupported operation
133 default:
134 instructionCount++;
135 break;
136 }
137
138 i += size(token[i], version);
139 }
140 }
141
142 return instructionCount;
143 }
144
145 bool VertexShader::containsTexldl() const
146 {
147 return texldl;
148 }
149
John Bauman19bac1e2014-05-06 15:23:49 -0400150 void VertexShader::analyze()
151 {
152 analyzeInput();
153 analyzeOutput();
154 analyzeDirtyConstants();
155 analyzeTexldl();
156 analyzeDynamicBranching();
157 analyzeSamplers();
158 analyzeCallSites();
159 analyzeDynamicIndexing();
160 }
161
John Bauman89401822014-05-06 15:04:28 -0400162 void VertexShader::analyzeInput()
163 {
John Bauman19bac1e2014-05-06 15:23:49 -0400164 for(unsigned int i = 0; i < instruction.size(); i++)
John Bauman89401822014-05-06 15:04:28 -0400165 {
John Bauman19bac1e2014-05-06 15:23:49 -0400166 if(instruction[i]->opcode == Shader::OPCODE_DCL &&
167 instruction[i]->dst.type == Shader::PARAMETER_INPUT)
John Bauman89401822014-05-06 15:04:28 -0400168 {
John Bauman19bac1e2014-05-06 15:23:49 -0400169 int index = instruction[i]->dst.index;
John Bauman89401822014-05-06 15:04:28 -0400170
John Bauman19bac1e2014-05-06 15:23:49 -0400171 input[index] = Semantic(instruction[i]->usage, instruction[i]->usageIndex);
John Bauman89401822014-05-06 15:04:28 -0400172 }
173 }
174 }
175
176 void VertexShader::analyzeOutput()
177 {
John Bauman89401822014-05-06 15:04:28 -0400178 if(version < 0x0300)
179 {
John Bauman19bac1e2014-05-06 15:23:49 -0400180 output[Pos][0] = Semantic(Shader::USAGE_POSITION, 0);
181 output[Pos][1] = Semantic(Shader::USAGE_POSITION, 0);
182 output[Pos][2] = Semantic(Shader::USAGE_POSITION, 0);
183 output[Pos][3] = Semantic(Shader::USAGE_POSITION, 0);
John Bauman89401822014-05-06 15:04:28 -0400184
John Bauman19bac1e2014-05-06 15:23:49 -0400185 for(unsigned int i = 0; i < instruction.size(); i++)
John Bauman89401822014-05-06 15:04:28 -0400186 {
John Bauman19bac1e2014-05-06 15:23:49 -0400187 const DestinationParameter &dst = instruction[i]->dst;
John Bauman89401822014-05-06 15:04:28 -0400188
189 switch(dst.type)
190 {
John Bauman19bac1e2014-05-06 15:23:49 -0400191 case Shader::PARAMETER_RASTOUT:
John Bauman89401822014-05-06 15:04:28 -0400192 switch(dst.index)
193 {
194 case 0:
195 // Position already assumed written
196 break;
197 case 1:
John Bauman19bac1e2014-05-06 15:23:49 -0400198 output[Fog][0] = Semantic(Shader::USAGE_FOG, 0);
John Bauman89401822014-05-06 15:04:28 -0400199 break;
200 case 2:
John Bauman19bac1e2014-05-06 15:23:49 -0400201 output[Pts][1] = Semantic(Shader::USAGE_PSIZE, 0);
John Bauman89401822014-05-06 15:04:28 -0400202 pointSizeRegister = Pts;
203 break;
204 default: ASSERT(false);
205 }
206 break;
John Bauman19bac1e2014-05-06 15:23:49 -0400207 case Shader::PARAMETER_ATTROUT:
John Bauman89401822014-05-06 15:04:28 -0400208 if(dst.index == 0)
209 {
John Bauman19bac1e2014-05-06 15:23:49 -0400210 if(dst.x) output[D0][0] = Semantic(Shader::USAGE_COLOR, 0);
211 if(dst.y) output[D0][1] = Semantic(Shader::USAGE_COLOR, 0);
212 if(dst.z) output[D0][2] = Semantic(Shader::USAGE_COLOR, 0);
213 if(dst.w) output[D0][3] = Semantic(Shader::USAGE_COLOR, 0);
John Bauman89401822014-05-06 15:04:28 -0400214 }
215 else if(dst.index == 1)
216 {
John Bauman19bac1e2014-05-06 15:23:49 -0400217 if(dst.x) output[D1][0] = Semantic(Shader::USAGE_COLOR, 1);
218 if(dst.y) output[D1][1] = Semantic(Shader::USAGE_COLOR, 1);
219 if(dst.z) output[D1][2] = Semantic(Shader::USAGE_COLOR, 1);
220 if(dst.w) output[D1][3] = Semantic(Shader::USAGE_COLOR, 1);
John Bauman89401822014-05-06 15:04:28 -0400221 }
222 else ASSERT(false);
223 break;
John Bauman19bac1e2014-05-06 15:23:49 -0400224 case Shader::PARAMETER_TEXCRDOUT:
225 if(dst.x) output[T0 + dst.index][0] = Semantic(Shader::USAGE_TEXCOORD, dst.index);
226 if(dst.y) output[T0 + dst.index][1] = Semantic(Shader::USAGE_TEXCOORD, dst.index);
227 if(dst.z) output[T0 + dst.index][2] = Semantic(Shader::USAGE_TEXCOORD, dst.index);
John Bauman66b8ab22014-05-06 15:57:45 -0400228 if(dst.w) output[T0 + dst.index][3] = Semantic(Shader::USAGE_TEXCOORD, dst.index);
John Bauman89401822014-05-06 15:04:28 -0400229 break;
230 default:
231 break;
232 }
233 }
234 }
235 else // Shader Model 3.0 input declaration
236 {
John Bauman19bac1e2014-05-06 15:23:49 -0400237 for(unsigned int i = 0; i < instruction.size(); i++)
John Bauman89401822014-05-06 15:04:28 -0400238 {
John Bauman19bac1e2014-05-06 15:23:49 -0400239 if(instruction[i]->opcode == Shader::OPCODE_DCL &&
240 instruction[i]->dst.type == Shader::PARAMETER_OUTPUT)
John Bauman89401822014-05-06 15:04:28 -0400241 {
John Bauman19bac1e2014-05-06 15:23:49 -0400242 unsigned char usage = instruction[i]->usage;
243 unsigned char usageIndex = instruction[i]->usageIndex;
John Bauman89401822014-05-06 15:04:28 -0400244
John Bauman19bac1e2014-05-06 15:23:49 -0400245 const DestinationParameter &dst = instruction[i]->dst;
John Bauman89401822014-05-06 15:04:28 -0400246
247 if(dst.x) output[dst.index][0] = Semantic(usage, usageIndex);
248 if(dst.y) output[dst.index][1] = Semantic(usage, usageIndex);
249 if(dst.z) output[dst.index][2] = Semantic(usage, usageIndex);
250 if(dst.w) output[dst.index][3] = Semantic(usage, usageIndex);
251
John Bauman19bac1e2014-05-06 15:23:49 -0400252 if(usage == Shader::USAGE_POSITION && usageIndex == 0)
John Bauman89401822014-05-06 15:04:28 -0400253 {
254 positionRegister = dst.index;
255 }
256
John Bauman19bac1e2014-05-06 15:23:49 -0400257 if(usage == Shader::USAGE_PSIZE && usageIndex == 0)
John Bauman89401822014-05-06 15:04:28 -0400258 {
259 pointSizeRegister = dst.index;
260 }
261 }
262 }
263 }
264 }
265
266 void VertexShader::analyzeTexldl()
267 {
268 texldl = false;
269
John Bauman19bac1e2014-05-06 15:23:49 -0400270 for(unsigned int i = 0; i < instruction.size(); i++)
John Bauman89401822014-05-06 15:04:28 -0400271 {
John Bauman19bac1e2014-05-06 15:23:49 -0400272 if(instruction[i]->opcode == Shader::OPCODE_TEXLDL)
John Bauman89401822014-05-06 15:04:28 -0400273 {
274 texldl = true;
275
276 break;
277 }
278 }
279 }
280}