• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2019 The Dawn Authors
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #include "tests/DawnTest.h"
16 
17 #include "common/Assert.h"
18 #include "common/Math.h"
19 #include "utils/ComboRenderPipelineDescriptor.h"
20 #include "utils/DawnHelpers.h"
21 
22 // Vertex format tests all work the same way: the test will render a triangle.
23 // Each test will set up a vertex buffer, and the vertex shader will check that
24 // the vertex content is the same as what we expected. On success it outputs green,
25 // otherwise red.
26 
27 constexpr uint32_t kRTSize = 400;
28 constexpr uint32_t kVertexNum = 3;
29 
Float32ToFloat16(std::vector<float> data)30 std::vector<uint16_t> Float32ToFloat16(std::vector<float> data) {
31     std::vector<uint16_t> expectedData;
32     for (auto& element : data) {
33         expectedData.push_back(Float32ToFloat16(element));
34     }
35     return expectedData;
36 }
37 
38 template <typename destType, typename srcType>
BitCast(std::vector<srcType> data)39 std::vector<destType> BitCast(std::vector<srcType> data) {
40     std::vector<destType> expectedData;
41     for (auto& element : data) {
42         expectedData.push_back(BitCast(element));
43     }
44     return expectedData;
45 }
46 
47 class VertexFormatTest : public DawnTest {
48   protected:
SetUp()49     void SetUp() override {
50         DawnTest::SetUp();
51 
52         renderPass = utils::CreateBasicRenderPass(device, kRTSize, kRTSize);
53     }
54 
55     utils::BasicRenderPass renderPass;
56 
IsNormalizedFormat(dawn::VertexFormat format)57     bool IsNormalizedFormat(dawn::VertexFormat format) {
58         switch (format) {
59             case dawn::VertexFormat::UChar2Norm:
60             case dawn::VertexFormat::UChar4Norm:
61             case dawn::VertexFormat::Char2Norm:
62             case dawn::VertexFormat::Char4Norm:
63             case dawn::VertexFormat::UShort2Norm:
64             case dawn::VertexFormat::UShort4Norm:
65             case dawn::VertexFormat::Short2Norm:
66             case dawn::VertexFormat::Short4Norm:
67                 return true;
68             default:
69                 return false;
70         }
71     }
72 
IsUnsignedFormat(dawn::VertexFormat format)73     bool IsUnsignedFormat(dawn::VertexFormat format) {
74         switch (format) {
75             case dawn::VertexFormat::UInt:
76             case dawn::VertexFormat::UChar2:
77             case dawn::VertexFormat::UChar4:
78             case dawn::VertexFormat::UShort2:
79             case dawn::VertexFormat::UShort4:
80             case dawn::VertexFormat::UInt2:
81             case dawn::VertexFormat::UInt3:
82             case dawn::VertexFormat::UInt4:
83             case dawn::VertexFormat::UChar2Norm:
84             case dawn::VertexFormat::UChar4Norm:
85             case dawn::VertexFormat::UShort2Norm:
86             case dawn::VertexFormat::UShort4Norm:
87                 return true;
88             default:
89                 return false;
90         }
91     }
92 
IsFloatFormat(dawn::VertexFormat format)93     bool IsFloatFormat(dawn::VertexFormat format) {
94         switch (format) {
95             case dawn::VertexFormat::Half2:
96             case dawn::VertexFormat::Half4:
97             case dawn::VertexFormat::Float:
98             case dawn::VertexFormat::Float2:
99             case dawn::VertexFormat::Float3:
100             case dawn::VertexFormat::Float4:
101                 return true;
102             default:
103                 return false;
104         }
105     }
106 
IsHalfFormat(dawn::VertexFormat format)107     bool IsHalfFormat(dawn::VertexFormat format) {
108         switch (format) {
109             case dawn::VertexFormat::Half2:
110             case dawn::VertexFormat::Half4:
111                 return true;
112             default:
113                 return false;
114         }
115     }
116 
BytesPerComponents(dawn::VertexFormat format)117     uint32_t BytesPerComponents(dawn::VertexFormat format) {
118         switch (format) {
119             case dawn::VertexFormat::Char2:
120             case dawn::VertexFormat::Char4:
121             case dawn::VertexFormat::UChar2:
122             case dawn::VertexFormat::UChar4:
123             case dawn::VertexFormat::UChar2Norm:
124             case dawn::VertexFormat::UChar4Norm:
125             case dawn::VertexFormat::Char2Norm:
126             case dawn::VertexFormat::Char4Norm:
127                 return 1;
128             case dawn::VertexFormat::UShort2:
129             case dawn::VertexFormat::UShort4:
130             case dawn::VertexFormat::Short2:
131             case dawn::VertexFormat::Short4:
132             case dawn::VertexFormat::UShort2Norm:
133             case dawn::VertexFormat::UShort4Norm:
134             case dawn::VertexFormat::Short2Norm:
135             case dawn::VertexFormat::Short4Norm:
136             case dawn::VertexFormat::Half2:
137             case dawn::VertexFormat::Half4:
138                 return 2;
139             case dawn::VertexFormat::UInt:
140             case dawn::VertexFormat::Int:
141             case dawn::VertexFormat::Float:
142             case dawn::VertexFormat::UInt2:
143             case dawn::VertexFormat::UInt3:
144             case dawn::VertexFormat::UInt4:
145             case dawn::VertexFormat::Int2:
146             case dawn::VertexFormat::Int3:
147             case dawn::VertexFormat::Int4:
148             case dawn::VertexFormat::Float2:
149             case dawn::VertexFormat::Float3:
150             case dawn::VertexFormat::Float4:
151                 return 4;
152             default:
153                 DAWN_UNREACHABLE();
154         }
155     }
156 
ComponentCount(dawn::VertexFormat format)157     uint32_t ComponentCount(dawn::VertexFormat format) {
158         switch (format) {
159             case dawn::VertexFormat::UInt:
160             case dawn::VertexFormat::Int:
161             case dawn::VertexFormat::Float:
162                 return 1;
163             case dawn::VertexFormat::UChar2:
164             case dawn::VertexFormat::UShort2:
165             case dawn::VertexFormat::UInt2:
166             case dawn::VertexFormat::Char2:
167             case dawn::VertexFormat::Short2:
168             case dawn::VertexFormat::Int2:
169             case dawn::VertexFormat::UChar2Norm:
170             case dawn::VertexFormat::Char2Norm:
171             case dawn::VertexFormat::UShort2Norm:
172             case dawn::VertexFormat::Short2Norm:
173             case dawn::VertexFormat::Half2:
174             case dawn::VertexFormat::Float2:
175                 return 2;
176             case dawn::VertexFormat::Int3:
177             case dawn::VertexFormat::UInt3:
178             case dawn::VertexFormat::Float3:
179                 return 3;
180             case dawn::VertexFormat::UChar4:
181             case dawn::VertexFormat::UShort4:
182             case dawn::VertexFormat::UInt4:
183             case dawn::VertexFormat::Char4:
184             case dawn::VertexFormat::Short4:
185             case dawn::VertexFormat::Int4:
186             case dawn::VertexFormat::UChar4Norm:
187             case dawn::VertexFormat::Char4Norm:
188             case dawn::VertexFormat::UShort4Norm:
189             case dawn::VertexFormat::Short4Norm:
190             case dawn::VertexFormat::Half4:
191             case dawn::VertexFormat::Float4:
192                 return 4;
193             default:
194                 DAWN_UNREACHABLE();
195         }
196     }
197 
ShaderTypeGenerator(bool isFloat,bool isNormalized,bool isUnsigned,uint32_t componentCount)198     std::string ShaderTypeGenerator(bool isFloat,
199                                     bool isNormalized,
200                                     bool isUnsigned,
201                                     uint32_t componentCount) {
202         if (componentCount == 1) {
203             if (isFloat || isNormalized) {
204                 return "float";
205             } else if (isUnsigned) {
206                 return "uint";
207             } else {
208                 return "int";
209             }
210         } else {
211             if (isNormalized || isFloat) {
212                 return "vec" + std::to_string(componentCount);
213             } else if (isUnsigned) {
214                 return "uvec" + std::to_string(componentCount);
215             } else {
216                 return "ivec" + std::to_string(componentCount);
217             }
218         }
219     }
220 
221     // The length of vertexData is fixed to 3, it aligns to triangle vertex number
222     template <typename T>
MakeTestPipeline(dawn::VertexFormat format,std::vector<T> & expectedData)223     dawn::RenderPipeline MakeTestPipeline(dawn::VertexFormat format, std::vector<T>& expectedData) {
224         bool isFloat = IsFloatFormat(format);
225         bool isNormalized = IsNormalizedFormat(format);
226         bool isUnsigned = IsUnsignedFormat(format);
227         bool isInputTypeFloat = isFloat || isNormalized;
228         bool isHalf = IsHalfFormat(format);
229         const uint16_t kNegativeZeroInHalf = 0x8000;
230 
231         uint32_t componentCount = ComponentCount(format);
232 
233         std::string variableType =
234             ShaderTypeGenerator(isFloat, isNormalized, isUnsigned, componentCount);
235         std::string expectedDataType = ShaderTypeGenerator(isFloat, isNormalized, isUnsigned, 1);
236         std::ostringstream vs;
237         vs << "#version 450\n";
238 
239         // layout(location = 0) in float/uint/int/ivecn/vecn/uvecn test;
240         vs << "layout(location = 0) in " << variableType << " test;\n";
241         vs << "layout(location = 0) out vec4 color;\n";
242         // Because x86 CPU using "extended
243         // precision"(https://en.wikipedia.org/wiki/Extended_precision) during float
244         // math(https://developer.nvidia.com/sites/default/files/akamai/cuda/files/NVIDIA-CUDA-Floating-Point.pdf),
245         // move normalization and Float16ToFloat32 into shader to generate
246         // expected value.
247         vs << "float Float16ToFloat32(uint fp16) {\n";
248         vs << "  uint magic = (uint(254) - uint(15)) << 23;\n";
249         vs << "  uint was_inf_nan = (uint(127) + uint(16)) << 23;\n";
250         vs << "  uint fp32u;\n";
251         vs << "  float fp32;\n";
252         vs << "  fp32u = (fp16 & 0x7FFF) << 13;\n";
253         vs << "  fp32 = uintBitsToFloat(fp32u) * uintBitsToFloat(magic);\n";
254         vs << "  fp32u = floatBitsToUint(fp32);\n";
255         vs << "  if (fp32 >= uintBitsToFloat(was_inf_nan)) {\n";
256         vs << "    fp32u |= uint(255) << 23;\n";
257         vs << "  }\n";
258         vs << "  fp32u |= (fp16 & 0x8000) << 16;\n";
259         vs << "  fp32 = uintBitsToFloat(fp32u);\n";
260         vs << "  return fp32;\n";
261         vs << "}\n";
262 
263         vs << "void main() {\n";
264 
265         // Hard code the triangle in the shader so that we don't have to add a vertex input for it.
266         vs << "    const vec2 pos[3] = vec2[3](vec2(-1.0f, 0.0f), vec2(-1.0f, -1.0f), vec2(0.0f, "
267               "-1.0f));\n";
268         vs << "    gl_Position = vec4(pos[gl_VertexIndex], 0.0, 1.0);\n";
269 
270         // Declare expected values.
271         vs << "    " << expectedDataType << " expected[" + std::to_string(kVertexNum) + "]";
272         vs << "[" + std::to_string(componentCount) + "];\n";
273         // Assign each elements in expected values
274         // e.g. expected[0][0] = uint(1);
275         //      expected[0][1] = uint(2);
276         for (uint32_t i = 0; i < kVertexNum; ++i) {
277             for (uint32_t j = 0; j < componentCount; ++j) {
278                 vs << "    expected[" + std::to_string(i) + "][" + std::to_string(j) + "] = "
279                    << expectedDataType << "(";
280                 if (isInputTypeFloat &&
281                     std::isnan(static_cast<float>(expectedData[i * componentCount + j]))) {
282                     // Set NaN.
283                     vs << "0.0 / 0.0);\n";
284                 } else if (isNormalized) {
285                     // Move normalize operation into shader because of CPU and GPU precision
286                     // different on float math.
287                     vs << "max(float(" << std::to_string(expectedData[i * componentCount + j])
288                        << ") / " << std::to_string(std::numeric_limits<T>::max()) << ", -1.0));\n";
289                 } else if (isHalf) {
290                     // Becasue Vulkan and D3D12 handle -0.0f through uintBitsToFloat have different
291                     // result (Vulkan take -0.0f as -0.0 but D3D12 take -0.0f as 0), add workaround
292                     // for -0.0f.
293                     if (static_cast<uint16_t>(expectedData[i * componentCount + j]) ==
294                         kNegativeZeroInHalf) {
295                         vs << "-0.0f);\n";
296                     } else {
297                         vs << "Float16ToFloat32("
298                            << std::to_string(expectedData[i * componentCount + j]);
299                         vs << "));\n";
300                     }
301                 } else {
302                     vs << std::to_string(expectedData[i * componentCount + j]) << ");\n";
303                 }
304             }
305         }
306 
307         vs << "    bool success = true;\n";
308         // Perform the checks by successively ANDing a boolean
309         for (uint32_t component = 0; component < componentCount; ++component) {
310             std::string suffix = componentCount == 1 ? "" : "[" + std::to_string(component) + "]";
311             std::string testVal = "testVal" + std::to_string(component);
312             std::string expectedVal = "expectedVal" + std::to_string(component);
313             vs << "    " << expectedDataType << " " << testVal << ";\n";
314             vs << "    " << expectedDataType << " " << expectedVal << ";\n";
315             vs << "    " << testVal << " = test" << suffix << ";\n";
316             vs << "    " << expectedVal << " = expected[gl_VertexIndex]"
317                << "[" << component << "];\n";
318             if (!isInputTypeFloat) {  // Integer / unsigned integer need to match exactly.
319                 vs << "    success = success && (" << testVal << " == " << expectedVal << ");\n";
320             } else {
321                 // TODO(shaobo.yan@intel.com) : a difference of 8 ULPs is allowed in this test
322                 // because it is required on MacbookPro 11.5,AMD Radeon HD 8870M(on macOS 10.13.6),
323                 // but that it might be possible to tighten.
324                 vs << "    if (isnan(" << expectedVal << ")) {\n";
325                 vs << "        success = success && isnan(" << testVal << ");\n";
326                 vs << "    } else {\n";
327                 vs << "        uint testValFloatToUint = floatBitsToUint(" << testVal << ");\n";
328                 vs << "        uint expectedValFloatToUint = floatBitsToUint(" << expectedVal
329                    << ");\n";
330                 vs << "        success = success && max(testValFloatToUint, "
331                       "expectedValFloatToUint)";
332                 vs << "        - min(testValFloatToUint, expectedValFloatToUint) < uint(8);\n";
333                 vs << "    }\n";
334             }
335         }
336         vs << "    if (success) {\n";
337         vs << "        color = vec4(0.0f, 1.0f, 0.0f, 1.0f);\n";
338         vs << "    } else {\n";
339         vs << "        color = vec4(1.0f, 0.0f, 0.0f, 1.0f);\n";
340         vs << "    }\n";
341         vs << "}\n";
342 
343         dawn::ShaderModule vsModule =
344             utils::CreateShaderModule(device, utils::ShaderStage::Vertex, vs.str().c_str());
345 
346         dawn::ShaderModule fsModule =
347             utils::CreateShaderModule(device, utils::ShaderStage::Fragment, R"(
348                 #version 450
349                 layout(location = 0) in vec4 color;
350                 layout(location = 0) out vec4 fragColor;
351                 void main() {
352                     fragColor = color;
353                 })");
354 
355         uint32_t bytesPerComponents = BytesPerComponents(format);
356         uint32_t strideBytes = bytesPerComponents * componentCount;
357         // Stride size must be multiple of 4 bytes.
358         if (strideBytes % 4 != 0) {
359             strideBytes += (4 - strideBytes % 4);
360         }
361 
362         utils::ComboRenderPipelineDescriptor descriptor(device);
363         descriptor.cVertexStage.module = vsModule;
364         descriptor.cFragmentStage.module = fsModule;
365         descriptor.cVertexInput.bufferCount = 1;
366         descriptor.cVertexInput.cBuffers[0].stride = strideBytes;
367         descriptor.cVertexInput.cBuffers[0].attributeCount = 1;
368         descriptor.cVertexInput.cAttributes[0].format = format;
369         descriptor.cColorStates[0]->format = renderPass.colorFormat;
370 
371         return device.CreateRenderPipeline(&descriptor);
372     }
373 
374     template <typename VertexType, typename ExpectedType>
DoVertexFormatTest(dawn::VertexFormat format,std::vector<VertexType> vertex,std::vector<ExpectedType> expectedData)375     void DoVertexFormatTest(dawn::VertexFormat format,
376                             std::vector<VertexType> vertex,
377                             std::vector<ExpectedType> expectedData) {
378         dawn::RenderPipeline pipeline = MakeTestPipeline(format, expectedData);
379         dawn::Buffer vertexBuffer =
380             utils::CreateBufferFromData(device, vertex.data(), vertex.size() * sizeof(VertexType),
381                                         dawn::BufferUsageBit::Vertex);
382         uint64_t zeroOffset = 0;
383         dawn::CommandEncoder encoder = device.CreateCommandEncoder();
384         {
385             dawn::RenderPassEncoder pass = encoder.BeginRenderPass(&renderPass.renderPassInfo);
386             pass.SetPipeline(pipeline);
387             pass.SetVertexBuffers(0, 1, &vertexBuffer, &zeroOffset);
388             pass.Draw(3, 1, 0, 0);
389             pass.EndPass();
390         }
391 
392         dawn::CommandBuffer commands = encoder.Finish();
393         queue.Submit(1, &commands);
394 
395         EXPECT_PIXEL_RGBA8_EQ(RGBA8(0, 255, 0, 255), renderPass.color, 0, 0);
396     }
397 };
398 
TEST_P(VertexFormatTest,UChar2)399 TEST_P(VertexFormatTest, UChar2) {
400     std::vector<uint8_t> vertexData = {
401         std::numeric_limits<uint8_t>::max(),
402         0,
403         0,  // padding two bytes for stride
404         0,
405         std::numeric_limits<uint8_t>::min(),
406         2,
407         0,
408         0,  // padding two bytes for stride
409         200,
410         201,
411         0,
412         0  // padding two bytes for buffer copy
413     };
414 
415     std::vector<uint8_t> expectedData = {
416         std::numeric_limits<uint8_t>::max(), 0, std::numeric_limits<uint8_t>::min(), 2, 200, 201,
417     };
418 
419     DoVertexFormatTest(dawn::VertexFormat::UChar2, vertexData, expectedData);
420 }
421 
TEST_P(VertexFormatTest,UChar4)422 TEST_P(VertexFormatTest, UChar4) {
423     std::vector<uint8_t> vertexData = {
424         std::numeric_limits<uint8_t>::max(),
425         0,
426         1,
427         2,
428         std::numeric_limits<uint8_t>::min(),
429         2,
430         3,
431         4,
432         200,
433         201,
434         202,
435         203,
436     };
437 
438     DoVertexFormatTest(dawn::VertexFormat::UChar4, vertexData, vertexData);
439 }
440 
TEST_P(VertexFormatTest,Char2)441 TEST_P(VertexFormatTest, Char2) {
442     std::vector<int8_t> vertexData = {
443         std::numeric_limits<int8_t>::max(),
444         0,
445         0,  // padding two bytes for stride
446         0,
447         std::numeric_limits<int8_t>::min(),
448         -2,
449         0,  // padding two bytes for stride
450         0,
451         120,
452         -121,
453         0,
454         0  // padding two bytes for buffer copy
455     };
456 
457     std::vector<int8_t> expectedData = {
458         std::numeric_limits<int8_t>::max(), 0, std::numeric_limits<int8_t>::min(), -2, 120, -121,
459     };
460 
461     DoVertexFormatTest(dawn::VertexFormat::Char2, vertexData, expectedData);
462 }
463 
TEST_P(VertexFormatTest,Char4)464 TEST_P(VertexFormatTest, Char4) {
465     std::vector<int8_t> vertexData = {
466         std::numeric_limits<int8_t>::max(),
467         0,
468         -1,
469         2,
470         std::numeric_limits<int8_t>::min(),
471         -2,
472         3,
473         4,
474         120,
475         -121,
476         122,
477         -123,
478     };
479 
480     DoVertexFormatTest(dawn::VertexFormat::Char4, vertexData, vertexData);
481 }
482 
TEST_P(VertexFormatTest,UChar2Norm)483 TEST_P(VertexFormatTest, UChar2Norm) {
484     std::vector<uint8_t> vertexData = {
485         std::numeric_limits<uint8_t>::max(),
486         std::numeric_limits<uint8_t>::min(),
487         0,  // padding two bytes for stride
488         0,
489         std::numeric_limits<uint8_t>::max() / 2u,
490         std::numeric_limits<uint8_t>::min() / 2u,
491         0,  // padding two bytes for stride
492         0,
493         200,
494         201,
495         0,
496         0  // padding two bytes for buffer copy
497     };
498 
499     std::vector<uint8_t> expectedData = {std::numeric_limits<uint8_t>::max(),
500                                          std::numeric_limits<uint8_t>::min(),
501                                          std::numeric_limits<uint8_t>::max() / 2u,
502                                          std::numeric_limits<uint8_t>::min() / 2u,
503                                          200,
504                                          201};
505 
506     DoVertexFormatTest(dawn::VertexFormat::UChar2Norm, vertexData, expectedData);
507 }
508 
TEST_P(VertexFormatTest,UChar4Norm)509 TEST_P(VertexFormatTest, UChar4Norm) {
510     std::vector<uint8_t> vertexData = {std::numeric_limits<uint8_t>::max(),
511                                        std::numeric_limits<uint8_t>::min(),
512                                        0,
513                                        0,
514                                        std::numeric_limits<uint8_t>::max() / 2u,
515                                        std::numeric_limits<uint8_t>::min() / 2u,
516                                        0,
517                                        0,
518                                        200,
519                                        201,
520                                        202,
521                                        203};
522 
523     DoVertexFormatTest(dawn::VertexFormat::UChar4Norm, vertexData, vertexData);
524 }
525 
TEST_P(VertexFormatTest,Char2Norm)526 TEST_P(VertexFormatTest, Char2Norm) {
527     std::vector<int8_t> vertexData = {
528         std::numeric_limits<int8_t>::max(),
529         std::numeric_limits<int8_t>::min(),
530         0,  // padding two bytes for stride
531         0,
532         std::numeric_limits<int8_t>::max() / 2,
533         std::numeric_limits<int8_t>::min() / 2,
534         0,  // padding two bytes for stride
535         0,
536         120,
537         -121,
538         0,
539         0  // padding two bytes for buffer copy
540     };
541 
542     std::vector<int8_t> expectedData = {
543         std::numeric_limits<int8_t>::max(),
544         std::numeric_limits<int8_t>::min(),
545         std::numeric_limits<int8_t>::max() / 2,
546         std::numeric_limits<int8_t>::min() / 2,
547         120,
548         -121,
549     };
550 
551     DoVertexFormatTest(dawn::VertexFormat::Char2Norm, vertexData, expectedData);
552 }
553 
TEST_P(VertexFormatTest,Char4Norm)554 TEST_P(VertexFormatTest, Char4Norm) {
555     std::vector<int8_t> vertexData = {std::numeric_limits<int8_t>::max(),
556                                       std::numeric_limits<int8_t>::min(),
557                                       0,
558                                       0,
559                                       std::numeric_limits<int8_t>::max() / 2,
560                                       std::numeric_limits<int8_t>::min() / 2,
561                                       -2,
562                                       2,
563                                       120,
564                                       -120,
565                                       102,
566                                       -123};
567 
568     DoVertexFormatTest(dawn::VertexFormat::Char4Norm, vertexData, vertexData);
569 }
570 
TEST_P(VertexFormatTest,UShort2)571 TEST_P(VertexFormatTest, UShort2) {
572     std::vector<uint16_t> vertexData = {std::numeric_limits<uint16_t>::max(),
573                                         0,
574                                         std::numeric_limits<uint16_t>::min(),
575                                         2,
576                                         65432,
577                                         4890};
578 
579     DoVertexFormatTest(dawn::VertexFormat::UShort2, vertexData, vertexData);
580 }
581 
TEST_P(VertexFormatTest,UShort4)582 TEST_P(VertexFormatTest, UShort4) {
583     std::vector<uint16_t> vertexData = {
584         std::numeric_limits<uint16_t>::max(),
585         std::numeric_limits<uint8_t>::max(),
586         1,
587         2,
588         std::numeric_limits<uint16_t>::min(),
589         2,
590         3,
591         4,
592         65520,
593         65521,
594         3435,
595         3467,
596     };
597 
598     DoVertexFormatTest(dawn::VertexFormat::UShort4, vertexData, vertexData);
599 }
600 
TEST_P(VertexFormatTest,Short2)601 TEST_P(VertexFormatTest, Short2) {
602     std::vector<int16_t> vertexData = {std::numeric_limits<int16_t>::max(),
603                                        0,
604                                        std::numeric_limits<int16_t>::min(),
605                                        -2,
606                                        3876,
607                                        -3948};
608 
609     DoVertexFormatTest(dawn::VertexFormat::Short2, vertexData, vertexData);
610 }
611 
TEST_P(VertexFormatTest,Short4)612 TEST_P(VertexFormatTest, Short4) {
613     std::vector<int16_t> vertexData = {
614         std::numeric_limits<int16_t>::max(),
615         0,
616         -1,
617         2,
618         std::numeric_limits<int16_t>::min(),
619         -2,
620         3,
621         4,
622         24567,
623         -23545,
624         4350,
625         -2987,
626     };
627 
628     DoVertexFormatTest(dawn::VertexFormat::Short4, vertexData, vertexData);
629 }
630 
TEST_P(VertexFormatTest,UShort2Norm)631 TEST_P(VertexFormatTest, UShort2Norm) {
632     std::vector<uint16_t> vertexData = {std::numeric_limits<uint16_t>::max(),
633                                         std::numeric_limits<uint16_t>::min(),
634                                         std::numeric_limits<uint16_t>::max() / 2u,
635                                         std::numeric_limits<uint16_t>::min() / 2u,
636                                         3456,
637                                         6543};
638 
639     DoVertexFormatTest(dawn::VertexFormat::UShort2Norm, vertexData, vertexData);
640 }
641 
TEST_P(VertexFormatTest,UShort4Norm)642 TEST_P(VertexFormatTest, UShort4Norm) {
643     std::vector<uint16_t> vertexData = {std::numeric_limits<uint16_t>::max(),
644                                         std::numeric_limits<uint16_t>::min(),
645                                         0,
646                                         0,
647                                         std::numeric_limits<uint16_t>::max() / 2u,
648                                         std::numeric_limits<uint16_t>::min() / 2u,
649                                         0,
650                                         0,
651                                         2987,
652                                         3055,
653                                         2987,
654                                         2987};
655 
656     DoVertexFormatTest(dawn::VertexFormat::UShort4Norm, vertexData, vertexData);
657 }
658 
TEST_P(VertexFormatTest,Short2Norm)659 TEST_P(VertexFormatTest, Short2Norm) {
660     std::vector<int16_t> vertexData = {std::numeric_limits<int16_t>::max(),
661                                        std::numeric_limits<int16_t>::min(),
662                                        std::numeric_limits<int16_t>::max() / 2,
663                                        std::numeric_limits<int16_t>::min() / 2,
664                                        4987,
665                                        -6789};
666 
667     DoVertexFormatTest(dawn::VertexFormat::Short2Norm, vertexData, vertexData);
668 }
669 
TEST_P(VertexFormatTest,Short4Norm)670 TEST_P(VertexFormatTest, Short4Norm) {
671     std::vector<int16_t> vertexData = {std::numeric_limits<int16_t>::max(),
672                                        std::numeric_limits<int16_t>::min(),
673                                        0,
674                                        0,
675                                        std::numeric_limits<int16_t>::max() / 2,
676                                        std::numeric_limits<int16_t>::min() / 2,
677                                        -2,
678                                        2,
679                                        2890,
680                                        -29011,
681                                        20432,
682                                        -2083};
683 
684     DoVertexFormatTest(dawn::VertexFormat::Short4Norm, vertexData, vertexData);
685 }
686 
TEST_P(VertexFormatTest,Half2)687 TEST_P(VertexFormatTest, Half2) {
688     std::vector<uint16_t> vertexData =
689         Float32ToFloat16(std::vector<float>({14.8f, -0.0f, 22.5f, 1.3f, +0.0f, -24.8f}));
690 
691     DoVertexFormatTest(dawn::VertexFormat::Half2, vertexData, vertexData);
692 }
693 
TEST_P(VertexFormatTest,Half4)694 TEST_P(VertexFormatTest, Half4) {
695     std::vector<uint16_t> vertexData = Float32ToFloat16(std::vector<float>(
696         {+0.0f, -16.8f, 18.2f, -0.0f, 12.5f, 1.3f, 14.8f, -12.4f, 22.5f, -48.8f, 47.4f, -24.8f}));
697 
698     DoVertexFormatTest(dawn::VertexFormat::Half4, vertexData, vertexData);
699 }
700 
TEST_P(VertexFormatTest,Float)701 TEST_P(VertexFormatTest, Float) {
702     std::vector<float> vertexData = {1.3f, +0.0f, -0.0f};
703 
704     DoVertexFormatTest(dawn::VertexFormat::Float, vertexData, vertexData);
705 
706     vertexData = std::vector<float>{+1.0f, -1.0f, 18.23f};
707 
708     DoVertexFormatTest(dawn::VertexFormat::Float, vertexData, vertexData);
709 }
710 
TEST_P(VertexFormatTest,Float2)711 TEST_P(VertexFormatTest, Float2) {
712     std::vector<float> vertexData = {18.23f, -0.0f, +0.0f, +1.0f, 1.3f, -1.0f};
713 
714     DoVertexFormatTest(dawn::VertexFormat::Float2, vertexData, vertexData);
715 }
716 
TEST_P(VertexFormatTest,Float3)717 TEST_P(VertexFormatTest, Float3) {
718     std::vector<float> vertexData = {
719         +0.0f, -1.0f, -0.0f, 1.0f, 1.3f, 99.45f, 23.6f, -81.2f, 55.0f,
720     };
721 
722     DoVertexFormatTest(dawn::VertexFormat::Float3, vertexData, vertexData);
723 }
724 
TEST_P(VertexFormatTest,Float4)725 TEST_P(VertexFormatTest, Float4) {
726     std::vector<float> vertexData = {
727         19.2f, -19.3f, +0.0f, 1.0f, -0.0f, 1.0f, 1.3f, -1.0f, 13.078f, 21.1965f, -1.1f, -1.2f,
728     };
729 
730     DoVertexFormatTest(dawn::VertexFormat::Float4, vertexData, vertexData);
731 }
732 
TEST_P(VertexFormatTest,UInt)733 TEST_P(VertexFormatTest, UInt) {
734     std::vector<uint32_t> vertexData = {std::numeric_limits<uint32_t>::max(),
735                                         std::numeric_limits<uint16_t>::max(),
736                                         std::numeric_limits<uint8_t>::max()};
737 
738     DoVertexFormatTest(dawn::VertexFormat::UInt, vertexData, vertexData);
739 }
740 
TEST_P(VertexFormatTest,UInt2)741 TEST_P(VertexFormatTest, UInt2) {
742     std::vector<uint32_t> vertexData = {std::numeric_limits<uint32_t>::max(), 32,
743                                         std::numeric_limits<uint16_t>::max(), 64,
744                                         std::numeric_limits<uint8_t>::max(),  128};
745 
746     DoVertexFormatTest(dawn::VertexFormat::UInt2, vertexData, vertexData);
747 }
748 
TEST_P(VertexFormatTest,UInt3)749 TEST_P(VertexFormatTest, UInt3) {
750     std::vector<uint32_t> vertexData = {std::numeric_limits<uint32_t>::max(), 32,   64,
751                                         std::numeric_limits<uint16_t>::max(), 164,  128,
752                                         std::numeric_limits<uint8_t>::max(),  1283, 256};
753 
754     DoVertexFormatTest(dawn::VertexFormat::UInt3, vertexData, vertexData);
755 }
756 
TEST_P(VertexFormatTest,UInt4)757 TEST_P(VertexFormatTest, UInt4) {
758     std::vector<uint32_t> vertexData = {std::numeric_limits<uint32_t>::max(), 32,   64,  5460,
759                                         std::numeric_limits<uint16_t>::max(), 164,  128, 0,
760                                         std::numeric_limits<uint8_t>::max(),  1283, 256, 4567};
761 
762     DoVertexFormatTest(dawn::VertexFormat::UInt4, vertexData, vertexData);
763 }
764 
TEST_P(VertexFormatTest,Int)765 TEST_P(VertexFormatTest, Int) {
766     std::vector<int32_t> vertexData = {std::numeric_limits<int32_t>::max(),
767                                        std::numeric_limits<int32_t>::min(),
768                                        std::numeric_limits<int8_t>::max()};
769 
770     DoVertexFormatTest(dawn::VertexFormat::Int, vertexData, vertexData);
771 }
772 
TEST_P(VertexFormatTest,Int2)773 TEST_P(VertexFormatTest, Int2) {
774     std::vector<int32_t> vertexData = {
775         std::numeric_limits<int32_t>::max(), std::numeric_limits<int32_t>::min(),
776         std::numeric_limits<int16_t>::max(), std::numeric_limits<int16_t>::min(),
777         std::numeric_limits<int8_t>::max(),  std::numeric_limits<int8_t>::min()};
778 
779     DoVertexFormatTest(dawn::VertexFormat::Int2, vertexData, vertexData);
780 }
781 
TEST_P(VertexFormatTest,Int3)782 TEST_P(VertexFormatTest, Int3) {
783     std::vector<int32_t> vertexData = {
784         std::numeric_limits<int32_t>::max(), std::numeric_limits<int32_t>::min(), 64,
785         std::numeric_limits<int16_t>::max(), std::numeric_limits<int16_t>::min(), 128,
786         std::numeric_limits<int8_t>::max(),  std::numeric_limits<int8_t>::min(),  256};
787 
788     DoVertexFormatTest(dawn::VertexFormat::Int3, vertexData, vertexData);
789 }
790 
TEST_P(VertexFormatTest,Int4)791 TEST_P(VertexFormatTest, Int4) {
792     std::vector<int32_t> vertexData = {
793         std::numeric_limits<int32_t>::max(), std::numeric_limits<int32_t>::min(), 64,   -5460,
794         std::numeric_limits<int16_t>::max(), std::numeric_limits<int16_t>::min(), -128, 0,
795         std::numeric_limits<int8_t>::max(),  std::numeric_limits<int8_t>::min(),  256,  -4567};
796 
797     DoVertexFormatTest(dawn::VertexFormat::Int4, vertexData, vertexData);
798 }
799 
800 DAWN_INSTANTIATE_TEST(VertexFormatTest, D3D12Backend, MetalBackend, OpenGLBackend, VulkanBackend);
801