// Copyright 2021 The Tint Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "src/tint/transform/decompose_memory_access.h" #include "src/tint/transform/test_helper.h" namespace tint::transform { namespace { using DecomposeMemoryAccessTest = TransformTest; TEST_F(DecomposeMemoryAccessTest, ShouldRunEmptyModule) { auto* src = R"()"; EXPECT_FALSE(ShouldRun(src)); } TEST_F(DecomposeMemoryAccessTest, ShouldRunStorageBuffer) { auto* src = R"( struct Buffer { i : i32, }; @group(0) @binding(0) var sb : Buffer; )"; EXPECT_TRUE(ShouldRun(src)); } TEST_F(DecomposeMemoryAccessTest, ShouldRunUniformBuffer) { auto* src = R"( struct Buffer { i : i32, }; @group(0) @binding(0) var ub : Buffer; )"; EXPECT_TRUE(ShouldRun(src)); } TEST_F(DecomposeMemoryAccessTest, SB_BasicLoad) { auto* src = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { var a : i32 = sb.a; var b : u32 = sb.b; var c : f32 = sb.c; var d : vec2 = sb.d; var e : vec2 = sb.e; var f : vec2 = sb.f; var g : vec3 = sb.g; var h : vec3 = sb.h; var i : vec3 = sb.i; var j : vec4 = sb.j; var k : vec4 = sb.k; var l : vec4 = sb.l; var m : mat2x2 = sb.m; var n : mat2x3 = sb.n; var o : mat2x4 = sb.o; var p : mat3x2 = sb.p; var q : mat3x3 = sb.q; var r : mat3x4 = sb.r; var s : mat4x2 = sb.s; var t : mat4x3 = sb.t; var u : mat4x4 = sb.u; var v : array, 2> = sb.v; } )"; auto* expect = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_load_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_load_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @internal(intrinsic_load_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x2 { return mat2x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u))); } fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x3 { return mat2x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u))); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x4 { return mat2x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u))); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x2 { return mat3x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u))); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x3 { return mat3x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u))); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x4 { return mat3x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u))); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x2 { return mat4x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u)), tint_symbol_5(buffer, (offset + 24u))); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x3 { return mat4x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u)), tint_symbol_8(buffer, (offset + 48u))); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x4 { return mat4x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u)), tint_symbol_11(buffer, (offset + 48u))); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> array, 2u> { var arr : array, 2u>; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { arr[i_1] = tint_symbol_8(buffer, (offset + (i_1 * 16u))); } return arr; } @compute @workgroup_size(1) fn main() { var a : i32 = tint_symbol(&(sb), 0u); var b : u32 = tint_symbol_1(&(sb), 4u); var c : f32 = tint_symbol_2(&(sb), 8u); var d : vec2 = tint_symbol_3(&(sb), 16u); var e : vec2 = tint_symbol_4(&(sb), 24u); var f : vec2 = tint_symbol_5(&(sb), 32u); var g : vec3 = tint_symbol_6(&(sb), 48u); var h : vec3 = tint_symbol_7(&(sb), 64u); var i : vec3 = tint_symbol_8(&(sb), 80u); var j : vec4 = tint_symbol_9(&(sb), 96u); var k : vec4 = tint_symbol_10(&(sb), 112u); var l : vec4 = tint_symbol_11(&(sb), 128u); var m : mat2x2 = tint_symbol_12(&(sb), 144u); var n : mat2x3 = tint_symbol_13(&(sb), 160u); var o : mat2x4 = tint_symbol_14(&(sb), 192u); var p : mat3x2 = tint_symbol_15(&(sb), 224u); var q : mat3x3 = tint_symbol_16(&(sb), 256u); var r : mat3x4 = tint_symbol_17(&(sb), 304u); var s : mat4x2 = tint_symbol_18(&(sb), 352u); var t : mat4x3 = tint_symbol_19(&(sb), 384u); var u : mat4x4 = tint_symbol_20(&(sb), 448u); var v : array, 2> = tint_symbol_21(&(sb), 512u); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, SB_BasicLoad_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { var a : i32 = sb.a; var b : u32 = sb.b; var c : f32 = sb.c; var d : vec2 = sb.d; var e : vec2 = sb.e; var f : vec2 = sb.f; var g : vec3 = sb.g; var h : vec3 = sb.h; var i : vec3 = sb.i; var j : vec4 = sb.j; var k : vec4 = sb.k; var l : vec4 = sb.l; var m : mat2x2 = sb.m; var n : mat2x3 = sb.n; var o : mat2x4 = sb.o; var p : mat3x2 = sb.p; var q : mat3x3 = sb.q; var r : mat3x4 = sb.r; var s : mat4x2 = sb.s; var t : mat4x3 = sb.t; var u : mat4x4 = sb.u; var v : array, 2> = sb.v; } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; )"; auto* expect = R"( @internal(intrinsic_load_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_load_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @internal(intrinsic_load_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x2 { return mat2x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u))); } fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x3 { return mat2x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u))); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x4 { return mat2x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u))); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x2 { return mat3x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u))); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x3 { return mat3x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u))); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x4 { return mat3x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u))); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x2 { return mat4x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u)), tint_symbol_5(buffer, (offset + 24u))); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x3 { return mat4x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u)), tint_symbol_8(buffer, (offset + 48u))); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x4 { return mat4x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u)), tint_symbol_11(buffer, (offset + 48u))); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> array, 2u> { var arr : array, 2u>; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { arr[i_1] = tint_symbol_8(buffer, (offset + (i_1 * 16u))); } return arr; } @compute @workgroup_size(1) fn main() { var a : i32 = tint_symbol(&(sb), 0u); var b : u32 = tint_symbol_1(&(sb), 4u); var c : f32 = tint_symbol_2(&(sb), 8u); var d : vec2 = tint_symbol_3(&(sb), 16u); var e : vec2 = tint_symbol_4(&(sb), 24u); var f : vec2 = tint_symbol_5(&(sb), 32u); var g : vec3 = tint_symbol_6(&(sb), 48u); var h : vec3 = tint_symbol_7(&(sb), 64u); var i : vec3 = tint_symbol_8(&(sb), 80u); var j : vec4 = tint_symbol_9(&(sb), 96u); var k : vec4 = tint_symbol_10(&(sb), 112u); var l : vec4 = tint_symbol_11(&(sb), 128u); var m : mat2x2 = tint_symbol_12(&(sb), 144u); var n : mat2x3 = tint_symbol_13(&(sb), 160u); var o : mat2x4 = tint_symbol_14(&(sb), 192u); var p : mat3x2 = tint_symbol_15(&(sb), 224u); var q : mat3x3 = tint_symbol_16(&(sb), 256u); var r : mat3x4 = tint_symbol_17(&(sb), 304u); var s : mat4x2 = tint_symbol_18(&(sb), 352u); var t : mat4x3 = tint_symbol_19(&(sb), 384u); var u : mat4x4 = tint_symbol_20(&(sb), 448u); var v : array, 2> = tint_symbol_21(&(sb), 512u); } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, UB_BasicLoad) { auto* src = R"( struct UB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; @group(0) @binding(0) var ub : UB; @compute @workgroup_size(1) fn main() { var a : i32 = ub.a; var b : u32 = ub.b; var c : f32 = ub.c; var d : vec2 = ub.d; var e : vec2 = ub.e; var f : vec2 = ub.f; var g : vec3 = ub.g; var h : vec3 = ub.h; var i : vec3 = ub.i; var j : vec4 = ub.j; var k : vec4 = ub.k; var l : vec4 = ub.l; var m : mat2x2 = ub.m; var n : mat2x3 = ub.n; var o : mat2x4 = ub.o; var p : mat3x2 = ub.p; var q : mat3x3 = ub.q; var r : mat3x4 = ub.r; var s : mat4x2 = ub.s; var t : mat4x3 = ub.t; var u : mat4x4 = ub.u; var v : array, 2> = ub.v; } )"; auto* expect = R"( struct UB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } @group(0) @binding(0) var ub : UB; @internal(intrinsic_load_uniform_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_load_uniform_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_load_uniform_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @internal(intrinsic_load_uniform_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_uniform_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_uniform_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_uniform_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_uniform_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_uniform_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_uniform_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_uniform_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_uniform_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x2 { return mat2x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u))); } fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x3 { return mat2x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u))); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x4 { return mat2x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u))); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x2 { return mat3x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u))); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x3 { return mat3x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u))); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x4 { return mat3x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u))); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x2 { return mat4x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u)), tint_symbol_5(buffer, (offset + 24u))); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x3 { return mat4x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u)), tint_symbol_8(buffer, (offset + 48u))); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x4 { return mat4x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u)), tint_symbol_11(buffer, (offset + 48u))); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> array, 2u> { var arr : array, 2u>; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { arr[i_1] = tint_symbol_8(buffer, (offset + (i_1 * 16u))); } return arr; } @compute @workgroup_size(1) fn main() { var a : i32 = tint_symbol(&(ub), 0u); var b : u32 = tint_symbol_1(&(ub), 4u); var c : f32 = tint_symbol_2(&(ub), 8u); var d : vec2 = tint_symbol_3(&(ub), 16u); var e : vec2 = tint_symbol_4(&(ub), 24u); var f : vec2 = tint_symbol_5(&(ub), 32u); var g : vec3 = tint_symbol_6(&(ub), 48u); var h : vec3 = tint_symbol_7(&(ub), 64u); var i : vec3 = tint_symbol_8(&(ub), 80u); var j : vec4 = tint_symbol_9(&(ub), 96u); var k : vec4 = tint_symbol_10(&(ub), 112u); var l : vec4 = tint_symbol_11(&(ub), 128u); var m : mat2x2 = tint_symbol_12(&(ub), 144u); var n : mat2x3 = tint_symbol_13(&(ub), 160u); var o : mat2x4 = tint_symbol_14(&(ub), 192u); var p : mat3x2 = tint_symbol_15(&(ub), 224u); var q : mat3x3 = tint_symbol_16(&(ub), 256u); var r : mat3x4 = tint_symbol_17(&(ub), 304u); var s : mat4x2 = tint_symbol_18(&(ub), 352u); var t : mat4x3 = tint_symbol_19(&(ub), 384u); var u : mat4x4 = tint_symbol_20(&(ub), 448u); var v : array, 2> = tint_symbol_21(&(ub), 512u); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, UB_BasicLoad_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { var a : i32 = ub.a; var b : u32 = ub.b; var c : f32 = ub.c; var d : vec2 = ub.d; var e : vec2 = ub.e; var f : vec2 = ub.f; var g : vec3 = ub.g; var h : vec3 = ub.h; var i : vec3 = ub.i; var j : vec4 = ub.j; var k : vec4 = ub.k; var l : vec4 = ub.l; var m : mat2x2 = ub.m; var n : mat2x3 = ub.n; var o : mat2x4 = ub.o; var p : mat3x2 = ub.p; var q : mat3x3 = ub.q; var r : mat3x4 = ub.r; var s : mat4x2 = ub.s; var t : mat4x3 = ub.t; var u : mat4x4 = ub.u; var v : array, 2> = ub.v; } @group(0) @binding(0) var ub : UB; struct UB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; )"; auto* expect = R"( @internal(intrinsic_load_uniform_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_load_uniform_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_load_uniform_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @internal(intrinsic_load_uniform_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_uniform_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_uniform_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_uniform_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_uniform_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_uniform_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_uniform_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_uniform_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_uniform_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x2 { return mat2x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u))); } fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x3 { return mat2x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u))); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x4 { return mat2x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u))); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x2 { return mat3x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u))); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x3 { return mat3x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u))); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x4 { return mat3x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u))); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x2 { return mat4x2(tint_symbol_5(buffer, (offset + 0u)), tint_symbol_5(buffer, (offset + 8u)), tint_symbol_5(buffer, (offset + 16u)), tint_symbol_5(buffer, (offset + 24u))); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x3 { return mat4x3(tint_symbol_8(buffer, (offset + 0u)), tint_symbol_8(buffer, (offset + 16u)), tint_symbol_8(buffer, (offset + 32u)), tint_symbol_8(buffer, (offset + 48u))); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x4 { return mat4x4(tint_symbol_11(buffer, (offset + 0u)), tint_symbol_11(buffer, (offset + 16u)), tint_symbol_11(buffer, (offset + 32u)), tint_symbol_11(buffer, (offset + 48u))); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> array, 2u> { var arr : array, 2u>; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { arr[i_1] = tint_symbol_8(buffer, (offset + (i_1 * 16u))); } return arr; } @compute @workgroup_size(1) fn main() { var a : i32 = tint_symbol(&(ub), 0u); var b : u32 = tint_symbol_1(&(ub), 4u); var c : f32 = tint_symbol_2(&(ub), 8u); var d : vec2 = tint_symbol_3(&(ub), 16u); var e : vec2 = tint_symbol_4(&(ub), 24u); var f : vec2 = tint_symbol_5(&(ub), 32u); var g : vec3 = tint_symbol_6(&(ub), 48u); var h : vec3 = tint_symbol_7(&(ub), 64u); var i : vec3 = tint_symbol_8(&(ub), 80u); var j : vec4 = tint_symbol_9(&(ub), 96u); var k : vec4 = tint_symbol_10(&(ub), 112u); var l : vec4 = tint_symbol_11(&(ub), 128u); var m : mat2x2 = tint_symbol_12(&(ub), 144u); var n : mat2x3 = tint_symbol_13(&(ub), 160u); var o : mat2x4 = tint_symbol_14(&(ub), 192u); var p : mat3x2 = tint_symbol_15(&(ub), 224u); var q : mat3x3 = tint_symbol_16(&(ub), 256u); var r : mat3x4 = tint_symbol_17(&(ub), 304u); var s : mat4x2 = tint_symbol_18(&(ub), 352u); var t : mat4x3 = tint_symbol_19(&(ub), 384u); var u : mat4x4 = tint_symbol_20(&(ub), 448u); var v : array, 2> = tint_symbol_21(&(ub), 512u); } @group(0) @binding(0) var ub : UB; struct UB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, SB_BasicStore) { auto* src = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { sb.a = i32(); sb.b = u32(); sb.c = f32(); sb.d = vec2(); sb.e = vec2(); sb.f = vec2(); sb.g = vec3(); sb.h = vec3(); sb.i = vec3(); sb.j = vec4(); sb.k = vec4(); sb.l = vec4(); sb.m = mat2x2(); sb.n = mat2x3(); sb.o = mat2x4(); sb.p = mat3x2(); sb.q = mat3x3(); sb.r = mat3x4(); sb.s = mat4x2(); sb.t = mat4x3(); sb.u = mat4x4(); sb.v = array, 2>(); } )"; auto* expect = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_store_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : i32) @internal(intrinsic_store_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : u32) @internal(intrinsic_store_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : f32) @internal(intrinsic_store_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x2) { tint_symbol_5(buffer, (offset + 0u), value[0u]); tint_symbol_5(buffer, (offset + 8u), value[1u]); } fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x3) { tint_symbol_8(buffer, (offset + 0u), value[0u]); tint_symbol_8(buffer, (offset + 16u), value[1u]); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x4) { tint_symbol_11(buffer, (offset + 0u), value[0u]); tint_symbol_11(buffer, (offset + 16u), value[1u]); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x2) { tint_symbol_5(buffer, (offset + 0u), value[0u]); tint_symbol_5(buffer, (offset + 8u), value[1u]); tint_symbol_5(buffer, (offset + 16u), value[2u]); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x3) { tint_symbol_8(buffer, (offset + 0u), value[0u]); tint_symbol_8(buffer, (offset + 16u), value[1u]); tint_symbol_8(buffer, (offset + 32u), value[2u]); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x4) { tint_symbol_11(buffer, (offset + 0u), value[0u]); tint_symbol_11(buffer, (offset + 16u), value[1u]); tint_symbol_11(buffer, (offset + 32u), value[2u]); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x2) { tint_symbol_5(buffer, (offset + 0u), value[0u]); tint_symbol_5(buffer, (offset + 8u), value[1u]); tint_symbol_5(buffer, (offset + 16u), value[2u]); tint_symbol_5(buffer, (offset + 24u), value[3u]); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x3) { tint_symbol_8(buffer, (offset + 0u), value[0u]); tint_symbol_8(buffer, (offset + 16u), value[1u]); tint_symbol_8(buffer, (offset + 32u), value[2u]); tint_symbol_8(buffer, (offset + 48u), value[3u]); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x4) { tint_symbol_11(buffer, (offset + 0u), value[0u]); tint_symbol_11(buffer, (offset + 16u), value[1u]); tint_symbol_11(buffer, (offset + 32u), value[2u]); tint_symbol_11(buffer, (offset + 48u), value[3u]); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : array, 2u>) { var array = value; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { tint_symbol_8(buffer, (offset + (i_1 * 16u)), array[i_1]); } } @compute @workgroup_size(1) fn main() { tint_symbol(&(sb), 0u, i32()); tint_symbol_1(&(sb), 4u, u32()); tint_symbol_2(&(sb), 8u, f32()); tint_symbol_3(&(sb), 16u, vec2()); tint_symbol_4(&(sb), 24u, vec2()); tint_symbol_5(&(sb), 32u, vec2()); tint_symbol_6(&(sb), 48u, vec3()); tint_symbol_7(&(sb), 64u, vec3()); tint_symbol_8(&(sb), 80u, vec3()); tint_symbol_9(&(sb), 96u, vec4()); tint_symbol_10(&(sb), 112u, vec4()); tint_symbol_11(&(sb), 128u, vec4()); tint_symbol_12(&(sb), 144u, mat2x2()); tint_symbol_13(&(sb), 160u, mat2x3()); tint_symbol_14(&(sb), 192u, mat2x4()); tint_symbol_15(&(sb), 224u, mat3x2()); tint_symbol_16(&(sb), 256u, mat3x3()); tint_symbol_17(&(sb), 304u, mat3x4()); tint_symbol_18(&(sb), 352u, mat4x2()); tint_symbol_19(&(sb), 384u, mat4x3()); tint_symbol_20(&(sb), 448u, mat4x4()); tint_symbol_21(&(sb), 512u, array, 2>()); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, SB_BasicStore_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { sb.a = i32(); sb.b = u32(); sb.c = f32(); sb.d = vec2(); sb.e = vec2(); sb.f = vec2(); sb.g = vec3(); sb.h = vec3(); sb.i = vec3(); sb.j = vec4(); sb.k = vec4(); sb.l = vec4(); sb.m = mat2x2(); sb.n = mat2x3(); sb.o = mat2x4(); sb.p = mat3x2(); sb.q = mat3x3(); sb.r = mat3x4(); sb.s = mat4x2(); sb.t = mat4x3(); sb.u = mat4x4(); sb.v = array, 2>(); } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; )"; auto* expect = R"( @internal(intrinsic_store_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : i32) @internal(intrinsic_store_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : u32) @internal(intrinsic_store_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : f32) @internal(intrinsic_store_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x2) { tint_symbol_5(buffer, (offset + 0u), value[0u]); tint_symbol_5(buffer, (offset + 8u), value[1u]); } fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x3) { tint_symbol_8(buffer, (offset + 0u), value[0u]); tint_symbol_8(buffer, (offset + 16u), value[1u]); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x4) { tint_symbol_11(buffer, (offset + 0u), value[0u]); tint_symbol_11(buffer, (offset + 16u), value[1u]); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x2) { tint_symbol_5(buffer, (offset + 0u), value[0u]); tint_symbol_5(buffer, (offset + 8u), value[1u]); tint_symbol_5(buffer, (offset + 16u), value[2u]); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x3) { tint_symbol_8(buffer, (offset + 0u), value[0u]); tint_symbol_8(buffer, (offset + 16u), value[1u]); tint_symbol_8(buffer, (offset + 32u), value[2u]); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x4) { tint_symbol_11(buffer, (offset + 0u), value[0u]); tint_symbol_11(buffer, (offset + 16u), value[1u]); tint_symbol_11(buffer, (offset + 32u), value[2u]); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x2) { tint_symbol_5(buffer, (offset + 0u), value[0u]); tint_symbol_5(buffer, (offset + 8u), value[1u]); tint_symbol_5(buffer, (offset + 16u), value[2u]); tint_symbol_5(buffer, (offset + 24u), value[3u]); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x3) { tint_symbol_8(buffer, (offset + 0u), value[0u]); tint_symbol_8(buffer, (offset + 16u), value[1u]); tint_symbol_8(buffer, (offset + 32u), value[2u]); tint_symbol_8(buffer, (offset + 48u), value[3u]); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x4) { tint_symbol_11(buffer, (offset + 0u), value[0u]); tint_symbol_11(buffer, (offset + 16u), value[1u]); tint_symbol_11(buffer, (offset + 32u), value[2u]); tint_symbol_11(buffer, (offset + 48u), value[3u]); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : array, 2u>) { var array = value; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { tint_symbol_8(buffer, (offset + (i_1 * 16u)), array[i_1]); } } @compute @workgroup_size(1) fn main() { tint_symbol(&(sb), 0u, i32()); tint_symbol_1(&(sb), 4u, u32()); tint_symbol_2(&(sb), 8u, f32()); tint_symbol_3(&(sb), 16u, vec2()); tint_symbol_4(&(sb), 24u, vec2()); tint_symbol_5(&(sb), 32u, vec2()); tint_symbol_6(&(sb), 48u, vec3()); tint_symbol_7(&(sb), 64u, vec3()); tint_symbol_8(&(sb), 80u, vec3()); tint_symbol_9(&(sb), 96u, vec4()); tint_symbol_10(&(sb), 112u, vec4()); tint_symbol_11(&(sb), 128u, vec4()); tint_symbol_12(&(sb), 144u, mat2x2()); tint_symbol_13(&(sb), 160u, mat2x3()); tint_symbol_14(&(sb), 192u, mat2x4()); tint_symbol_15(&(sb), 224u, mat3x2()); tint_symbol_16(&(sb), 256u, mat3x3()); tint_symbol_17(&(sb), 304u, mat3x4()); tint_symbol_18(&(sb), 352u, mat4x2()); tint_symbol_19(&(sb), 384u, mat4x3()); tint_symbol_20(&(sb), 448u, mat4x4()); tint_symbol_21(&(sb), 512u, array, 2>()); } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, LoadStructure) { auto* src = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { var x : SB = sb; } )"; auto* expect = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_load_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_load_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @internal(intrinsic_load_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x2 { return mat2x2(tint_symbol_6(buffer, (offset + 0u)), tint_symbol_6(buffer, (offset + 8u))); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x3 { return mat2x3(tint_symbol_9(buffer, (offset + 0u)), tint_symbol_9(buffer, (offset + 16u))); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x4 { return mat2x4(tint_symbol_12(buffer, (offset + 0u)), tint_symbol_12(buffer, (offset + 16u))); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x2 { return mat3x2(tint_symbol_6(buffer, (offset + 0u)), tint_symbol_6(buffer, (offset + 8u)), tint_symbol_6(buffer, (offset + 16u))); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x3 { return mat3x3(tint_symbol_9(buffer, (offset + 0u)), tint_symbol_9(buffer, (offset + 16u)), tint_symbol_9(buffer, (offset + 32u))); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x4 { return mat3x4(tint_symbol_12(buffer, (offset + 0u)), tint_symbol_12(buffer, (offset + 16u)), tint_symbol_12(buffer, (offset + 32u))); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x2 { return mat4x2(tint_symbol_6(buffer, (offset + 0u)), tint_symbol_6(buffer, (offset + 8u)), tint_symbol_6(buffer, (offset + 16u)), tint_symbol_6(buffer, (offset + 24u))); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x3 { return mat4x3(tint_symbol_9(buffer, (offset + 0u)), tint_symbol_9(buffer, (offset + 16u)), tint_symbol_9(buffer, (offset + 32u)), tint_symbol_9(buffer, (offset + 48u))); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x4 { return mat4x4(tint_symbol_12(buffer, (offset + 0u)), tint_symbol_12(buffer, (offset + 16u)), tint_symbol_12(buffer, (offset + 32u)), tint_symbol_12(buffer, (offset + 48u))); } fn tint_symbol_22(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> array, 2u> { var arr : array, 2u>; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { arr[i_1] = tint_symbol_9(buffer, (offset + (i_1 * 16u))); } return arr; } fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> SB { return SB(tint_symbol_1(buffer, (offset + 0u)), tint_symbol_2(buffer, (offset + 4u)), tint_symbol_3(buffer, (offset + 8u)), tint_symbol_4(buffer, (offset + 16u)), tint_symbol_5(buffer, (offset + 24u)), tint_symbol_6(buffer, (offset + 32u)), tint_symbol_7(buffer, (offset + 48u)), tint_symbol_8(buffer, (offset + 64u)), tint_symbol_9(buffer, (offset + 80u)), tint_symbol_10(buffer, (offset + 96u)), tint_symbol_11(buffer, (offset + 112u)), tint_symbol_12(buffer, (offset + 128u)), tint_symbol_13(buffer, (offset + 144u)), tint_symbol_14(buffer, (offset + 160u)), tint_symbol_15(buffer, (offset + 192u)), tint_symbol_16(buffer, (offset + 224u)), tint_symbol_17(buffer, (offset + 256u)), tint_symbol_18(buffer, (offset + 304u)), tint_symbol_19(buffer, (offset + 352u)), tint_symbol_20(buffer, (offset + 384u)), tint_symbol_21(buffer, (offset + 448u)), tint_symbol_22(buffer, (offset + 512u))); } @compute @workgroup_size(1) fn main() { var x : SB = tint_symbol(&(sb), 0u); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, LoadStructure_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { var x : SB = sb; } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; )"; auto* expect = R"( @internal(intrinsic_load_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_load_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @internal(intrinsic_load_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec2 @internal(intrinsic_load_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec3 @internal(intrinsic_load_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 @internal(intrinsic_load_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> vec4 fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x2 { return mat2x2(tint_symbol_6(buffer, (offset + 0u)), tint_symbol_6(buffer, (offset + 8u))); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x3 { return mat2x3(tint_symbol_9(buffer, (offset + 0u)), tint_symbol_9(buffer, (offset + 16u))); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat2x4 { return mat2x4(tint_symbol_12(buffer, (offset + 0u)), tint_symbol_12(buffer, (offset + 16u))); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x2 { return mat3x2(tint_symbol_6(buffer, (offset + 0u)), tint_symbol_6(buffer, (offset + 8u)), tint_symbol_6(buffer, (offset + 16u))); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x3 { return mat3x3(tint_symbol_9(buffer, (offset + 0u)), tint_symbol_9(buffer, (offset + 16u)), tint_symbol_9(buffer, (offset + 32u))); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat3x4 { return mat3x4(tint_symbol_12(buffer, (offset + 0u)), tint_symbol_12(buffer, (offset + 16u)), tint_symbol_12(buffer, (offset + 32u))); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x2 { return mat4x2(tint_symbol_6(buffer, (offset + 0u)), tint_symbol_6(buffer, (offset + 8u)), tint_symbol_6(buffer, (offset + 16u)), tint_symbol_6(buffer, (offset + 24u))); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x3 { return mat4x3(tint_symbol_9(buffer, (offset + 0u)), tint_symbol_9(buffer, (offset + 16u)), tint_symbol_9(buffer, (offset + 32u)), tint_symbol_9(buffer, (offset + 48u))); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> mat4x4 { return mat4x4(tint_symbol_12(buffer, (offset + 0u)), tint_symbol_12(buffer, (offset + 16u)), tint_symbol_12(buffer, (offset + 32u)), tint_symbol_12(buffer, (offset + 48u))); } fn tint_symbol_22(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> array, 2u> { var arr : array, 2u>; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { arr[i_1] = tint_symbol_9(buffer, (offset + (i_1 * 16u))); } return arr; } fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> SB { return SB(tint_symbol_1(buffer, (offset + 0u)), tint_symbol_2(buffer, (offset + 4u)), tint_symbol_3(buffer, (offset + 8u)), tint_symbol_4(buffer, (offset + 16u)), tint_symbol_5(buffer, (offset + 24u)), tint_symbol_6(buffer, (offset + 32u)), tint_symbol_7(buffer, (offset + 48u)), tint_symbol_8(buffer, (offset + 64u)), tint_symbol_9(buffer, (offset + 80u)), tint_symbol_10(buffer, (offset + 96u)), tint_symbol_11(buffer, (offset + 112u)), tint_symbol_12(buffer, (offset + 128u)), tint_symbol_13(buffer, (offset + 144u)), tint_symbol_14(buffer, (offset + 160u)), tint_symbol_15(buffer, (offset + 192u)), tint_symbol_16(buffer, (offset + 224u)), tint_symbol_17(buffer, (offset + 256u)), tint_symbol_18(buffer, (offset + 304u)), tint_symbol_19(buffer, (offset + 352u)), tint_symbol_20(buffer, (offset + 384u)), tint_symbol_21(buffer, (offset + 448u)), tint_symbol_22(buffer, (offset + 512u))); } @compute @workgroup_size(1) fn main() { var x : SB = tint_symbol(&(sb), 0u); } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, StoreStructure) { auto* src = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { sb = SB(); } )"; auto* expect = R"( struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_store_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : i32) @internal(intrinsic_store_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : u32) @internal(intrinsic_store_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : f32) @internal(intrinsic_store_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x2) { tint_symbol_6(buffer, (offset + 0u), value[0u]); tint_symbol_6(buffer, (offset + 8u), value[1u]); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x3) { tint_symbol_9(buffer, (offset + 0u), value[0u]); tint_symbol_9(buffer, (offset + 16u), value[1u]); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x4) { tint_symbol_12(buffer, (offset + 0u), value[0u]); tint_symbol_12(buffer, (offset + 16u), value[1u]); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x2) { tint_symbol_6(buffer, (offset + 0u), value[0u]); tint_symbol_6(buffer, (offset + 8u), value[1u]); tint_symbol_6(buffer, (offset + 16u), value[2u]); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x3) { tint_symbol_9(buffer, (offset + 0u), value[0u]); tint_symbol_9(buffer, (offset + 16u), value[1u]); tint_symbol_9(buffer, (offset + 32u), value[2u]); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x4) { tint_symbol_12(buffer, (offset + 0u), value[0u]); tint_symbol_12(buffer, (offset + 16u), value[1u]); tint_symbol_12(buffer, (offset + 32u), value[2u]); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x2) { tint_symbol_6(buffer, (offset + 0u), value[0u]); tint_symbol_6(buffer, (offset + 8u), value[1u]); tint_symbol_6(buffer, (offset + 16u), value[2u]); tint_symbol_6(buffer, (offset + 24u), value[3u]); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x3) { tint_symbol_9(buffer, (offset + 0u), value[0u]); tint_symbol_9(buffer, (offset + 16u), value[1u]); tint_symbol_9(buffer, (offset + 32u), value[2u]); tint_symbol_9(buffer, (offset + 48u), value[3u]); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x4) { tint_symbol_12(buffer, (offset + 0u), value[0u]); tint_symbol_12(buffer, (offset + 16u), value[1u]); tint_symbol_12(buffer, (offset + 32u), value[2u]); tint_symbol_12(buffer, (offset + 48u), value[3u]); } fn tint_symbol_22(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : array, 2u>) { var array = value; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { tint_symbol_9(buffer, (offset + (i_1 * 16u)), array[i_1]); } } fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : SB) { tint_symbol_1(buffer, (offset + 0u), value.a); tint_symbol_2(buffer, (offset + 4u), value.b); tint_symbol_3(buffer, (offset + 8u), value.c); tint_symbol_4(buffer, (offset + 16u), value.d); tint_symbol_5(buffer, (offset + 24u), value.e); tint_symbol_6(buffer, (offset + 32u), value.f); tint_symbol_7(buffer, (offset + 48u), value.g); tint_symbol_8(buffer, (offset + 64u), value.h); tint_symbol_9(buffer, (offset + 80u), value.i); tint_symbol_10(buffer, (offset + 96u), value.j); tint_symbol_11(buffer, (offset + 112u), value.k); tint_symbol_12(buffer, (offset + 128u), value.l); tint_symbol_13(buffer, (offset + 144u), value.m); tint_symbol_14(buffer, (offset + 160u), value.n); tint_symbol_15(buffer, (offset + 192u), value.o); tint_symbol_16(buffer, (offset + 224u), value.p); tint_symbol_17(buffer, (offset + 256u), value.q); tint_symbol_18(buffer, (offset + 304u), value.r); tint_symbol_19(buffer, (offset + 352u), value.s); tint_symbol_20(buffer, (offset + 384u), value.t); tint_symbol_21(buffer, (offset + 448u), value.u); tint_symbol_22(buffer, (offset + 512u), value.v); } @compute @workgroup_size(1) fn main() { tint_symbol(&(sb), 0u, SB()); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, StoreStructure_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { sb = SB(); } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, }; )"; auto* expect = R"( @internal(intrinsic_store_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : i32) @internal(intrinsic_store_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_2(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : u32) @internal(intrinsic_store_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_3(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : f32) @internal(intrinsic_store_storage_vec2_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_4(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_5(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec2_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_6(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec2) @internal(intrinsic_store_storage_vec3_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_7(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_8(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec3_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_9(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec3) @internal(intrinsic_store_storage_vec4_i32) @internal(disable_validation__function_has_no_body) fn tint_symbol_10(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_u32) @internal(disable_validation__function_has_no_body) fn tint_symbol_11(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) @internal(intrinsic_store_storage_vec4_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol_12(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : vec4) fn tint_symbol_13(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x2) { tint_symbol_6(buffer, (offset + 0u), value[0u]); tint_symbol_6(buffer, (offset + 8u), value[1u]); } fn tint_symbol_14(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x3) { tint_symbol_9(buffer, (offset + 0u), value[0u]); tint_symbol_9(buffer, (offset + 16u), value[1u]); } fn tint_symbol_15(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat2x4) { tint_symbol_12(buffer, (offset + 0u), value[0u]); tint_symbol_12(buffer, (offset + 16u), value[1u]); } fn tint_symbol_16(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x2) { tint_symbol_6(buffer, (offset + 0u), value[0u]); tint_symbol_6(buffer, (offset + 8u), value[1u]); tint_symbol_6(buffer, (offset + 16u), value[2u]); } fn tint_symbol_17(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x3) { tint_symbol_9(buffer, (offset + 0u), value[0u]); tint_symbol_9(buffer, (offset + 16u), value[1u]); tint_symbol_9(buffer, (offset + 32u), value[2u]); } fn tint_symbol_18(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat3x4) { tint_symbol_12(buffer, (offset + 0u), value[0u]); tint_symbol_12(buffer, (offset + 16u), value[1u]); tint_symbol_12(buffer, (offset + 32u), value[2u]); } fn tint_symbol_19(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x2) { tint_symbol_6(buffer, (offset + 0u), value[0u]); tint_symbol_6(buffer, (offset + 8u), value[1u]); tint_symbol_6(buffer, (offset + 16u), value[2u]); tint_symbol_6(buffer, (offset + 24u), value[3u]); } fn tint_symbol_20(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x3) { tint_symbol_9(buffer, (offset + 0u), value[0u]); tint_symbol_9(buffer, (offset + 16u), value[1u]); tint_symbol_9(buffer, (offset + 32u), value[2u]); tint_symbol_9(buffer, (offset + 48u), value[3u]); } fn tint_symbol_21(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : mat4x4) { tint_symbol_12(buffer, (offset + 0u), value[0u]); tint_symbol_12(buffer, (offset + 16u), value[1u]); tint_symbol_12(buffer, (offset + 32u), value[2u]); tint_symbol_12(buffer, (offset + 48u), value[3u]); } fn tint_symbol_22(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : array, 2u>) { var array = value; for(var i_1 = 0u; (i_1 < 2u); i_1 = (i_1 + 1u)) { tint_symbol_9(buffer, (offset + (i_1 * 16u)), array[i_1]); } } fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, value : SB) { tint_symbol_1(buffer, (offset + 0u), value.a); tint_symbol_2(buffer, (offset + 4u), value.b); tint_symbol_3(buffer, (offset + 8u), value.c); tint_symbol_4(buffer, (offset + 16u), value.d); tint_symbol_5(buffer, (offset + 24u), value.e); tint_symbol_6(buffer, (offset + 32u), value.f); tint_symbol_7(buffer, (offset + 48u), value.g); tint_symbol_8(buffer, (offset + 64u), value.h); tint_symbol_9(buffer, (offset + 80u), value.i); tint_symbol_10(buffer, (offset + 96u), value.j); tint_symbol_11(buffer, (offset + 112u), value.k); tint_symbol_12(buffer, (offset + 128u), value.l); tint_symbol_13(buffer, (offset + 144u), value.m); tint_symbol_14(buffer, (offset + 160u), value.n); tint_symbol_15(buffer, (offset + 192u), value.o); tint_symbol_16(buffer, (offset + 224u), value.p); tint_symbol_17(buffer, (offset + 256u), value.q); tint_symbol_18(buffer, (offset + 304u), value.r); tint_symbol_19(buffer, (offset + 352u), value.s); tint_symbol_20(buffer, (offset + 384u), value.t); tint_symbol_21(buffer, (offset + 448u), value.u); tint_symbol_22(buffer, (offset + 512u), value.v); } @compute @workgroup_size(1) fn main() { tint_symbol(&(sb), 0u, SB()); } @group(0) @binding(0) var sb : SB; struct SB { a : i32, b : u32, c : f32, d : vec2, e : vec2, f : vec2, g : vec3, h : vec3, i : vec3, j : vec4, k : vec4, l : vec4, m : mat2x2, n : mat2x3, o : mat2x4, p : mat3x2, q : mat3x3, r : mat3x4, s : mat4x2, t : mat4x3, u : mat4x4, v : array, 2>, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, ComplexStaticAccessChain) { auto* src = R"( // sizeof(S1) == 32 // alignof(S1) == 16 struct S1 { a : i32, b : vec3, c : i32, }; // sizeof(S2) == 116 // alignof(S2) == 16 struct S2 { a : i32, b : array, c : i32, }; struct SB { @size(128) a : i32, b : array, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { var x : f32 = sb.b[4].b[1].b.z; } )"; // sb.b[4].b[1].b.z // ^ ^ ^ ^ ^ ^ // | | | | | | // 128 | |688 | 712 // | | | // 640 656 704 auto* expect = R"( struct S1 { a : i32, b : vec3, c : i32, } struct S2 { a : i32, b : array, c : i32, } struct SB { @size(128) a : i32, b : array, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @compute @workgroup_size(1) fn main() { var x : f32 = tint_symbol(&(sb), 712u); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, ComplexStaticAccessChain_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { var x : f32 = sb.b[4].b[1].b.z; } @group(0) @binding(0) var sb : SB; struct SB { @size(128) a : i32, b : array, }; struct S2 { a : i32, b : array, c : i32, }; struct S1 { a : i32, b : vec3, c : i32, }; )"; // sb.b[4].b[1].b.z // ^ ^ ^ ^ ^ ^ // | | | | | | // 128 | |688 | 712 // | | | // 640 656 704 auto* expect = R"( @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @compute @workgroup_size(1) fn main() { var x : f32 = tint_symbol(&(sb), 712u); } @group(0) @binding(0) var sb : SB; struct SB { @size(128) a : i32, b : array, } struct S2 { a : i32, b : array, c : i32, } struct S1 { a : i32, b : vec3, c : i32, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, ComplexDynamicAccessChain) { auto* src = R"( struct S1 { a : i32, b : vec3, c : i32, }; struct S2 { a : i32, b : array, c : i32, }; struct SB { @size(128) a : i32, b : array, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = sb.b[i].b[j].b[k]; } )"; auto* expect = R"( struct S1 { a : i32, b : vec3, c : i32, } struct S2 { a : i32, b : array, c : i32, } struct SB { @size(128) a : i32, b : array, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = tint_symbol(&(sb), (((((128u + (128u * u32(i))) + 16u) + (32u * j)) + 16u) + (4u * u32(k)))); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, ComplexDynamicAccessChain_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = sb.b[i].b[j].b[k]; } @group(0) @binding(0) var sb : SB; struct SB { @size(128) a : i32, b : array }; struct S2 { a : i32, b : array, c : i32, }; struct S1 { a : i32, b : vec3, c : i32, }; )"; auto* expect = R"( @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = tint_symbol(&(sb), (((((128u + (128u * u32(i))) + 16u) + (32u * j)) + 16u) + (4u * u32(k)))); } @group(0) @binding(0) var sb : SB; struct SB { @size(128) a : i32, b : array, } struct S2 { a : i32, b : array, c : i32, } struct S1 { a : i32, b : vec3, c : i32, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, ComplexDynamicAccessChainWithAliases) { auto* src = R"( struct S1 { a : i32, b : vec3, c : i32, }; type A1 = S1; type A1_Array = array; struct S2 { a : i32, b : A1_Array, c : i32, }; type A2 = S2; type A2_Array = array; struct SB { @size(128) a : i32, b : A2_Array, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = sb.b[i].b[j].b[k]; } )"; auto* expect = R"( struct S1 { a : i32, b : vec3, c : i32, } type A1 = S1; type A1_Array = array; struct S2 { a : i32, b : A1_Array, c : i32, } type A2 = S2; type A2_Array = array; struct SB { @size(128) a : i32, b : A2_Array, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = tint_symbol(&(sb), (((((128u + (128u * u32(i))) + 16u) + (32u * j)) + 16u) + (4u * u32(k)))); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, ComplexDynamicAccessChainWithAliases_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = sb.b[i].b[j].b[k]; } @group(0) @binding(0) var sb : SB; struct SB { @size(128) a : i32, b : A2_Array, }; type A2_Array = array; type A2 = S2; struct S2 { a : i32, b : A1_Array, c : i32, }; type A1 = S1; type A1_Array = array; struct S1 { a : i32, b : vec3, c : i32, }; )"; auto* expect = R"( @internal(intrinsic_load_storage_f32) @internal(disable_validation__function_has_no_body) fn tint_symbol(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> f32 @compute @workgroup_size(1) fn main() { var i : i32 = 4; var j : u32 = 1u; var k : i32 = 2; var x : f32 = tint_symbol(&(sb), (((((128u + (128u * u32(i))) + 16u) + (32u * j)) + 16u) + (4u * u32(k)))); } @group(0) @binding(0) var sb : SB; struct SB { @size(128) a : i32, b : A2_Array, } type A2_Array = array; type A2 = S2; struct S2 { a : i32, b : A1_Array, c : i32, } type A1 = S1; type A1_Array = array; struct S1 { a : i32, b : vec3, c : i32, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, StorageBufferAtomics) { auto* src = R"( struct SB { padding : vec4, a : atomic, b : atomic, }; @group(0) @binding(0) var sb : SB; @compute @workgroup_size(1) fn main() { atomicStore(&sb.a, 123); atomicLoad(&sb.a); atomicAdd(&sb.a, 123); atomicSub(&sb.a, 123); atomicMax(&sb.a, 123); atomicMin(&sb.a, 123); atomicAnd(&sb.a, 123); atomicOr(&sb.a, 123); atomicXor(&sb.a, 123); atomicExchange(&sb.a, 123); atomicCompareExchangeWeak(&sb.a, 123, 345); atomicStore(&sb.b, 123u); atomicLoad(&sb.b); atomicAdd(&sb.b, 123u); atomicSub(&sb.b, 123u); atomicMax(&sb.b, 123u); atomicMin(&sb.b, 123u); atomicAnd(&sb.b, 123u); atomicOr(&sb.b, 123u); atomicXor(&sb.b, 123u); atomicExchange(&sb.b, 123u); atomicCompareExchangeWeak(&sb.b, 123u, 345u); } )"; auto* expect = R"( struct SB { padding : vec4, a : atomic, b : atomic, } @group(0) @binding(0) var sb : SB; @internal(intrinsic_atomic_store_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicStore(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) @internal(intrinsic_atomic_load_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicLoad(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_atomic_add_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicAdd(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_sub_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicSub(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_max_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicMax(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_min_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicMin(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_and_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicAnd(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_or_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicOr(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_xor_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicXor(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_exchange_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicExchange(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 struct atomic_compare_exchange_weak_ret_type { old_value : i32, exchanged : bool, } @internal(intrinsic_atomic_compare_exchange_weak_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicCompareExchangeWeak(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32, param_2 : i32) -> atomic_compare_exchange_weak_ret_type @internal(intrinsic_atomic_store_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicStore_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) @internal(intrinsic_atomic_load_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicLoad_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_atomic_add_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicAdd_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_sub_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicSub_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_max_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicMax_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_min_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicMin_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_and_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicAnd_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_or_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicOr_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_xor_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicXor_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_exchange_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicExchange_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 struct atomic_compare_exchange_weak_ret_type_1 { old_value : u32, exchanged : bool, } @internal(intrinsic_atomic_compare_exchange_weak_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicCompareExchangeWeak_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32, param_2 : u32) -> atomic_compare_exchange_weak_ret_type_1 @compute @workgroup_size(1) fn main() { tint_atomicStore(&(sb), 16u, 123); tint_atomicLoad(&(sb), 16u); tint_atomicAdd(&(sb), 16u, 123); tint_atomicSub(&(sb), 16u, 123); tint_atomicMax(&(sb), 16u, 123); tint_atomicMin(&(sb), 16u, 123); tint_atomicAnd(&(sb), 16u, 123); tint_atomicOr(&(sb), 16u, 123); tint_atomicXor(&(sb), 16u, 123); tint_atomicExchange(&(sb), 16u, 123); tint_atomicCompareExchangeWeak(&(sb), 16u, 123, 345); tint_atomicStore_1(&(sb), 20u, 123u); tint_atomicLoad_1(&(sb), 20u); tint_atomicAdd_1(&(sb), 20u, 123u); tint_atomicSub_1(&(sb), 20u, 123u); tint_atomicMax_1(&(sb), 20u, 123u); tint_atomicMin_1(&(sb), 20u, 123u); tint_atomicAnd_1(&(sb), 20u, 123u); tint_atomicOr_1(&(sb), 20u, 123u); tint_atomicXor_1(&(sb), 20u, 123u); tint_atomicExchange_1(&(sb), 20u, 123u); tint_atomicCompareExchangeWeak_1(&(sb), 20u, 123u, 345u); } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, StorageBufferAtomics_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { atomicStore(&sb.a, 123); atomicLoad(&sb.a); atomicAdd(&sb.a, 123); atomicSub(&sb.a, 123); atomicMax(&sb.a, 123); atomicMin(&sb.a, 123); atomicAnd(&sb.a, 123); atomicOr(&sb.a, 123); atomicXor(&sb.a, 123); atomicExchange(&sb.a, 123); atomicCompareExchangeWeak(&sb.a, 123, 345); atomicStore(&sb.b, 123u); atomicLoad(&sb.b); atomicAdd(&sb.b, 123u); atomicSub(&sb.b, 123u); atomicMax(&sb.b, 123u); atomicMin(&sb.b, 123u); atomicAnd(&sb.b, 123u); atomicOr(&sb.b, 123u); atomicXor(&sb.b, 123u); atomicExchange(&sb.b, 123u); atomicCompareExchangeWeak(&sb.b, 123u, 345u); } @group(0) @binding(0) var sb : SB; struct SB { padding : vec4, a : atomic, b : atomic, }; )"; auto* expect = R"( @internal(intrinsic_atomic_store_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicStore(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) @internal(intrinsic_atomic_load_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicLoad(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> i32 @internal(intrinsic_atomic_add_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicAdd(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_sub_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicSub(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_max_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicMax(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_min_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicMin(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_and_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicAnd(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_or_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicOr(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_xor_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicXor(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 @internal(intrinsic_atomic_exchange_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicExchange(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32) -> i32 struct atomic_compare_exchange_weak_ret_type { old_value : i32, exchanged : bool, } @internal(intrinsic_atomic_compare_exchange_weak_storage_i32) @internal(disable_validation__function_has_no_body) fn tint_atomicCompareExchangeWeak(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : i32, param_2 : i32) -> atomic_compare_exchange_weak_ret_type @internal(intrinsic_atomic_store_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicStore_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) @internal(intrinsic_atomic_load_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicLoad_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32) -> u32 @internal(intrinsic_atomic_add_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicAdd_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_sub_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicSub_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_max_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicMax_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_min_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicMin_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_and_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicAnd_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_or_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicOr_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_xor_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicXor_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 @internal(intrinsic_atomic_exchange_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicExchange_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32) -> u32 struct atomic_compare_exchange_weak_ret_type_1 { old_value : u32, exchanged : bool, } @internal(intrinsic_atomic_compare_exchange_weak_storage_u32) @internal(disable_validation__function_has_no_body) fn tint_atomicCompareExchangeWeak_1(@internal(disable_validation__function_parameter) buffer : ptr, offset : u32, param_1 : u32, param_2 : u32) -> atomic_compare_exchange_weak_ret_type_1 @compute @workgroup_size(1) fn main() { tint_atomicStore(&(sb), 16u, 123); tint_atomicLoad(&(sb), 16u); tint_atomicAdd(&(sb), 16u, 123); tint_atomicSub(&(sb), 16u, 123); tint_atomicMax(&(sb), 16u, 123); tint_atomicMin(&(sb), 16u, 123); tint_atomicAnd(&(sb), 16u, 123); tint_atomicOr(&(sb), 16u, 123); tint_atomicXor(&(sb), 16u, 123); tint_atomicExchange(&(sb), 16u, 123); tint_atomicCompareExchangeWeak(&(sb), 16u, 123, 345); tint_atomicStore_1(&(sb), 20u, 123u); tint_atomicLoad_1(&(sb), 20u); tint_atomicAdd_1(&(sb), 20u, 123u); tint_atomicSub_1(&(sb), 20u, 123u); tint_atomicMax_1(&(sb), 20u, 123u); tint_atomicMin_1(&(sb), 20u, 123u); tint_atomicAnd_1(&(sb), 20u, 123u); tint_atomicOr_1(&(sb), 20u, 123u); tint_atomicXor_1(&(sb), 20u, 123u); tint_atomicExchange_1(&(sb), 20u, 123u); tint_atomicCompareExchangeWeak_1(&(sb), 20u, 123u, 345u); } @group(0) @binding(0) var sb : SB; struct SB { padding : vec4, a : atomic, b : atomic, } )"; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, WorkgroupBufferAtomics) { auto* src = R"( struct S { padding : vec4, a : atomic, b : atomic, } var w : S; @compute @workgroup_size(1) fn main() { atomicStore(&(w.a), 123); atomicLoad(&(w.a)); atomicAdd(&(w.a), 123); atomicSub(&(w.a), 123); atomicMax(&(w.a), 123); atomicMin(&(w.a), 123); atomicAnd(&(w.a), 123); atomicOr(&(w.a), 123); atomicXor(&(w.a), 123); atomicExchange(&(w.a), 123); atomicCompareExchangeWeak(&(w.a), 123, 345); atomicStore(&(w.b), 123u); atomicLoad(&(w.b)); atomicAdd(&(w.b), 123u); atomicSub(&(w.b), 123u); atomicMax(&(w.b), 123u); atomicMin(&(w.b), 123u); atomicAnd(&(w.b), 123u); atomicOr(&(w.b), 123u); atomicXor(&(w.b), 123u); atomicExchange(&(w.b), 123u); atomicCompareExchangeWeak(&(w.b), 123u, 345u); } )"; auto* expect = src; auto got = Run(src); EXPECT_EQ(expect, str(got)); } TEST_F(DecomposeMemoryAccessTest, WorkgroupBufferAtomics_OutOfOrder) { auto* src = R"( @compute @workgroup_size(1) fn main() { atomicStore(&(w.a), 123); atomicLoad(&(w.a)); atomicAdd(&(w.a), 123); atomicSub(&(w.a), 123); atomicMax(&(w.a), 123); atomicMin(&(w.a), 123); atomicAnd(&(w.a), 123); atomicOr(&(w.a), 123); atomicXor(&(w.a), 123); atomicExchange(&(w.a), 123); atomicCompareExchangeWeak(&(w.a), 123, 345); atomicStore(&(w.b), 123u); atomicLoad(&(w.b)); atomicAdd(&(w.b), 123u); atomicSub(&(w.b), 123u); atomicMax(&(w.b), 123u); atomicMin(&(w.b), 123u); atomicAnd(&(w.b), 123u); atomicOr(&(w.b), 123u); atomicXor(&(w.b), 123u); atomicExchange(&(w.b), 123u); atomicCompareExchangeWeak(&(w.b), 123u, 345u); } var w : S; struct S { padding : vec4, a : atomic, b : atomic, } )"; auto* expect = src; auto got = Run(src); EXPECT_EQ(expect, str(got)); } } // namespace } // namespace tint::transform