/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "include/private/SkSLModifiers.h" #include "include/sksl/SkSLErrorReporter.h" #include "include/sksl/SkSLPosition.h" #include "src/sksl/SkSLBuiltinTypes.h" #include "src/sksl/SkSLContext.h" #include "src/sksl/SkSLMangler.h" #include "src/sksl/SkSLMemoryLayout.h" #include "src/sksl/SkSLUtil.h" #include "src/sksl/ir/SkSLType.h" #include "tests/Test.h" #include #include #include #include DEF_TEST(SkSLMemoryLayout140Test, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::ShaderCaps caps; SkSL::Mangler mangler; SkSL::Context context(errors, caps, mangler); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::k140); // basic types REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fInt)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 1 == layout.size(*context.fTypes.fBool)); REPORTER_ASSERT(r, 2 == layout.size(*context.fTypes.fBool2)); REPORTER_ASSERT(r, 3 == layout.size(*context.fTypes.fBool3)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fBool4)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fInt)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 1 == layout.alignment(*context.fTypes.fBool)); REPORTER_ASSERT(r, 2 == layout.alignment(*context.fTypes.fBool2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fBool3)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fBool4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x4)); // struct 1 std::vector fields1; fields1.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("a"), context.fTypes.fFloat3.get()); std::unique_ptr s1 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s1"), fields1); REPORTER_ASSERT(r, 16 == layout.size(*s1)); REPORTER_ASSERT(r, 16 == layout.alignment(*s1)); fields1.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("b"), context.fTypes.fFloat.get()); std::unique_ptr s2 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s2"), fields1); REPORTER_ASSERT(r, 16 == layout.size(*s2)); REPORTER_ASSERT(r, 16 == layout.alignment(*s2)); fields1.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("c"), context.fTypes.fBool.get()); std::unique_ptr s3 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s3"), fields1); REPORTER_ASSERT(r, 32 == layout.size(*s3)); REPORTER_ASSERT(r, 16 == layout.alignment(*s3)); // struct 2 std::vector fields2; fields2.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("a"), context.fTypes.fInt.get()); std::unique_ptr s4 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s4"), fields2); REPORTER_ASSERT(r, 16 == layout.size(*s4)); REPORTER_ASSERT(r, 16 == layout.alignment(*s4)); fields2.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("b"), context.fTypes.fFloat3.get()); std::unique_ptr s5 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s5"), fields2); REPORTER_ASSERT(r, 32 == layout.size(*s5)); REPORTER_ASSERT(r, 16 == layout.alignment(*s5)); // arrays std::unique_ptr array1 = SkSL::Type::MakeArrayType(std::string("float[4]"), *context.fTypes.fFloat, 4); REPORTER_ASSERT(r, 64 == layout.size(*array1)); REPORTER_ASSERT(r, 16 == layout.alignment(*array1)); REPORTER_ASSERT(r, 16 == layout.stride(*array1)); std::unique_ptr array2 = SkSL::Type::MakeArrayType(std::string("float4[4]"), *context.fTypes.fFloat4, 4); REPORTER_ASSERT(r, 64 == layout.size(*array2)); REPORTER_ASSERT(r, 16 == layout.alignment(*array2)); REPORTER_ASSERT(r, 16 == layout.stride(*array2)); } DEF_TEST(SkSLMemoryLayout430Test, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::ShaderCaps caps; SkSL::Mangler mangler; SkSL::Context context(errors, caps, mangler); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::k430); // basic types REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fInt)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 1 == layout.size(*context.fTypes.fBool)); REPORTER_ASSERT(r, 2 == layout.size(*context.fTypes.fBool2)); REPORTER_ASSERT(r, 3 == layout.size(*context.fTypes.fBool3)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fBool4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fInt)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 1 == layout.alignment(*context.fTypes.fBool)); REPORTER_ASSERT(r, 2 == layout.alignment(*context.fTypes.fBool2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fBool3)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fBool4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x4)); // struct 1 std::vector fields1; fields1.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("a"), context.fTypes.fFloat3.get()); std::unique_ptr s1 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s1"), fields1); REPORTER_ASSERT(r, 16 == layout.size(*s1)); REPORTER_ASSERT(r, 16 == layout.alignment(*s1)); fields1.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("b"), context.fTypes.fFloat.get()); std::unique_ptr s2 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s2"), fields1); REPORTER_ASSERT(r, 16 == layout.size(*s2)); REPORTER_ASSERT(r, 16 == layout.alignment(*s2)); fields1.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("c"), context.fTypes.fBool.get()); std::unique_ptr s3 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s3"), fields1); REPORTER_ASSERT(r, 32 == layout.size(*s3)); REPORTER_ASSERT(r, 16 == layout.alignment(*s3)); // struct 2 std::vector fields2; fields2.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("a"), context.fTypes.fInt.get()); std::unique_ptr s4 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s4"), fields2); REPORTER_ASSERT(r, 4 == layout.size(*s4)); REPORTER_ASSERT(r, 4 == layout.alignment(*s4)); fields2.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("b"), context.fTypes.fFloat3.get()); std::unique_ptr s5 = SkSL::Type::MakeStructType(SkSL::Position(), std::string("s5"), fields2); REPORTER_ASSERT(r, 32 == layout.size(*s5)); REPORTER_ASSERT(r, 16 == layout.alignment(*s5)); // arrays std::unique_ptr array1 = SkSL::Type::MakeArrayType(std::string("float[4]"), *context.fTypes.fFloat, 4); REPORTER_ASSERT(r, 16 == layout.size(*array1)); REPORTER_ASSERT(r, 4 == layout.alignment(*array1)); REPORTER_ASSERT(r, 4 == layout.stride(*array1)); std::unique_ptr array2 = SkSL::Type::MakeArrayType(std::string("float4[4]"), *context.fTypes.fFloat4, 4); REPORTER_ASSERT(r, 64 == layout.size(*array2)); REPORTER_ASSERT(r, 16 == layout.alignment(*array2)); REPORTER_ASSERT(r, 16 == layout.stride(*array2)); } DEF_TEST(SkSLMemoryLayoutWGSLUniformTest, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::ShaderCaps caps; SkSL::Mangler mangler; SkSL::Context context(errors, caps, mangler); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform); // The values here are taken from https://www.w3.org/TR/WGSL/#alignment-and-size, table titled // "Alignment and size for host-shareable types". // scalars (i32, u32, f32, f16) REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fInt)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fUInt)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 2 == layout.size(*context.fTypes.fHalf)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fUInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 2 == layout.alignment(*context.fTypes.fHalf)); // vec2, T: i32, u32, f32, f16 REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fUInt2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fHalf2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fUInt2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf2)); // vec3, T: i32, u32, f32, f16 REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fUInt3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 6 == layout.size(*context.fTypes.fHalf3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUInt3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3)); // vec4, T: i32, u32, f32, f16 REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fUInt4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fHalf4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUInt4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4)); // mat2x2, mat2x2 REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 4 == layout.stride(*context.fTypes.fHalf2x2)); // mat3x2, mat3x2 REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 4 == layout.stride(*context.fTypes.fHalf3x2)); // mat4x2, mat4x2 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 4 == layout.stride(*context.fTypes.fHalf4x2)); // mat2x3, mat2x3 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf2x3)); // mat3x3, mat3x3 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf3x3)); // mat4x3, mat4x3 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf4x3)); // mat2x4, mat2x4 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf2x4)); // mat3x4, mat3x4 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf3x4)); // mat4x4, mat4x4 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf4x4)); // bool is not a host-shareable type and returns 0 for WGSL. REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool)); REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool2)); REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool3)); REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool4)); // Arrays // array { auto array = SkSL::Type::MakeArrayType("float[4]", *context.fTypes.fFloat, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array { auto array = SkSL::Type::MakeArrayType("half[4]", *context.fTypes.fHalf, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("float2[4]", *context.fTypes.fFloat2, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("float3[4]", *context.fTypes.fFloat3, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("float4[4]", *context.fTypes.fFloat4, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("mat3[4]", *context.fTypes.fFloat3x3, 4); REPORTER_ASSERT(r, 192 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 48 == layout.stride(*array)); } // Structs A and B from example in https://www.w3.org/TR/WGSL/#structure-member-layout, with // offsets adjusted for uniform address space constraints. // // struct A { // align(roundUp(16, 8)) size(roundUp(16, 24)) // u: f32, // offset(0) align(4) size(4) // v: f32, // offset(4) align(4) size(4) // w: vec2, // offset(8) align(8) size(8) // x: f32 // offset(16) align(4) size(4) // // padding // offset(20) size(12) // } std::vector fields; fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("u"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("v"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("v"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("w"), context.fTypes.fFloat.get()); std::unique_ptr structA = SkSL::Type::MakeStructType(SkSL::Position(), std::string_view("A"), std::move(fields)); REPORTER_ASSERT(r, 32 == layout.size(*structA)); REPORTER_ASSERT(r, 16 == layout.alignment(*structA)); fields = {}; // struct B { // align(16) size(208) // a: vec2, // offset(0) align(8) size(8) // // padding // offset(8) size(8) // b: vec3, // offset(16) align(16) size(12) // c: f32, // offset(28) align(4) size(4) // d: f32, // offset(32) align(4) size(4) // // padding // offset(36) size(12) // e: A, // offset(48) align(16) size(32) // f: vec3, // offset(80) align(16) size(12) // // padding // offset(92) size(4) // g: array, // offset(96) align(16) size(96) // h: i32 // offset(192) align(4) size(4) // // padding // offset(196) size(12) // } fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("a"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("b"), context.fTypes.fFloat3.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("c"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("d"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("e"), structA.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("f"), context.fTypes.fFloat3.get()); auto array = SkSL::Type::MakeArrayType("A[3]", *structA, 3); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("g"), array.get()); fields.emplace_back( SkSL::Position(), SkSL::Modifiers(), std::string_view("h"), context.fTypes.fInt.get()); std::unique_ptr structB = SkSL::Type::MakeStructType(SkSL::Position(), std::string_view("B"), std::move(fields)); REPORTER_ASSERT(r, 208 == layout.size(*structB)); REPORTER_ASSERT(r, 16 == layout.alignment(*structB)); } DEF_TEST(SkSLMemoryLayoutWGSLStorageTest, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::ShaderCaps caps; SkSL::Mangler mangler; SkSL::Context context(errors, caps, mangler); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLStorage); // The values here are taken from https://www.w3.org/TR/WGSL/#alignment-and-size, table titled // "Alignment and size for host-shareable types". // scalars (i32, u32, f32, f16) REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fInt)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fUInt)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 2 == layout.size(*context.fTypes.fHalf)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fUInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 2 == layout.alignment(*context.fTypes.fHalf)); // vec2, T: i32, u32, f32, f16 REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fUInt2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fHalf2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fInt2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fUInt2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf2)); // vec3, T: i32, u32, f32, f16 REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fUInt3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 6 == layout.size(*context.fTypes.fHalf3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUInt3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3)); // vec4, T: i32, u32, f32, f16 REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fUInt4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fHalf4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fInt4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUInt4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4)); // mat2x2, mat2x2 REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 4 == layout.stride(*context.fTypes.fHalf2x2)); // mat3x2, mat3x2 REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 4 == layout.stride(*context.fTypes.fHalf3x2)); // mat4x2, mat4x2 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 4 == layout.stride(*context.fTypes.fHalf4x2)); // mat2x3, mat2x3 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf2x3)); // mat3x3, mat3x3 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf3x3)); // mat4x3, mat4x3 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf4x3)); // mat2x4, mat2x4 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf2x4)); // mat3x4, mat3x4 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf3x4)); // mat4x4, mat4x4 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf4x4)); // bool is not a host-shareable type and returns 0 for WGSL. REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool)); REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool2)); REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool3)); REPORTER_ASSERT(r, 0 == layout.size(*context.fTypes.fBool4)); // Arrays // array { auto array = SkSL::Type::MakeArrayType("float[4]", *context.fTypes.fFloat, 4); REPORTER_ASSERT(r, 16 == layout.size(*array)); REPORTER_ASSERT(r, 4 == layout.alignment(*array)); REPORTER_ASSERT(r, 4 == layout.stride(*array)); } // array { auto array = SkSL::Type::MakeArrayType("half[4]", *context.fTypes.fHalf, 4); REPORTER_ASSERT(r, 8 == layout.size(*array)); REPORTER_ASSERT(r, 2 == layout.alignment(*array)); REPORTER_ASSERT(r, 2 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("float2[4]", *context.fTypes.fFloat2, 4); REPORTER_ASSERT(r, 32 == layout.size(*array)); REPORTER_ASSERT(r, 8 == layout.alignment(*array)); REPORTER_ASSERT(r, 8 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("float3[4]", *context.fTypes.fFloat3, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("float4[4]", *context.fTypes.fFloat4, 4); REPORTER_ASSERT(r, 64 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 16 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType("mat3[4]", *context.fTypes.fFloat3x3, 4); REPORTER_ASSERT(r, 192 == layout.size(*array)); REPORTER_ASSERT(r, 16 == layout.alignment(*array)); REPORTER_ASSERT(r, 48 == layout.stride(*array)); } // Structs A and B from example in https://www.w3.org/TR/WGSL/#structure-member-layout // // struct A { // align(8) size(24) // u: f32, // offset(0) align(4) size(4) // v: f32, // offset(4) align(4) size(4) // w: vec2, // offset(8) align(8) size(8) // x: f32 // offset(16) align(4) size(4) // // padding // offset(20) size(4) // } std::vector fields; fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("u"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("v"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("v"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("w"), context.fTypes.fFloat.get()); std::unique_ptr structA = SkSL::Type::MakeStructType(SkSL::Position(), std::string_view("A"), std::move(fields)); REPORTER_ASSERT(r, 24 == layout.size(*structA)); REPORTER_ASSERT(r, 8 == layout.alignment(*structA)); fields = {}; // struct B { // align(16) size(160) // a: vec2, // offset(0) align(8) size(8) // // padding // offset(8) size(8) // b: vec3, // offset(16) align(16) size(12) // c: f32, // offset(28) align(4) size(4) // d: f32, // offset(32) align(4) size(4) // // padding // offset(36) size(4) // e: A, // offset(40) align(8) size(24) // f: vec3, // offset(64) align(16) size(12) // // padding // offset(76) size(4) // g: array, // offset(80) align(16) size(72) // h: i32 // offset(152) align(4) size(4) // // padding // offset(156) size(4) // } fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("a"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("b"), context.fTypes.fFloat3.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("c"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("d"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("e"), structA.get()); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("f"), context.fTypes.fFloat3.get()); auto array = SkSL::Type::MakeArrayType("A[3]", *structA, 3); fields.emplace_back(SkSL::Position(), SkSL::Modifiers(), std::string_view("g"), array.get()); fields.emplace_back( SkSL::Position(), SkSL::Modifiers(), std::string_view("h"), context.fTypes.fInt.get()); std::unique_ptr structB = SkSL::Type::MakeStructType(SkSL::Position(), std::string_view("B"), std::move(fields)); REPORTER_ASSERT(r, 160 == layout.size(*structB)); REPORTER_ASSERT(r, 16 == layout.alignment(*structB)); } DEF_TEST(SkSLMemoryLayoutWGSLUnsupportedTypesTest, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::ShaderCaps caps; SkSL::Mangler mangler; SkSL::Context context(errors, caps, mangler); auto testArray = SkSL::Type::MakeArrayType("bool[3]", *context.fTypes.fBool, 3); std::vector fields; fields.emplace_back( SkSL::Position(), SkSL::Modifiers(), std::string_view("foo"), testArray.get()); auto testStruct = SkSL::Type::MakeStructType( SkSL::Position(), std::string_view("Test"), std::move(fields)); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fBool)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fBool2)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fBool3)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fBool4)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fShort)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fShort2)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fShort3)); REPORTER_ASSERT(r, !layout.isSupported(*context.fTypes.fShort4)); REPORTER_ASSERT(r, !layout.isSupported(*testArray)); REPORTER_ASSERT(r, !layout.isSupported(*testStruct)); } DEF_TEST(SkSLMemoryLayoutWGSLSupportedTypesTest, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::ShaderCaps caps; SkSL::Mangler mangler; SkSL::Context context(errors, caps, mangler); auto testArray = SkSL::Type::MakeArrayType("float[3]", *context.fTypes.fFloat, 3); std::vector fields; fields.emplace_back( SkSL::Position(), SkSL::Modifiers(), std::string_view("foo"), testArray.get()); auto testStruct = SkSL::Type::MakeStructType( SkSL::Position(), std::string_view("Test"), std::move(fields)); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform); // scalars (i32, u32, f32, f16) REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fInt)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUInt)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf)); // vec2, T: i32, u32, f32, f16 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fInt2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUInt2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf2)); // vec3, T: i32, u32, f32, f16 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fInt3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUInt3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf3)); // vec4, T: i32, u32, f32, f16 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fInt4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUInt4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf4)); // mat2x2, mat2x2 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf2x2)); // mat3x2, mat3x2 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf3x2)); // mat4x2, mat4x2 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf4x2)); // mat2x3, mat2x3 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf2x3)); // mat3x3, mat3x3 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf3x3)); // mat4x3, mat4x3 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf4x3)); // mat2x4, mat2x4 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf2x4)); // mat3x4, mat3x4 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf3x4)); // mat4x4, mat4x4 REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fHalf4x4)); // arrays and structs REPORTER_ASSERT(r, layout.isSupported(*testArray)); REPORTER_ASSERT(r, layout.isSupported(*testStruct)); }