1 # -*- coding: utf-8 -*- 2 3 #------------------------------------------------------------------------- 4 # drawElements Quality Program utilities 5 # -------------------------------------- 6 # 7 # Copyright 2015 The Android Open Source Project 8 # 9 # Licensed under the Apache License, Version 2.0 (the "License"); 10 # you may not use this file except in compliance with the License. 11 # You may obtain a copy of the License at 12 # 13 # http://www.apache.org/licenses/LICENSE-2.0 14 # 15 # Unless required by applicable law or agreed to in writing, software 16 # distributed under the License is distributed on an "AS IS" BASIS, 17 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 18 # See the License for the specific language governing permissions and 19 # limitations under the License. 20 # 21 #------------------------------------------------------------------------- 22 23 import sys 24 import random 25 import operator 26 import itertools 27 28 from genutil import * 29 30 random.seed(1234567) 31 indices = xrange(sys.maxint) 32 33 # Constructors: 34 # 35 # - scalars types 36 # * int <-> float <-> bool (also float(float) etc.) 37 # * to bool: zero means false, others true 38 # * from bool: false==0, true==1 39 # * \todo [petri] float<->int rounding rules? 40 # - scalar type from vector 41 # * choose the first component 42 # - vectors & matrices 43 # * vector from scalar: broadcast to all components 44 # * matrix from scalar: broadcast scalar to diagonal, other components zero 45 # * vector from vector: copy existing components 46 # + illegal: vector from smaller vector 47 # * mat from mat: copy existing components, other components from identity matrix 48 # * from components: consumed by-component in column-major order, must have same 49 # number of components, 50 # + note: vec4(mat2) valid 51 # \todo [petri] Implement! 52 # - notes: 53 # * type conversions are always allowed: mat3(ivec3, bvec3, bool, int, float) is valid! 54 # 55 # Accessors: 56 # 57 # - vector components 58 # * .xyzw, .rgba, .stpq 59 # * illegal to mix 60 # * now allowed for scalar types 61 # * legal to chain: vec4.rgba.xyzw.stpq 62 # * illegal to select more than 4 components 63 # * array indexing with [] operator 64 # * can also write! 65 # - matrix columns 66 # * [] accessor 67 # * note: mat4[0].x = 1.0; vs mat4[0][0] = 1.0; ?? 68 # * out-of-bounds accesses 69 # 70 # \todo [petri] Accessors! 71 # 72 # Spec issues: 73 # 74 # - constructing larger vector from smaller: vec3(vec2) ? 75 # - base type and size conversion at same time: vec4(bool), int(vec3) allowed? 76 77 def combineVec(comps): 78 res = [] 79 for ndx in range(len(comps[0])): 80 # for x in comps: 81 # print x[ndx].toFloat().getScalars() , 82 scalars = reduce(operator.add, [x[ndx].toFloat().getScalars() for x in comps]) 83 # print "->", scalars 84 res.append(Vec.fromScalarList(scalars)) 85 return res 86 87 def combineIVec(comps): 88 res = [] 89 for ndx in range(len(comps[0])): 90 res.append(Vec.fromScalarList(reduce(operator.add, [x[ndx].toInt().getScalars() for x in comps]))) 91 return res 92 93 def combineBVec(comps): 94 res = [] 95 for ndx in range(len(comps[0])): 96 res.append(Vec.fromScalarList(reduce(operator.add, [x[ndx].toBool().getScalars() for x in comps]))) 97 return res 98 99 def combineMat(numCols, numRows, comps): 100 res = [] 101 for ndx in range(len(comps[0])): 102 scalars = reduce(operator.add, [x[ndx].toFloat().getScalars() for x in comps]) 103 res.append(Mat(numCols, numRows, scalars)) 104 return res 105 106 def combineMat2(comps): return combineMat(2, 2, comps) 107 def combineMat3(comps): return combineMat(3, 3, comps) 108 def combineMat4(comps): return combineMat(4, 4, comps) 109 110 # 0 \+ [f*f for f in lst] 111 # r = 0 \+ [f in lst -> f*f] 112 # r = 0 \+ lst 113 114 # Templates. 115 116 s_simpleCaseTemplate = """ 117 case ${{NAME}} 118 values 119 { 120 ${{VALUES}} 121 } 122 123 both "" 124 precision mediump float; 125 precision mediump int; 126 127 ${DECLARATIONS} 128 129 void main() 130 { 131 ${SETUP} 132 ${{OP}} 133 ${OUTPUT} 134 } 135 "" 136 end 137 """[1:] 138 139 s_simpleIllegalCaseTemplate = """ 140 case ${{NAME}} 141 expect compile_fail 142 values {} 143 144 both "" 145 precision mediump float; 146 precision mediump int; 147 148 ${DECLARATIONS} 149 150 void main() 151 { 152 ${SETUP} 153 ${{OP}} 154 ${OUTPUT} 155 } 156 "" 157 end 158 """[1:] 159 160 class SimpleCase(ShaderCase): 161 def __init__(self, name, inputs, outputs, op): 162 self.name = name 163 self.inputs = inputs 164 self.outputs = outputs 165 self.op = op 166 167 def __str__(self): 168 params = { 169 "NAME": self.name, 170 "VALUES": genValues(self.inputs, self.outputs), 171 "OP": self.op 172 } 173 return fillTemplate(s_simpleCaseTemplate, params) 174 175 class ConversionCase(ShaderCase): 176 def __init__(self, inValues, convFunc): 177 outValues = convFunc(inValues) 178 inType = inValues[0].typeString() 179 outType = outValues[0].typeString() 180 self.name = "%s_to_%s" % (inType, outType) 181 self.op = "out0 = %s(in0);" % outType 182 self.inputs = [("%s in0" % inType, inValues)] 183 self.outputs = [("%s out0" % outType, outValues)] 184 185 def __str__(self): 186 params = { 187 "NAME": self.name, 188 "VALUES": genValues(self.inputs, self.outputs), 189 "OP": self.op 190 } 191 return fillTemplate(s_simpleCaseTemplate, params) 192 193 class IllegalConversionCase(ShaderCase): 194 def __init__(self, inValue, outValue): 195 inType = inValue.typeString() 196 outType = outValue.typeString() 197 self.name = "%s_to_%s" % (inType, outType) 198 self.op = "%s in0 = %s;\n%s out0 = %s(in0);" % (inType, str(inValue), outType, outType) 199 self.inType = inType 200 self.outType = outType 201 202 def __str__(self): 203 params = { 204 "NAME": self.name, 205 "OP": self.op 206 } 207 return fillTemplate(s_simpleIllegalCaseTemplate, params) 208 209 class CombineCase(ShaderCase): 210 def __init__(self, inComps, combFunc): 211 self.inComps = inComps 212 self.outValues = combFunc(inComps) 213 self.outType = self.outValues[0].typeString() 214 inTypes = [values[0].typeString() for values in inComps] 215 self.name = "%s_to_%s" % ("_".join(inTypes), self.outType) 216 self.inputs = [("%s in%s" % (comp[0].typeString(), ndx), comp) for (comp, ndx) in zip(inComps, indices)] 217 self.outputs = [("%s out0" % self.outType, self.outValues)] 218 self.op = "out0 = %s(%s);" % (self.outType, ", ".join(["in%d" % x for x in range(len(inComps))])) 219 220 def __str__(self): 221 params = { 222 "NAME": self.name, 223 "VALUES": genValues(self.inputs, self.outputs), 224 "OP": self.op 225 } 226 return fillTemplate(s_simpleCaseTemplate, params) 227 228 # CASE DECLARATIONS 229 230 inFloat = [Scalar(x) for x in [0.0, 1.0, 2.0, 3.5, -0.5, -8.25, -20.125, 36.8125]] 231 inInt = [Scalar(x) for x in [0, 1, 2, 5, 8, 11, -12, -66, -192, 255]] 232 inBool = [Scalar(x) for x in [True, False]] 233 234 inVec4 = [Vec4(0.0, 0.5, 0.75, 0.825), Vec4(1.0, 1.25, 1.125, 1.75), 235 Vec4(-0.5, -2.25, -4.875, 9.0), Vec4(-32.0, 64.0, -51.0, 24.0), 236 Vec4(-0.75, -1.0/31.0, 1.0/19.0, 1.0/4.0)] 237 inVec3 = toVec3(inVec4) 238 inVec2 = toVec2(inVec4) 239 inIVec4 = toIVec4(inVec4) 240 inIVec3 = toIVec3(inVec4) 241 inIVec2 = toIVec2(inVec4) 242 inBVec4 = [Vec4(True, False, False, True), Vec4(False, False, False, True), Vec4(False, True, False, False), Vec4(True, True, True, True), Vec4(False, False, False, False)] 243 inBVec3 = toBVec3(inBVec4) 244 inBVec2 = toBVec2(inBVec4) 245 246 # \todo [petri] Enable large values when epsilon adapts to the values. 247 inMat4 = [Mat4(1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0), 248 Mat4(6.5, 12.5, -0.75, 9.975, 32.0, 1.0/48.0, -8.425, -6.542, 1.0/8.0, 1.0/16.0, 1.0/32.0, 1.0/64.0, -6.725, -0.5, -0.0125, 9.975), 249 #Mat4(128.0, 256.0, -512.0, -1024.0, 2048.0, -4096.0, 8192.0, -8192.0, 192.0, -384.0, 768.0, -1536.0, 8192.0, -8192.0, 6144.0, -6144.0) 250 ] 251 inMat3 = [Mat3(1.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 1.0), 252 Mat3(6.5, 12.5, -0.75, 32.0, 1.0/32.0, 1.0/64.0, 1.0/8.0, 1.0/16.0, 1.0/32.0), 253 #Mat3(-18.725, -0.5, -0.0125, 19.975, -0.25, -17.75, 9.25, 65.125, -21.425), 254 #Mat3(128.0, -4096.0, -8192.0, 192.0, 768.0, -1536.0, 8192.0, 6144.0, -6144.0) 255 ] 256 inMat2 = [Mat2(1.0, 0.0, 0.0, 1.0), 257 Mat2(6.5, 12.5, -0.75, 9.975), 258 Mat2(6.5, 12.5, -0.75, 9.975), 259 Mat2(8.0, 16.0, -24.0, -16.0), 260 Mat2(1.0/8.0, 1.0/16.0, 1.0/32.0, 1.0/64.0), 261 Mat2(-18.725, -0.5, -0.0125, 19.975), 262 #Mat2(128.0, -4096.0, 192.0, -1536.0), 263 #Mat2(-1536.0, 8192.0, 6144.0, -6144.0) 264 ] 265 266 def genConversionCases(inValueList, convFuncList): 267 combinations = list(itertools.product(inValueList, convFuncList)) 268 return [ConversionCase(inValues, convFunc) for (inValues, convFunc) in combinations] 269 270 def genIllegalConversionCases(inValueList, outValueList): 271 inValues = [x[0] for x in inValueList] 272 outValues = [x[0] for x in outValueList] 273 combinations = list(itertools.product(inValues, outValues)) 274 return [IllegalConversionCase(inVal, outVal) for (inVal, outVal) in combinations] 275 276 def shuffleSubLists(outer): 277 return [shuffled(inner) for inner in outer] 278 279 # Generate all combinations of CombineCases. 280 # inTupleList a list of tuples of value-lists 281 # combFuncList a list of comb* functions to combine 282 def genComponentCases(inCompLists, combFuncList): 283 res = [] 284 for comps in inCompLists: 285 maxLen = reduce(max, [len(values) for values in comps]) 286 comps = [repeatToLength(values, maxLen) for values in comps] 287 comps = [shuffled(values) for values in comps] 288 for combFunc in combFuncList: 289 res += [CombineCase(comps, combFunc)] 290 return res 291 292 allConversionCases = [] 293 294 # Scalar-to-scalar conversions. 295 allConversionCases.append(CaseGroup("scalar_to_scalar", "Scalar to Scalar Conversions", 296 genConversionCases([inFloat, inInt, inBool], [toFloat, toInt, toBool]))) 297 298 # Scalar-to-vector conversions. 299 allConversionCases.append(CaseGroup("scalar_to_vector", "Scalar to Vector Conversions", 300 genConversionCases([inFloat, inInt, inBool], [toVec2, toVec3, toVec4, toIVec2, toIVec3, toIVec4, toBVec2, toBVec3, toBVec4]))) 301 302 # Vector-to-scalar conversions. 303 allConversionCases.append(CaseGroup("vector_to_scalar", "Vector to Scalar Conversions", 304 genConversionCases([inVec2, inVec3, inVec4, inIVec2, inIVec3, inIVec4, inBVec2, inBVec3, inBVec4], [toFloat, toInt, toBool]))) 305 306 # Illegal vector-to-vector conversions (to longer vec). 307 allConversionCases.append(CaseGroup("vector_illegal", "Illegal Vector Conversions", 308 genIllegalConversionCases([inVec2, inIVec2, inBVec2], [inVec3, inIVec3, inBVec3, inVec4, inIVec4, inBVec4]) +\ 309 genIllegalConversionCases([inVec3, inIVec3, inBVec3], [inVec4, inIVec4, inBVec4]))) 310 311 # Vector-to-vector conversions (type conversions, downcasts). 312 allConversionCases.append(CaseGroup("vector_to_vector", "Vector to Vector Conversions", 313 genConversionCases([inVec4, inIVec4, inBVec4], [toVec4, toVec3, toVec2, toIVec4, toIVec3, toIVec2, toBVec4, toBVec3, toBVec2]) +\ 314 genConversionCases([inVec3, inIVec3, inBVec3], [toVec3, toVec2, toIVec3, toIVec2, toBVec3, toBVec2]) +\ 315 genConversionCases([inVec2, inIVec2, inBVec2], [toVec2, toIVec2, toBVec2]))) 316 317 # Scalar-to-matrix. 318 allConversionCases.append(CaseGroup("scalar_to_matrix", "Scalar to Matrix Conversions", 319 genConversionCases([inFloat, inInt, inBool], [toMat4, toMat3, toMat2]))) 320 321 # Vector-to-matrix. 322 #allConversionCases += genConversionCases([inVec4, inIVec4, inBVec4], [toMat4]) 323 #allConversionCases += genConversionCases([inVec3, inIVec3, inBVec3], [toMat3]) 324 #allConversionCases += genConversionCases([inVec2, inIVec2, inBVec2], [toMat2]) 325 326 # Matrix-to-matrix. 327 allConversionCases.append(CaseGroup("matrix_to_matrix", "Matrix to Matrix Conversions", 328 genConversionCases([inMat4, inMat3, inMat2], [toMat4, toMat3, toMat2]))) 329 330 # Vector-from-components, matrix-from-components. 331 in2Comp = [[inFloat, inFloat], [inInt, inInt], [inBool, inBool], [inFloat, inInt], [inFloat, inBool], [inInt, inBool]] 332 in3Comp = [[inFloat, inFloat, inFloat], [inInt, inInt, inInt], [inBool, inBool, inBool], [inBool, inFloat, inInt], [inVec2, inBool], [inBVec2, inFloat], [inBVec2, inInt], [inBool, inIVec2]] 333 in4Comp = [[inVec2, inVec2], [inBVec2, inBVec2], [inFloat, inFloat, inFloat, inFloat], [inInt, inInt, inInt, inInt], [inBool, inBool, inBool, inBool], [inBool, inFloat, inInt, inBool], [inVec2, inIVec2], [inVec2, inBVec2], [inBVec3, inFloat], [inVec3, inFloat], [inInt, inIVec2, inInt], [inBool, inFloat, inIVec2]] 334 in9Comp = [[inVec3, inVec3, inVec3], [inIVec3, inIVec3, inIVec3], [inVec2, inIVec2, inFloat, inFloat, inInt, inBool, inBool], [inBool, inFloat, inInt, inVec2, inBool, inBVec2, inFloat], [inBool, inBVec2, inInt, inVec4, inBool], [inFloat, inBVec4, inIVec2, inBool, inBool]] 335 in16Comp = [[inVec4, inVec4, inVec4, inVec4], [inIVec4, inIVec4, inIVec4, inIVec4], [inBVec4, inBVec4, inBVec4, inBVec4], [inFloat, inIVec3, inBVec3, inVec4, inIVec2, inFloat, inVec2]] 336 337 allConversionCases.append(CaseGroup("vector_combine", "Vector Combine Constructors", 338 genComponentCases(in4Comp, [combineVec, combineIVec, combineBVec]) +\ 339 genComponentCases(in3Comp, [combineVec, combineIVec, combineBVec]) +\ 340 genComponentCases(in2Comp, [combineVec, combineIVec, combineBVec]))) 341 342 allConversionCases.append(CaseGroup("matrix_combine", "Matrix Combine Constructors", 343 genComponentCases(in4Comp, [combineMat2]) +\ 344 genComponentCases(in9Comp, [combineMat3]) +\ 345 genComponentCases(in16Comp, [combineMat4]) 346 )) 347 348 # Main program. 349 350 if __name__ == "__main__": 351 print "Generating shader case files." 352 writeAllCases("conversions.test", allConversionCases) 353