/* * 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/base/SkTArray.h" #include "src/sksl/SkSLBuiltinTypes.h" #include "src/sksl/SkSLContext.h" #include "src/sksl/SkSLErrorReporter.h" #include "src/sksl/SkSLMemoryLayout.h" #include "src/sksl/SkSLPosition.h" #include "src/sksl/SkSLProgramSettings.h" #include "src/sksl/SkSLUtil.h" #include "src/sksl/ir/SkSLLayout.h" #include "src/sksl/ir/SkSLModifierFlags.h" #include "src/sksl/ir/SkSLType.h" #include "tests/Test.h" #include #include #include #include using namespace skia_private; DEF_TEST(SkSLMemoryLayoutTest_std140, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::k140); SkSL::ProgramConfig config = {}; context.fConfig = &config; // 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.size(*context.fTypes.fAtomicUInt)); 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)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fAtomicUInt)); // struct 1 TArray fields1; fields1.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fFloat3.get()); std::unique_ptr s1 = SkSL::Type::MakeStructType(context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat.get()); std::unique_ptr s2 = SkSL::Type::MakeStructType(context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("c"), context.fTypes.fBool.get()); std::unique_ptr s3 = SkSL::Type::MakeStructType(context, SkSL::Position(), std::string("s3"), fields1); REPORTER_ASSERT(r, 32 == layout.size(*s3)); REPORTER_ASSERT(r, 16 == layout.alignment(*s3)); // struct 2 TArray fields2; fields2.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fInt.get()); std::unique_ptr s4 = SkSL::Type::MakeStructType(context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat3.get()); std::unique_ptr s5 = SkSL::Type::MakeStructType(context, 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(context, 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(context, 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(SkSLMemoryLayoutTest_std430, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::k430); SkSL::ProgramConfig config = {}; context.fConfig = &config; // 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.size(*context.fTypes.fAtomicUInt)); 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)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fAtomicUInt)); // struct 1 TArray fields1; fields1.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fFloat3.get()); std::unique_ptr s1 = SkSL::Type::MakeStructType(context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat.get()); std::unique_ptr s2 = SkSL::Type::MakeStructType(context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("c"), context.fTypes.fBool.get()); std::unique_ptr s3 = SkSL::Type::MakeStructType(context, SkSL::Position(), std::string("s3"), fields1); REPORTER_ASSERT(r, 32 == layout.size(*s3)); REPORTER_ASSERT(r, 16 == layout.alignment(*s3)); // struct 2 TArray fields2; fields2.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fInt.get()); std::unique_ptr s4 = SkSL::Type::MakeStructType(context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat3.get()); std::unique_ptr s5 = SkSL::Type::MakeStructType(context, 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(context, 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(context, 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(SkSLMemoryLayoutTest_WGSLUniform_Base, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform_Base); SkSL::ProgramConfig config = {}; context.fConfig = &config; // The values here are taken from https://www.w3.org/TR/WGSL/#alignment-and-size, table titled // "Alignment and size for host-shareable types". WGSL does not have an i16 type, so short and // unsigned-short integer types are treated as full-size integers in WGSL. // 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.fShort)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fUShort)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 4 == 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.fShort)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fUShort)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 4 == 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fUShort2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 8 == 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fUShort2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 8 == 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.fShort3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fUShort3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 12 == 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.fShort3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 16 == 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fUShort4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 16 == 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf4)); // mat2x2, mat2x2 REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf2x2)); // mat3x2, mat3x2 REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf3x2)); // mat4x2, mat4x2 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf4x2)); // mat2x3, mat2x3 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf2x3)); // mat3x3, mat3x3 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf3x3)); // mat4x3, mat4x3 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf4x3)); // mat2x4, mat2x4 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf2x4)); // mat3x4, mat3x4 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf3x4)); // mat4x4, mat4x4 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf4x4)); // atomic REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fAtomicUInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fAtomicUInt)); // 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(context, "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(context, "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(context, "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(context, "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(context, "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(context, "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) // } TArray fields; fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("u"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("v"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("w"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("x"), context.fTypes.fFloat.get()); std::unique_ptr structA = SkSL::Type::MakeStructType( context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat3.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("c"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("d"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("e"), structA.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("f"), context.fTypes.fFloat3.get()); auto array = SkSL::Type::MakeArrayType(context, "A[3]", *structA, 3); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("g"), array.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("h"), context.fTypes.fInt.get()); std::unique_ptr structB = SkSL::Type::MakeStructType( context, 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(SkSLMemoryLayoutTest_WGSLUniform_EnableF16, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform_EnableF16); SkSL::ProgramConfig config = {}; context.fConfig = &config; // The values here are taken from https://www.w3.org/TR/WGSL/#alignment-and-size, table titled // "Alignment and size for host-shareable types". WGSL does not have an i16 type, so short and // unsigned-short integer types are treated as full-size integers in WGSL. // 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.fShort)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fUShort)); 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.fShort)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fUShort)); 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fUShort2)); 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fUShort2)); 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.fShort3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fUShort3)); 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.fShort3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort3)); 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fUShort4)); 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort4)); 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)); // atomic REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fAtomicUInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fAtomicUInt)); // 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(context, "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(context, "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(context, "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(context, "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(context, "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(context, "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) // } TArray fields; fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("u"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("v"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("w"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("x"), context.fTypes.fFloat.get()); std::unique_ptr structA = SkSL::Type::MakeStructType( context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat3.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("c"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("d"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("e"), structA.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("f"), context.fTypes.fFloat3.get()); auto array = SkSL::Type::MakeArrayType(context, "A[3]", *structA, 3); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("g"), array.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("h"), context.fTypes.fInt.get()); std::unique_ptr structB = SkSL::Type::MakeStructType( context, 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(SkSLMemoryLayoutTest_WGSLStorage_Base, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLStorage_Base); SkSL::ProgramConfig config = {}; context.fConfig = &config; // 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.fShort)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fUShort)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 4 == 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.fShort)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fUShort)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fFloat)); REPORTER_ASSERT(r, 4 == 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fUShort2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 8 == 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fUShort2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2)); REPORTER_ASSERT(r, 8 == 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.fShort3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fUShort3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 12 == 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.fShort3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3)); REPORTER_ASSERT(r, 16 == 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fUShort4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 16 == 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf4)); // mat2x2, mat2x2 REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf2x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat2x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf2x2)); // mat3x2, mat3x2 REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 24 == layout.size(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf3x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat3x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf3x2)); // mat4x2, mat4x2 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fHalf4x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fFloat4x2)); REPORTER_ASSERT(r, 8 == layout.stride(*context.fTypes.fHalf4x2)); // mat2x3, mat2x3 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf2x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf2x3)); // mat3x3, mat3x3 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf3x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf3x3)); // mat4x3, mat4x3 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf4x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x3)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf4x3)); // mat2x4, mat2x4 REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 32 == layout.size(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf2x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat2x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf2x4)); // mat3x4, mat3x4 REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 48 == layout.size(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf3x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat3x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf3x4)); // mat4x4, mat4x4 REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 64 == layout.size(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fHalf4x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fFloat4x4)); REPORTER_ASSERT(r, 16 == layout.stride(*context.fTypes.fHalf4x4)); // atomic REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fAtomicUInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fAtomicUInt)); // 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(context, "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(context, "half[4]", *context.fTypes.fHalf, 4); REPORTER_ASSERT(r, 16 == layout.size(*array)); REPORTER_ASSERT(r, 4 == layout.alignment(*array)); REPORTER_ASSERT(r, 4 == layout.stride(*array)); } // array, 4> { auto array = SkSL::Type::MakeArrayType(context, "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(context, "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(context, "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(context, "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) // } TArray fields; fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("u"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("v"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("w"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("x"), context.fTypes.fFloat.get()); std::unique_ptr structA = SkSL::Type::MakeStructType( context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat3.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("c"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("d"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("e"), structA.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("f"), context.fTypes.fFloat3.get()); auto array = SkSL::Type::MakeArrayType(context, "A[3]", *structA, 3); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("g"), array.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("h"), context.fTypes.fInt.get()); std::unique_ptr structB = SkSL::Type::MakeStructType( context, 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(SkSLMemoryLayoutTest_WGSLStorage_EnableF16, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLStorage_EnableF16); SkSL::ProgramConfig config = {}; context.fConfig = &config; // 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.fShort)); REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fUShort)); 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.fShort)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fUShort)); 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.size(*context.fTypes.fUShort2)); 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.fShort2)); REPORTER_ASSERT(r, 8 == layout.alignment(*context.fTypes.fUShort2)); 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.fShort3)); REPORTER_ASSERT(r, 12 == layout.size(*context.fTypes.fUShort3)); 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.fShort3)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort3)); 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.size(*context.fTypes.fUShort4)); 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.fShort4)); REPORTER_ASSERT(r, 16 == layout.alignment(*context.fTypes.fUShort4)); 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)); // atomic REPORTER_ASSERT(r, 4 == layout.size(*context.fTypes.fAtomicUInt)); REPORTER_ASSERT(r, 4 == layout.alignment(*context.fTypes.fAtomicUInt)); // 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(context, "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(context, "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(context, "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(context, "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(context, "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(context, "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) // } TArray fields; fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("u"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("v"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("w"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("x"), context.fTypes.fFloat.get()); std::unique_ptr structA = SkSL::Type::MakeStructType( context, 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::Layout(), SkSL::ModifierFlag::kNone, std::string_view("a"), context.fTypes.fFloat2.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("b"), context.fTypes.fFloat3.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("c"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("d"), context.fTypes.fFloat.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("e"), structA.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("f"), context.fTypes.fFloat3.get()); auto array = SkSL::Type::MakeArrayType(context, "A[3]", *structA, 3); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("g"), array.get()); fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("h"), context.fTypes.fInt.get()); std::unique_ptr structB = SkSL::Type::MakeStructType( context, 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(SkSLMemoryLayoutTest_WGSLUnsupportedTypes, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::ProgramConfig config = {}; context.fConfig = &config; auto testArray = SkSL::Type::MakeArrayType(context, "bool[3]", *context.fTypes.fBool, 3); TArray fields; fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("foo"), testArray.get()); auto testStruct = SkSL::Type::MakeStructType( context, SkSL::Position(), std::string_view("Test"), std::move(fields)); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform_EnableF16); 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(*testArray)); REPORTER_ASSERT(r, !layout.isSupported(*testStruct)); } DEF_TEST(SkSLMemoryLayoutTest_WGSLSupportedTypes, r) { SkSL::TestingOnly_AbortErrorReporter errors; SkSL::BuiltinTypes types; SkSL::Context context(types, errors); SkSL::ProgramConfig config = {}; context.fConfig = &config; auto testArray = SkSL::Type::MakeArrayType(context, "float[3]", *context.fTypes.fFloat, 3); TArray fields; fields.emplace_back(SkSL::Position(), SkSL::Layout(), SkSL::ModifierFlag::kNone, std::string_view("foo"), testArray.get()); auto testStruct = SkSL::Type::MakeStructType( context, SkSL::Position(), std::string_view("Test"), std::move(fields)); SkSL::MemoryLayout layout(SkSL::MemoryLayout::Standard::kWGSLUniform_EnableF16); // 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.fShort)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUShort)); 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.fShort2)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUShort2)); 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.fShort3)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUShort3)); 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.fShort4)); REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fUShort4)); 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)); // atomic REPORTER_ASSERT(r, layout.isSupported(*context.fTypes.fAtomicUInt)); // arrays and structs REPORTER_ASSERT(r, layout.isSupported(*testArray)); REPORTER_ASSERT(r, layout.isSupported(*testStruct)); }