[wgsl-writer] Emit texture types.
This CL adds texture type emission to the WGSL writer.
Bug: tint:144
Change-Id: I18e40a587b77b953f17a6d0ad1a75a1bc4158ef0
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/28460
Reviewed-by: David Neto <dneto@google.com>
Commit-Queue: dan sinclair <dsinclair@chromium.org>
diff --git a/src/writer/wgsl/generator_impl.cc b/src/writer/wgsl/generator_impl.cc
index 9acde4f..ec38eb9 100644
--- a/src/writer/wgsl/generator_impl.cc
+++ b/src/writer/wgsl/generator_impl.cc
@@ -50,8 +50,12 @@
#include "src/ast/struct_member_offset_decoration.h"
#include "src/ast/switch_statement.h"
#include "src/ast/type/array_type.h"
+#include "src/ast/type/depth_texture_type.h"
#include "src/ast/type/matrix_type.h"
#include "src/ast/type/pointer_type.h"
+#include "src/ast/type/sampled_texture_type.h"
+#include "src/ast/type/sampler_type.h"
+#include "src/ast/type/storage_texture_type.h"
#include "src/ast/type/struct_type.h"
#include "src/ast/type/vector_type.h"
#include "src/ast/type_constructor_expression.h"
@@ -445,6 +449,120 @@
return EmitBlockAndNewline(func->body());
}
+bool GeneratorImpl::EmitImageFormat(const ast::type::ImageFormat fmt) {
+ switch (fmt) {
+ case ast::type::ImageFormat::kBgra8Unorm:
+ out_ << "bgra8unorm";
+ break;
+ case ast::type::ImageFormat::kBgra8UnormSrgb:
+ out_ << "bgra8unorm_srgb";
+ break;
+ case ast::type::ImageFormat::kR16Float:
+ out_ << "r16float";
+ break;
+ case ast::type::ImageFormat::kR16Sint:
+ out_ << "r16sint";
+ break;
+ case ast::type::ImageFormat::kR16Uint:
+ out_ << "r16uint";
+ break;
+ case ast::type::ImageFormat::kR32Float:
+ out_ << "r32float";
+ break;
+ case ast::type::ImageFormat::kR32Sint:
+ out_ << "r32sint";
+ break;
+ case ast::type::ImageFormat::kR32Uint:
+ out_ << "r32uint";
+ break;
+ case ast::type::ImageFormat::kR8Sint:
+ out_ << "r8sint";
+ break;
+ case ast::type::ImageFormat::kR8Snorm:
+ out_ << "r8snorm";
+ break;
+ case ast::type::ImageFormat::kR8Uint:
+ out_ << "r8uint";
+ break;
+ case ast::type::ImageFormat::kR8Unorm:
+ out_ << "r8unorm";
+ break;
+ case ast::type::ImageFormat::kRg11B10Float:
+ out_ << "rg11b10float";
+ break;
+ case ast::type::ImageFormat::kRg16Float:
+ out_ << "rg16float";
+ break;
+ case ast::type::ImageFormat::kRg16Sint:
+ out_ << "rg16sint";
+ break;
+ case ast::type::ImageFormat::kRg16Uint:
+ out_ << "rg16uint";
+ break;
+ case ast::type::ImageFormat::kRg32Float:
+ out_ << "rg32float";
+ break;
+ case ast::type::ImageFormat::kRg32Sint:
+ out_ << "rg32sint";
+ break;
+ case ast::type::ImageFormat::kRg32Uint:
+ out_ << "rg32uint";
+ break;
+ case ast::type::ImageFormat::kRg8Sint:
+ out_ << "rg8sint";
+ break;
+ case ast::type::ImageFormat::kRg8Snorm:
+ out_ << "rg8snorm";
+ break;
+ case ast::type::ImageFormat::kRg8Uint:
+ out_ << "rg8uint";
+ break;
+ case ast::type::ImageFormat::kRg8Unorm:
+ out_ << "rg8unorm";
+ break;
+ case ast::type::ImageFormat::kRgb10A2Unorm:
+ out_ << "rgb10a2unorm";
+ break;
+ case ast::type::ImageFormat::kRgba16Float:
+ out_ << "rgba16float";
+ break;
+ case ast::type::ImageFormat::kRgba16Sint:
+ out_ << "rgba16sint";
+ break;
+ case ast::type::ImageFormat::kRgba16Uint:
+ out_ << "rgba16uint";
+ break;
+ case ast::type::ImageFormat::kRgba32Float:
+ out_ << "rgba32float";
+ break;
+ case ast::type::ImageFormat::kRgba32Sint:
+ out_ << "rgba32sint";
+ break;
+ case ast::type::ImageFormat::kRgba32Uint:
+ out_ << "rgba32uint";
+ break;
+ case ast::type::ImageFormat::kRgba8Sint:
+ out_ << "rgba8sint";
+ break;
+ case ast::type::ImageFormat::kRgba8Snorm:
+ out_ << "rgba8snorm";
+ break;
+ case ast::type::ImageFormat::kRgba8Uint:
+ out_ << "rgba8uint";
+ break;
+ case ast::type::ImageFormat::kRgba8Unorm:
+ out_ << "rgba8unorm";
+ break;
+ case ast::type::ImageFormat::kRgba8UnormSrgb:
+ out_ << "rgba8unorm_srgb";
+ break;
+ default:
+ error_ = "unknown image format";
+ return false;
+ }
+ return true;
+}
+
bool GeneratorImpl::EmitType(ast::type::Type* type) {
if (type->IsAlias()) {
auto* alias = type->AsAlias();
@@ -485,6 +603,13 @@
return false;
}
out_ << ">";
+ } else if (type->IsSampler()) {
+ auto* sampler = type->AsSampler();
+ out_ << "sampler";
+
+ if (sampler->IsComparison()) {
+ out_ << "_comparison";
+ }
} else if (type->IsStruct()) {
auto* str = type->AsStruct()->impl();
if (str->decoration() != ast::StructDecoration::kNone) {
@@ -522,6 +647,81 @@
make_indent();
out_ << "}";
+ } else if (type->IsTexture()) {
+ auto* texture = type->AsTexture();
+
+ out_ << "texture_";
+ if (texture->IsDepth()) {
+ out_ << "depth_";
+ } else if (texture->IsSampled()) {
+ out_ << "sampled_";
+ } else if (texture->IsStorage()) {
+ auto* storage = texture->AsStorage();
+
+ if (storage->access() == ast::type::StorageAccess::kRead) {
+ out_ << "ro_";
+ } else if (storage->access() == ast::type::StorageAccess::kWrite) {
+ out_ << "wo_";
+ } else {
+ error_ = "unknown storage texture access";
+ return false;
+ }
+ } else {
+ error_ = "unknown texture type";
+ return false;
+ }
+
+ switch (texture->dim()) {
+ case ast::type::TextureDimension::k1d:
+ out_ << "1d";
+ break;
+ case ast::type::TextureDimension::k1dArray:
+ out_ << "1d_array";
+ break;
+ case ast::type::TextureDimension::k2d:
+ out_ << "2d";
+ break;
+ case ast::type::TextureDimension::k2dArray:
+ out_ << "2d_array";
+ break;
+ case ast::type::TextureDimension::k2dMs:
+ out_ << "2d_ms";
+ break;
+ case ast::type::TextureDimension::k2dMsArray:
+ out_ << "2d_ms_array";
+ break;
+ case ast::type::TextureDimension::k3d:
+ out_ << "3d";
+ break;
+ case ast::type::TextureDimension::kCube:
+ out_ << "cube";
+ break;
+ case ast::type::TextureDimension::kCubeArray:
+ out_ << "cube_array";
+ break;
+ default:
+ error_ = "unknown texture dimension";
+ return false;
+ }
+
+ if (texture->IsSampled()) {
+ auto* sampled = texture->AsSampled();
+
+ out_ << "<";
+ if (!EmitType(sampled->type())) {
+ return false;
+ }
+ out_ << ">";
+ } else if (texture->IsStorage()) {
+ auto* storage = texture->AsStorage();
+
+ out_ << "<";
+ if (!EmitImageFormat(storage->image_format())) {
+ return false;
+ }
+ out_ << ">";
+ }
+
} else if (type->IsU32()) {
out_ << "u32";
} else if (type->IsVector()) {
diff --git a/src/writer/wgsl/generator_impl.h b/src/writer/wgsl/generator_impl.h
index 778698b..e01b66b 100644
--- a/src/writer/wgsl/generator_impl.h
+++ b/src/writer/wgsl/generator_impl.h
@@ -26,6 +26,7 @@
#include "src/ast/module.h"
#include "src/ast/scalar_constructor_expression.h"
#include "src/ast/type/alias_type.h"
+#include "src/ast/type/storage_texture_type.h"
#include "src/ast/type/type.h"
#include "src/ast/type_constructor_expression.h"
#include "src/ast/variable.h"
@@ -179,6 +180,10 @@
/// @param type the type to generate
/// @returns true if the type is emitted
bool EmitType(ast::type::Type* type);
+ /// Handles emitting an image format
+ /// @param fmt the format to generate
+ /// @returns true if the format is emitted
+ bool EmitImageFormat(const ast::type::ImageFormat fmt);
/// Handles emitting a type constructor
/// @param expr the type constructor expression
/// @returns true if the constructor is emitted
diff --git a/src/writer/wgsl/generator_impl_type_test.cc b/src/writer/wgsl/generator_impl_type_test.cc
index 48c8fc3..46e06fc 100644
--- a/src/writer/wgsl/generator_impl_type_test.cc
+++ b/src/writer/wgsl/generator_impl_type_test.cc
@@ -20,10 +20,14 @@
#include "src/ast/struct_member_offset_decoration.h"
#include "src/ast/type/array_type.h"
#include "src/ast/type/bool_type.h"
+#include "src/ast/type/depth_texture_type.h"
#include "src/ast/type/f32_type.h"
#include "src/ast/type/i32_type.h"
#include "src/ast/type/matrix_type.h"
#include "src/ast/type/pointer_type.h"
+#include "src/ast/type/sampled_texture_type.h"
+#include "src/ast/type/sampler_type.h"
+#include "src/ast/type/storage_texture_type.h"
#include "src/ast/type/struct_type.h"
#include "src/ast/type/u32_type.h"
#include "src/ast/type/vector_type.h"
@@ -194,6 +198,219 @@
EXPECT_EQ(g.result(), "void");
}
+struct TextureData {
+ ast::type::TextureDimension dim;
+ const char* name;
+};
+inline std::ostream& operator<<(std::ostream& out, TextureData data) {
+ out << data.name;
+ return out;
+}
+using WgslGenerator_DepthTextureTest = testing::TestWithParam<TextureData>;
+
+TEST_P(WgslGenerator_DepthTextureTest, EmitType_DepthTexture) {
+ auto param = GetParam();
+
+ ast::type::DepthTextureType d(param.dim);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&d)) << g.error();
+ EXPECT_EQ(g.result(), param.name);
+}
+INSTANTIATE_TEST_SUITE_P(
+ WgslGeneratorImplTest,
+ WgslGenerator_DepthTextureTest,
+ testing::Values(
+ TextureData{ast::type::TextureDimension::k2d, "texture_depth_2d"},
+ TextureData{ast::type::TextureDimension::k2dArray,
+ "texture_depth_2d_array"},
+ TextureData{ast::type::TextureDimension::kCube, "texture_depth_cube"},
+ TextureData{ast::type::TextureDimension::kCubeArray,
+ "texture_depth_cube_array"}));
+
+using WgslGenerator_SampledTextureTest = testing::TestWithParam<TextureData>;
+TEST_P(WgslGenerator_SampledTextureTest, EmitType_SampledTexture_F32) {
+ auto param = GetParam();
+
+ ast::type::F32Type f32;
+ ast::type::SampledTextureType t(param.dim, &f32);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&t)) << g.error();
+ EXPECT_EQ(g.result(), std::string(param.name) + "<f32>");
+}
+
+TEST_P(WgslGenerator_SampledTextureTest, EmitType_SampledTexture_I32) {
+ auto param = GetParam();
+
+ ast::type::I32Type i32;
+ ast::type::SampledTextureType t(param.dim, &i32);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&t)) << g.error();
+ EXPECT_EQ(g.result(), std::string(param.name) + "<i32>");
+}
+
+TEST_P(WgslGenerator_SampledTextureTest, EmitType_SampledTexture_U32) {
+ auto param = GetParam();
+
+ ast::type::U32Type u32;
+ ast::type::SampledTextureType t(param.dim, &u32);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&t)) << g.error();
+ EXPECT_EQ(g.result(), std::string(param.name) + "<u32>");
+}
+INSTANTIATE_TEST_SUITE_P(
+ WgslGeneratorImplTest,
+ WgslGenerator_SampledTextureTest,
+ testing::Values(
+ TextureData{ast::type::TextureDimension::k1d, "texture_sampled_1d"},
+ TextureData{ast::type::TextureDimension::k1dArray,
+ "texture_sampled_1d_array"},
+ TextureData{ast::type::TextureDimension::k2d, "texture_sampled_2d"},
+ TextureData{ast::type::TextureDimension::k2dArray,
+ "texture_sampled_2d_array"},
+ TextureData{ast::type::TextureDimension::k3d, "texture_sampled_3d"},
+ TextureData{ast::type::TextureDimension::kCube, "texture_sampled_cube"},
+ TextureData{ast::type::TextureDimension::kCubeArray,
+ "texture_sampled_cube_array"}));
+
+struct StorageTextureData {
+ ast::type::ImageFormat fmt;
+ ast::type::TextureDimension dim;
+ ast::type::StorageAccess access;
+ const char* name;
+};
+inline std::ostream& operator<<(std::ostream& out, StorageTextureData data) {
+ out << data.name;
+ return out;
+}
+using WgslGenerator_StorageTextureTest =
+ testing::TestWithParam<StorageTextureData>;
+TEST_P(WgslGenerator_StorageTextureTest, EmitType_StorageTexture) {
+ auto param = GetParam();
+
+ ast::type::StorageTextureType t(param.dim, param.access, param.fmt);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&t)) << g.error();
+ EXPECT_EQ(g.result(), param.name);
+}
+INSTANTIATE_TEST_SUITE_P(
+ WgslGeneratorImplTest,
+ WgslGenerator_StorageTextureTest,
+ testing::Values(
+ StorageTextureData{
+ ast::type::ImageFormat::kR8Unorm, ast::type::TextureDimension::k1d,
+ ast::type::StorageAccess::kRead, "texture_ro_1d<r8unorm>"},
+ StorageTextureData{ast::type::ImageFormat::kR8Unorm,
+ ast::type::TextureDimension::k1dArray,
+ ast::type::StorageAccess::kRead,
+ "texture_ro_1d_array<r8unorm>"},
+ StorageTextureData{
+ ast::type::ImageFormat::kR8Unorm, ast::type::TextureDimension::k2d,
+ ast::type::StorageAccess::kRead, "texture_ro_2d<r8unorm>"},
+ StorageTextureData{ast::type::ImageFormat::kR8Unorm,
+ ast::type::TextureDimension::k2dArray,
+ ast::type::StorageAccess::kRead,
+ "texture_ro_2d_array<r8unorm>"},
+ StorageTextureData{
+ ast::type::ImageFormat::kR8Unorm, ast::type::TextureDimension::k3d,
+ ast::type::StorageAccess::kRead, "texture_ro_3d<r8unorm>"},
+ StorageTextureData{
+ ast::type::ImageFormat::kR8Unorm, ast::type::TextureDimension::k1d,
+ ast::type::StorageAccess::kWrite, "texture_wo_1d<r8unorm>"},
+ StorageTextureData{ast::type::ImageFormat::kR8Unorm,
+ ast::type::TextureDimension::k1dArray,
+ ast::type::StorageAccess::kWrite,
+ "texture_wo_1d_array<r8unorm>"},
+ StorageTextureData{
+ ast::type::ImageFormat::kR8Unorm, ast::type::TextureDimension::k2d,
+ ast::type::StorageAccess::kWrite, "texture_wo_2d<r8unorm>"},
+ StorageTextureData{ast::type::ImageFormat::kR8Unorm,
+ ast::type::TextureDimension::k2dArray,
+ ast::type::StorageAccess::kWrite,
+ "texture_wo_2d_array<r8unorm>"},
+ StorageTextureData{
+ ast::type::ImageFormat::kR8Unorm, ast::type::TextureDimension::k3d,
+ ast::type::StorageAccess::kWrite, "texture_wo_3d<r8unorm>"}));
+
+struct ImageFormatData {
+ ast::type::ImageFormat fmt;
+ const char* name;
+};
+inline std::ostream& operator<<(std::ostream& out, ImageFormatData data) {
+ out << data.name;
+ return out;
+}
+using WgslGenerator_ImageFormatTest = testing::TestWithParam<ImageFormatData>;
+TEST_P(WgslGenerator_ImageFormatTest, EmitType_StorageTexture_ImageFormat) {
+ auto param = GetParam();
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitImageFormat(param.fmt)) << g.error();
+ EXPECT_EQ(g.result(), param.name);
+}
+
+INSTANTIATE_TEST_SUITE_P(
+ WgslGeneratorImplTest,
+ WgslGenerator_ImageFormatTest,
+ testing::Values(
+ ImageFormatData{ast::type::ImageFormat::kR8Unorm, "r8unorm"},
+ ImageFormatData{ast::type::ImageFormat::kR8Snorm, "r8snorm"},
+ ImageFormatData{ast::type::ImageFormat::kR8Uint, "r8uint"},
+ ImageFormatData{ast::type::ImageFormat::kR8Sint, "r8sint"},
+ ImageFormatData{ast::type::ImageFormat::kR16Uint, "r16uint"},
+ ImageFormatData{ast::type::ImageFormat::kR16Sint, "r16sint"},
+ ImageFormatData{ast::type::ImageFormat::kR16Float, "r16float"},
+ ImageFormatData{ast::type::ImageFormat::kRg8Unorm, "rg8unorm"},
+ ImageFormatData{ast::type::ImageFormat::kRg8Snorm, "rg8snorm"},
+ ImageFormatData{ast::type::ImageFormat::kRg8Uint, "rg8uint"},
+ ImageFormatData{ast::type::ImageFormat::kRg8Sint, "rg8sint"},
+ ImageFormatData{ast::type::ImageFormat::kR32Uint, "r32uint"},
+ ImageFormatData{ast::type::ImageFormat::kR32Sint, "r32sint"},
+ ImageFormatData{ast::type::ImageFormat::kR32Float, "r32float"},
+ ImageFormatData{ast::type::ImageFormat::kRg16Uint, "rg16uint"},
+ ImageFormatData{ast::type::ImageFormat::kRg16Sint, "rg16sint"},
+ ImageFormatData{ast::type::ImageFormat::kRg16Float, "rg16float"},
+ ImageFormatData{ast::type::ImageFormat::kRgba8Unorm, "rgba8unorm"},
+ ImageFormatData{ast::type::ImageFormat::kRgba8UnormSrgb,
+ "rgba8unorm_srgb"},
+ ImageFormatData{ast::type::ImageFormat::kRgba8Snorm, "rgba8snorm"},
+ ImageFormatData{ast::type::ImageFormat::kRgba8Uint, "rgba8uint"},
+ ImageFormatData{ast::type::ImageFormat::kRgba8Sint, "rgba8sint"},
+ ImageFormatData{ast::type::ImageFormat::kBgra8Unorm, "bgra8unorm"},
+ ImageFormatData{ast::type::ImageFormat::kBgra8UnormSrgb,
+ "bgra8unorm_srgb"},
+ ImageFormatData{ast::type::ImageFormat::kRgb10A2Unorm, "rgb10a2unorm"},
+ ImageFormatData{ast::type::ImageFormat::kRg11B10Float, "rg11b10float"},
+ ImageFormatData{ast::type::ImageFormat::kRg32Uint, "rg32uint"},
+ ImageFormatData{ast::type::ImageFormat::kRg32Sint, "rg32sint"},
+ ImageFormatData{ast::type::ImageFormat::kRg32Float, "rg32float"},
+ ImageFormatData{ast::type::ImageFormat::kRgba16Uint, "rgba16uint"},
+ ImageFormatData{ast::type::ImageFormat::kRgba16Sint, "rgba16sint"},
+ ImageFormatData{ast::type::ImageFormat::kRgba16Float, "rgba16float"},
+ ImageFormatData{ast::type::ImageFormat::kRgba32Uint, "rgba32uint"},
+ ImageFormatData{ast::type::ImageFormat::kRgba32Sint, "rgba32sint"},
+ ImageFormatData{ast::type::ImageFormat::kRgba32Float, "rgba32float"}));
+
+TEST_F(WgslGeneratorImplTest, EmitType_Sampler) {
+ ast::type::SamplerType sampler(ast::type::SamplerKind::kSampler);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&sampler)) << g.error();
+ EXPECT_EQ(g.result(), "sampler");
+}
+
+TEST_F(WgslGeneratorImplTest, EmitType_SamplerComparison) {
+ ast::type::SamplerType sampler(ast::type::SamplerKind::kComparisonSampler);
+
+ GeneratorImpl g;
+ ASSERT_TRUE(g.EmitType(&sampler)) << g.error();
+ EXPECT_EQ(g.result(), "sampler_comparison");
+}
+
} // namespace
} // namespace wgsl
} // namespace writer