Replace Type::(Is|As)Vector with Castable
Change-Id: Ic838aa783a279d0939a972773206fee2e33c4bff Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/34274 Reviewed-by: dan sinclair <dsinclair@chromium.org>
This commit is contained in:
parent
d11ced4765
commit
8a083ce9c8
|
@ -32,6 +32,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -64,7 +65,7 @@ TEST_F(AccessControlTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(AccessControlTypeTest, AccessRead) {
|
TEST_F(AccessControlTypeTest, AccessRead) {
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -65,7 +66,7 @@ TEST_F(AliasTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(AliasTypeTest, TypeName) {
|
TEST_F(AliasTypeTest, TypeName) {
|
||||||
|
|
|
@ -28,6 +28,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -71,7 +72,7 @@ TEST_F(ArrayTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ArrayTypeTest, TypeName) {
|
TEST_F(ArrayTypeTest, TypeName) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -47,7 +48,7 @@ TEST_F(BoolTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(BoolTypeTest, TypeName) {
|
TEST_F(BoolTypeTest, TypeName) {
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
#include "src/ast/type/pointer_type.h"
|
#include "src/ast/type/pointer_type.h"
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -48,7 +49,7 @@ TEST_F(DepthTextureTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_TRUE(ty->Is<TextureType>());
|
EXPECT_TRUE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(DepthTextureTypeTest, IsTextureType) {
|
TEST_F(DepthTextureTypeTest, IsTextureType) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -47,7 +48,7 @@ TEST_F(F32TypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(F32TypeTest, TypeName) {
|
TEST_F(F32TypeTest, TypeName) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -47,7 +48,7 @@ TEST_F(I32TypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(I32TypeTest, TypeName) {
|
TEST_F(I32TypeTest, TypeName) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -56,7 +57,7 @@ TEST_F(MatrixTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(MatrixTypeTest, TypeName) {
|
TEST_F(MatrixTypeTest, TypeName) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/pointer_type.h"
|
#include "src/ast/type/pointer_type.h"
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -48,7 +49,7 @@ TEST_F(MultisampledTextureTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_TRUE(ty->Is<TextureType>());
|
EXPECT_TRUE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(MultisampledTextureTypeTest, IsTextureType) {
|
TEST_F(MultisampledTextureTypeTest, IsTextureType) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -55,7 +56,7 @@ TEST_F(PointerTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(PointerTypeTest, TypeName) {
|
TEST_F(PointerTypeTest, TypeName) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/ast/type/pointer_type.h"
|
#include "src/ast/type/pointer_type.h"
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -48,7 +49,7 @@ TEST_F(SampledTextureTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_TRUE(ty->Is<TextureType>());
|
EXPECT_TRUE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(SampledTextureTypeTest, IsTextureType) {
|
TEST_F(SampledTextureTypeTest, IsTextureType) {
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -59,7 +60,7 @@ TEST_F(SamplerTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(SamplerTypeTest, TypeName_Sampler) {
|
TEST_F(SamplerTypeTest, TypeName_Sampler) {
|
||||||
|
|
|
@ -27,6 +27,7 @@
|
||||||
#include "src/ast/type/pointer_type.h"
|
#include "src/ast/type/pointer_type.h"
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/u32_type.h"
|
#include "src/ast/type/u32_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
#include "src/type_determiner.h"
|
#include "src/type_determiner.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
|
@ -52,7 +53,7 @@ TEST_F(StorageTextureTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_TRUE(ty->Is<TextureType>());
|
EXPECT_TRUE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(StorageTextureTypeTest, IsTextureType) {
|
TEST_F(StorageTextureTypeTest, IsTextureType) {
|
||||||
|
|
|
@ -62,7 +62,7 @@ TEST_F(StructTypeTest, Is) {
|
||||||
EXPECT_TRUE(ty->Is<StructType>());
|
EXPECT_TRUE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(StructTypeTest, TypeName) {
|
TEST_F(StructTypeTest, TypeName) {
|
||||||
|
|
|
@ -66,10 +66,6 @@ Type* Type::UnwrapAll() {
|
||||||
return UnwrapIfNeeded()->UnwrapPtrIfNeeded()->UnwrapIfNeeded();
|
return UnwrapIfNeeded()->UnwrapPtrIfNeeded()->UnwrapIfNeeded();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::IsVector() const {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Type::IsVoid() const {
|
bool Type::IsVoid() const {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -95,7 +91,7 @@ bool Type::is_float_matrix() {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_float_vector() {
|
bool Type::is_float_vector() {
|
||||||
return IsVector() && AsVector()->type()->is_float_scalar();
|
return Is<VectorType>() && As<VectorType>()->type()->is_float_scalar();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_float_scalar_or_vector() {
|
bool Type::is_float_scalar_or_vector() {
|
||||||
|
@ -107,40 +103,32 @@ bool Type::is_integer_scalar() {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_unsigned_integer_vector() {
|
bool Type::is_unsigned_integer_vector() {
|
||||||
return IsVector() && AsVector()->type()->Is<U32Type>();
|
return Is<VectorType>() && As<VectorType>()->type()->Is<U32Type>();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_signed_integer_vector() {
|
bool Type::is_signed_integer_vector() {
|
||||||
return IsVector() && AsVector()->type()->Is<I32Type>();
|
return Is<VectorType>() && As<VectorType>()->type()->Is<I32Type>();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_unsigned_scalar_or_vector() {
|
bool Type::is_unsigned_scalar_or_vector() {
|
||||||
return Is<U32Type>() || (IsVector() && AsVector()->type()->Is<U32Type>());
|
return Is<U32Type>() ||
|
||||||
|
(Is<VectorType>() && As<VectorType>()->type()->Is<U32Type>());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_signed_scalar_or_vector() {
|
bool Type::is_signed_scalar_or_vector() {
|
||||||
return Is<I32Type>() || (IsVector() && AsVector()->type()->Is<I32Type>());
|
return Is<I32Type>() ||
|
||||||
|
(Is<VectorType>() && As<VectorType>()->type()->Is<I32Type>());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Type::is_integer_scalar_or_vector() {
|
bool Type::is_integer_scalar_or_vector() {
|
||||||
return is_unsigned_scalar_or_vector() || is_signed_scalar_or_vector();
|
return is_unsigned_scalar_or_vector() || is_signed_scalar_or_vector();
|
||||||
}
|
}
|
||||||
|
|
||||||
const VectorType* Type::AsVector() const {
|
|
||||||
assert(IsVector());
|
|
||||||
return static_cast<const VectorType*>(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
const VoidType* Type::AsVoid() const {
|
const VoidType* Type::AsVoid() const {
|
||||||
assert(IsVoid());
|
assert(IsVoid());
|
||||||
return static_cast<const VoidType*>(this);
|
return static_cast<const VoidType*>(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
VectorType* Type::AsVector() {
|
|
||||||
assert(IsVector());
|
|
||||||
return static_cast<VectorType*>(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
VoidType* Type::AsVoid() {
|
VoidType* Type::AsVoid() {
|
||||||
assert(IsVoid());
|
assert(IsVoid());
|
||||||
return static_cast<VoidType*>(this);
|
return static_cast<VoidType*>(this);
|
||||||
|
|
|
@ -23,7 +23,6 @@ namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
namespace type {
|
namespace type {
|
||||||
|
|
||||||
class VectorType;
|
|
||||||
class VoidType;
|
class VoidType;
|
||||||
|
|
||||||
/// Supported memory layouts for calculating sizes
|
/// Supported memory layouts for calculating sizes
|
||||||
|
@ -36,8 +35,6 @@ class Type : public Castable<Type> {
|
||||||
Type(Type&&);
|
Type(Type&&);
|
||||||
~Type() override;
|
~Type() override;
|
||||||
|
|
||||||
/// @returns true if the type is a vec type
|
|
||||||
virtual bool IsVector() const;
|
|
||||||
/// @returns true if the type is a void type
|
/// @returns true if the type is a void type
|
||||||
virtual bool IsVoid() const;
|
virtual bool IsVoid() const;
|
||||||
|
|
||||||
|
@ -95,13 +92,9 @@ class Type : public Castable<Type> {
|
||||||
/// @returns true if this type is an integer scalar or vector
|
/// @returns true if this type is an integer scalar or vector
|
||||||
bool is_integer_scalar_or_vector();
|
bool is_integer_scalar_or_vector();
|
||||||
|
|
||||||
/// @returns the type as a vector type
|
|
||||||
const VectorType* AsVector() const;
|
|
||||||
/// @returns the type as a void type
|
/// @returns the type as a void type
|
||||||
const VoidType* AsVoid() const;
|
const VoidType* AsVoid() const;
|
||||||
|
|
||||||
/// @returns the type as a vector type
|
|
||||||
VectorType* AsVector();
|
|
||||||
/// @returns the type as a void type
|
/// @returns the type as a void type
|
||||||
VoidType* AsVoid();
|
VoidType* AsVoid();
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
#include "src/ast/type/sampler_type.h"
|
#include "src/ast/type/sampler_type.h"
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
#include "src/ast/type/texture_type.h"
|
#include "src/ast/type/texture_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace ast {
|
namespace ast {
|
||||||
|
@ -48,7 +49,7 @@ TEST_F(U32TypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_TRUE(ty->Is<U32Type>());
|
EXPECT_TRUE(ty->Is<U32Type>());
|
||||||
EXPECT_FALSE(ty->IsVector());
|
EXPECT_FALSE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(U32TypeTest, TypeName) {
|
TEST_F(U32TypeTest, TypeName) {
|
||||||
|
|
|
@ -31,10 +31,6 @@ VectorType::VectorType(VectorType&&) = default;
|
||||||
|
|
||||||
VectorType::~VectorType() = default;
|
VectorType::~VectorType() = default;
|
||||||
|
|
||||||
bool VectorType::IsVector() const {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string VectorType::type_name() const {
|
std::string VectorType::type_name() const {
|
||||||
return "__vec_" + std::to_string(size_) + subtype_->type_name();
|
return "__vec_" + std::to_string(size_) + subtype_->type_name();
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,9 +34,6 @@ class VectorType : public Castable<VectorType, Type> {
|
||||||
VectorType(VectorType&&);
|
VectorType(VectorType&&);
|
||||||
~VectorType() override;
|
~VectorType() override;
|
||||||
|
|
||||||
/// @returns true if the type is a vector type
|
|
||||||
bool IsVector() const override;
|
|
||||||
|
|
||||||
/// @returns the type of the vector elements
|
/// @returns the type of the vector elements
|
||||||
Type* type() const { return subtype_; }
|
Type* type() const { return subtype_; }
|
||||||
/// @returns the size of the vector
|
/// @returns the size of the vector
|
||||||
|
|
|
@ -56,7 +56,7 @@ TEST_F(VectorTypeTest, Is) {
|
||||||
EXPECT_FALSE(ty->Is<StructType>());
|
EXPECT_FALSE(ty->Is<StructType>());
|
||||||
EXPECT_FALSE(ty->Is<TextureType>());
|
EXPECT_FALSE(ty->Is<TextureType>());
|
||||||
EXPECT_FALSE(ty->Is<U32Type>());
|
EXPECT_FALSE(ty->Is<U32Type>());
|
||||||
EXPECT_TRUE(ty->IsVector());
|
EXPECT_TRUE(ty->Is<VectorType>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(VectorTypeTest, TypeName) {
|
TEST_F(VectorTypeTest, TypeName) {
|
||||||
|
|
|
@ -391,8 +391,8 @@ std::vector<ResourceBinding> Inspector::GetSampledTextureResourceBindingsImpl(
|
||||||
base_type = base_type->As<ast::type::ArrayType>()->type();
|
base_type = base_type->As<ast::type::ArrayType>()->type();
|
||||||
} else if (base_type->Is<ast::type::MatrixType>()) {
|
} else if (base_type->Is<ast::type::MatrixType>()) {
|
||||||
base_type = base_type->As<ast::type::MatrixType>()->type();
|
base_type = base_type->As<ast::type::MatrixType>()->type();
|
||||||
} else if (base_type->IsVector()) {
|
} else if (base_type->Is<ast::type::VectorType>()) {
|
||||||
base_type = base_type->AsVector()->type();
|
base_type = base_type->As<ast::type::VectorType>()->type();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (base_type->Is<ast::type::F32Type>()) {
|
if (base_type->Is<ast::type::F32Type>()) {
|
||||||
|
|
|
@ -3206,7 +3206,7 @@ TypedExpression FunctionEmitter::MakeVectorShuffle(
|
||||||
// Generate an ast::TypeConstructor expression.
|
// Generate an ast::TypeConstructor expression.
|
||||||
// Assume the literal indices are valid, and there is a valid number of them.
|
// Assume the literal indices are valid, and there is a valid number of them.
|
||||||
ast::type::VectorType* result_type =
|
ast::type::VectorType* result_type =
|
||||||
parser_impl_.ConvertType(inst.type_id())->AsVector();
|
parser_impl_.ConvertType(inst.type_id())->As<ast::type::VectorType>();
|
||||||
ast::ExpressionList values;
|
ast::ExpressionList values;
|
||||||
for (uint32_t i = 2; i < inst.NumInOperands(); ++i) {
|
for (uint32_t i = 2; i < inst.NumInOperands(); ++i) {
|
||||||
const auto index = inst.GetSingleWordInOperand(i);
|
const auto index = inst.GetSingleWordInOperand(i);
|
||||||
|
@ -3598,7 +3598,7 @@ TypedExpression FunctionEmitter::MakeSimpleSelect(
|
||||||
// - you can't select over pointers or pointer vectors, unless you also have
|
// - you can't select over pointers or pointer vectors, unless you also have
|
||||||
// a VariablePointers* capability, which is not allowed in by WebGPU.
|
// a VariablePointers* capability, which is not allowed in by WebGPU.
|
||||||
auto* op_ty = operand1.type;
|
auto* op_ty = operand1.type;
|
||||||
if (op_ty->IsVector() || op_ty->is_float_scalar() ||
|
if (op_ty->Is<ast::type::VectorType>() || op_ty->is_float_scalar() ||
|
||||||
op_ty->is_integer_scalar() || op_ty->Is<ast::type::BoolType>()) {
|
op_ty->is_integer_scalar() || op_ty->Is<ast::type::BoolType>()) {
|
||||||
ast::ExpressionList params;
|
ast::ExpressionList params;
|
||||||
params.push_back(operand1.expr);
|
params.push_back(operand1.expr);
|
||||||
|
@ -3829,8 +3829,8 @@ ast::ExpressionList FunctionEmitter::MakeCoordinateOperandsForImageAccess(
|
||||||
uint32_t num_coords_supplied = 0;
|
uint32_t num_coords_supplied = 0;
|
||||||
if (raw_coords.type->Is<ast::type::F32Type>()) {
|
if (raw_coords.type->Is<ast::type::F32Type>()) {
|
||||||
num_coords_supplied = 1;
|
num_coords_supplied = 1;
|
||||||
} else if (raw_coords.type->IsVector()) {
|
} else if (raw_coords.type->Is<ast::type::VectorType>()) {
|
||||||
num_coords_supplied = raw_coords.type->AsVector()->size();
|
num_coords_supplied = raw_coords.type->As<ast::type::VectorType>()->size();
|
||||||
}
|
}
|
||||||
if (num_coords_supplied == 0) {
|
if (num_coords_supplied == 0) {
|
||||||
Fail() << "bad or unsupported coordinate type for image access: "
|
Fail() << "bad or unsupported coordinate type for image access: "
|
||||||
|
|
|
@ -1351,8 +1351,8 @@ ast::Expression* ParserImpl::MakeNullValue(ast::type::Type* type) {
|
||||||
return create<ast::ScalarConstructorExpression>(
|
return create<ast::ScalarConstructorExpression>(
|
||||||
create<ast::FloatLiteral>(type, 0.0f));
|
create<ast::FloatLiteral>(type, 0.0f));
|
||||||
}
|
}
|
||||||
if (type->IsVector()) {
|
if (type->Is<ast::type::VectorType>()) {
|
||||||
const auto* vec_ty = type->AsVector();
|
const auto* vec_ty = type->As<ast::type::VectorType>();
|
||||||
ast::ExpressionList ast_components;
|
ast::ExpressionList ast_components;
|
||||||
for (size_t i = 0; i < vec_ty->size(); ++i) {
|
for (size_t i = 0; i < vec_ty->size(); ++i) {
|
||||||
ast_components.emplace_back(MakeNullValue(vec_ty->type()));
|
ast_components.emplace_back(MakeNullValue(vec_ty->type()));
|
||||||
|
@ -1450,7 +1450,7 @@ ast::type::Type* ParserImpl::GetSignedIntMatchingShape(ast::type::Type* other) {
|
||||||
other->Is<ast::type::I32Type>()) {
|
other->Is<ast::type::I32Type>()) {
|
||||||
return i32;
|
return i32;
|
||||||
}
|
}
|
||||||
auto* vec_ty = other->AsVector();
|
auto* vec_ty = other->As<ast::type::VectorType>();
|
||||||
if (vec_ty) {
|
if (vec_ty) {
|
||||||
return ast_module_.create<ast::type::VectorType>(i32, vec_ty->size());
|
return ast_module_.create<ast::type::VectorType>(i32, vec_ty->size());
|
||||||
}
|
}
|
||||||
|
@ -1469,7 +1469,7 @@ ast::type::Type* ParserImpl::GetUnsignedIntMatchingShape(
|
||||||
other->Is<ast::type::I32Type>()) {
|
other->Is<ast::type::I32Type>()) {
|
||||||
return u32;
|
return u32;
|
||||||
}
|
}
|
||||||
auto* vec_ty = other->AsVector();
|
auto* vec_ty = other->As<ast::type::VectorType>();
|
||||||
if (vec_ty) {
|
if (vec_ty) {
|
||||||
return ast_module_.create<ast::type::VectorType>(u32, vec_ty->size());
|
return ast_module_.create<ast::type::VectorType>(u32, vec_ty->size());
|
||||||
}
|
}
|
||||||
|
|
|
@ -171,19 +171,22 @@ TEST_F(SpvParserTest, ConvertType_VecOverF32) {
|
||||||
EXPECT_TRUE(p->BuildInternalModule());
|
EXPECT_TRUE(p->BuildInternalModule());
|
||||||
|
|
||||||
auto* v2xf32 = p->ConvertType(20);
|
auto* v2xf32 = p->ConvertType(20);
|
||||||
EXPECT_TRUE(v2xf32->IsVector());
|
EXPECT_TRUE(v2xf32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v2xf32->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v2xf32->AsVector()->size(), 2u);
|
v2xf32->As<ast::type::VectorType>()->type()->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(v2xf32->As<ast::type::VectorType>()->size(), 2u);
|
||||||
|
|
||||||
auto* v3xf32 = p->ConvertType(30);
|
auto* v3xf32 = p->ConvertType(30);
|
||||||
EXPECT_TRUE(v3xf32->IsVector());
|
EXPECT_TRUE(v3xf32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v3xf32->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v3xf32->AsVector()->size(), 3u);
|
v3xf32->As<ast::type::VectorType>()->type()->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(v3xf32->As<ast::type::VectorType>()->size(), 3u);
|
||||||
|
|
||||||
auto* v4xf32 = p->ConvertType(40);
|
auto* v4xf32 = p->ConvertType(40);
|
||||||
EXPECT_TRUE(v4xf32->IsVector());
|
EXPECT_TRUE(v4xf32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v4xf32->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v4xf32->AsVector()->size(), 4u);
|
v4xf32->As<ast::type::VectorType>()->type()->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(v4xf32->As<ast::type::VectorType>()->size(), 4u);
|
||||||
|
|
||||||
EXPECT_TRUE(p->error().empty());
|
EXPECT_TRUE(p->error().empty());
|
||||||
}
|
}
|
||||||
|
@ -198,19 +201,22 @@ TEST_F(SpvParserTest, ConvertType_VecOverI32) {
|
||||||
EXPECT_TRUE(p->BuildInternalModule());
|
EXPECT_TRUE(p->BuildInternalModule());
|
||||||
|
|
||||||
auto* v2xi32 = p->ConvertType(20);
|
auto* v2xi32 = p->ConvertType(20);
|
||||||
EXPECT_TRUE(v2xi32->IsVector());
|
EXPECT_TRUE(v2xi32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v2xi32->AsVector()->type()->Is<ast::type::I32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v2xi32->AsVector()->size(), 2u);
|
v2xi32->As<ast::type::VectorType>()->type()->Is<ast::type::I32Type>());
|
||||||
|
EXPECT_EQ(v2xi32->As<ast::type::VectorType>()->size(), 2u);
|
||||||
|
|
||||||
auto* v3xi32 = p->ConvertType(30);
|
auto* v3xi32 = p->ConvertType(30);
|
||||||
EXPECT_TRUE(v3xi32->IsVector());
|
EXPECT_TRUE(v3xi32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v3xi32->AsVector()->type()->Is<ast::type::I32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v3xi32->AsVector()->size(), 3u);
|
v3xi32->As<ast::type::VectorType>()->type()->Is<ast::type::I32Type>());
|
||||||
|
EXPECT_EQ(v3xi32->As<ast::type::VectorType>()->size(), 3u);
|
||||||
|
|
||||||
auto* v4xi32 = p->ConvertType(40);
|
auto* v4xi32 = p->ConvertType(40);
|
||||||
EXPECT_TRUE(v4xi32->IsVector());
|
EXPECT_TRUE(v4xi32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v4xi32->AsVector()->type()->Is<ast::type::I32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v4xi32->AsVector()->size(), 4u);
|
v4xi32->As<ast::type::VectorType>()->type()->Is<ast::type::I32Type>());
|
||||||
|
EXPECT_EQ(v4xi32->As<ast::type::VectorType>()->size(), 4u);
|
||||||
|
|
||||||
EXPECT_TRUE(p->error().empty());
|
EXPECT_TRUE(p->error().empty());
|
||||||
}
|
}
|
||||||
|
@ -225,19 +231,22 @@ TEST_F(SpvParserTest, ConvertType_VecOverU32) {
|
||||||
EXPECT_TRUE(p->BuildInternalModule());
|
EXPECT_TRUE(p->BuildInternalModule());
|
||||||
|
|
||||||
auto* v2xu32 = p->ConvertType(20);
|
auto* v2xu32 = p->ConvertType(20);
|
||||||
EXPECT_TRUE(v2xu32->IsVector());
|
EXPECT_TRUE(v2xu32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v2xu32->AsVector()->type()->Is<ast::type::U32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v2xu32->AsVector()->size(), 2u);
|
v2xu32->As<ast::type::VectorType>()->type()->Is<ast::type::U32Type>());
|
||||||
|
EXPECT_EQ(v2xu32->As<ast::type::VectorType>()->size(), 2u);
|
||||||
|
|
||||||
auto* v3xu32 = p->ConvertType(30);
|
auto* v3xu32 = p->ConvertType(30);
|
||||||
EXPECT_TRUE(v3xu32->IsVector());
|
EXPECT_TRUE(v3xu32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v3xu32->AsVector()->type()->Is<ast::type::U32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v3xu32->AsVector()->size(), 3u);
|
v3xu32->As<ast::type::VectorType>()->type()->Is<ast::type::U32Type>());
|
||||||
|
EXPECT_EQ(v3xu32->As<ast::type::VectorType>()->size(), 3u);
|
||||||
|
|
||||||
auto* v4xu32 = p->ConvertType(40);
|
auto* v4xu32 = p->ConvertType(40);
|
||||||
EXPECT_TRUE(v4xu32->IsVector());
|
EXPECT_TRUE(v4xu32->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(v4xu32->AsVector()->type()->Is<ast::type::U32Type>());
|
EXPECT_TRUE(
|
||||||
EXPECT_EQ(v4xu32->AsVector()->size(), 4u);
|
v4xu32->As<ast::type::VectorType>()->type()->Is<ast::type::U32Type>());
|
||||||
|
EXPECT_EQ(v4xu32->As<ast::type::VectorType>()->size(), 4u);
|
||||||
|
|
||||||
EXPECT_TRUE(p->error().empty());
|
EXPECT_TRUE(p->error().empty());
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,8 +35,8 @@ TEST_F(ParserImplTest, ConstExpr_TypeDecl) {
|
||||||
ASSERT_TRUE(e->AsConstructor()->IsTypeConstructor());
|
ASSERT_TRUE(e->AsConstructor()->IsTypeConstructor());
|
||||||
|
|
||||||
auto* t = e->AsConstructor()->AsTypeConstructor();
|
auto* t = e->AsConstructor()->AsTypeConstructor();
|
||||||
ASSERT_TRUE(t->type()->IsVector());
|
ASSERT_TRUE(t->type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_EQ(t->type()->AsVector()->size(), 2u);
|
EXPECT_EQ(t->type()->As<ast::type::VectorType>()->size(), 2u);
|
||||||
|
|
||||||
ASSERT_EQ(t->values().size(), 2u);
|
ASSERT_EQ(t->values().size(), 2u);
|
||||||
auto& v = t->values();
|
auto& v = t->values();
|
||||||
|
|
|
@ -151,8 +151,8 @@ TEST_P(VecTest, Parse) {
|
||||||
EXPECT_FALSE(t.errored);
|
EXPECT_FALSE(t.errored);
|
||||||
ASSERT_NE(t.value, nullptr) << p->error();
|
ASSERT_NE(t.value, nullptr) << p->error();
|
||||||
ASSERT_FALSE(p->has_error());
|
ASSERT_FALSE(p->has_error());
|
||||||
EXPECT_TRUE(t->IsVector());
|
EXPECT_TRUE(t->Is<ast::type::VectorType>());
|
||||||
EXPECT_EQ(t->AsVector()->size(), params.count);
|
EXPECT_EQ(t->As<ast::type::VectorType>()->size(), params.count);
|
||||||
}
|
}
|
||||||
INSTANTIATE_TEST_SUITE_P(ParserImplTest,
|
INSTANTIATE_TEST_SUITE_P(ParserImplTest,
|
||||||
VecTest,
|
VecTest,
|
||||||
|
@ -256,10 +256,10 @@ TEST_F(ParserImplTest, TypeDecl_Ptr_ToVec) {
|
||||||
ASSERT_TRUE(t->Is<ast::type::PointerType>());
|
ASSERT_TRUE(t->Is<ast::type::PointerType>());
|
||||||
|
|
||||||
auto* ptr = t->As<ast::type::PointerType>();
|
auto* ptr = t->As<ast::type::PointerType>();
|
||||||
ASSERT_TRUE(ptr->type()->IsVector());
|
ASSERT_TRUE(ptr->type()->Is<ast::type::VectorType>());
|
||||||
ASSERT_EQ(ptr->storage_class(), ast::StorageClass::kFunction);
|
ASSERT_EQ(ptr->storage_class(), ast::StorageClass::kFunction);
|
||||||
|
|
||||||
auto* vec = ptr->type()->AsVector();
|
auto* vec = ptr->type()->As<ast::type::VectorType>();
|
||||||
ASSERT_EQ(vec->size(), 2u);
|
ASSERT_EQ(vec->size(), 2u);
|
||||||
ASSERT_TRUE(vec->type()->Is<ast::type::F32Type>());
|
ASSERT_TRUE(vec->type()->Is<ast::type::F32Type>());
|
||||||
}
|
}
|
||||||
|
|
|
@ -185,13 +185,15 @@ bool BoundArrayAccessorsTransform::ProcessArrayAccessor(
|
||||||
|
|
||||||
auto* ret_type = expr->array()->result_type()->UnwrapAll();
|
auto* ret_type = expr->array()->result_type()->UnwrapAll();
|
||||||
if (!ret_type->Is<ast::type::ArrayType>() &&
|
if (!ret_type->Is<ast::type::ArrayType>() &&
|
||||||
!ret_type->Is<ast::type::MatrixType>() && !ret_type->IsVector()) {
|
!ret_type->Is<ast::type::MatrixType>() &&
|
||||||
|
!ret_type->Is<ast::type::VectorType>()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret_type->IsVector() || ret_type->Is<ast::type::ArrayType>()) {
|
if (ret_type->Is<ast::type::VectorType>() ||
|
||||||
uint32_t size = ret_type->IsVector()
|
ret_type->Is<ast::type::ArrayType>()) {
|
||||||
? ret_type->AsVector()->size()
|
uint32_t size = ret_type->Is<ast::type::VectorType>()
|
||||||
|
? ret_type->As<ast::type::VectorType>()->size()
|
||||||
: ret_type->As<ast::type::ArrayType>()->size();
|
: ret_type->As<ast::type::ArrayType>()->size();
|
||||||
if (size == 0) {
|
if (size == 0) {
|
||||||
error_ = "invalid 0 size for array or vector";
|
error_ = "invalid 0 size for array or vector";
|
||||||
|
|
|
@ -337,8 +337,8 @@ bool TypeDeterminer::DetermineArrayAccessor(
|
||||||
ast::type::Type* ret = nullptr;
|
ast::type::Type* ret = nullptr;
|
||||||
if (parent_type->Is<ast::type::ArrayType>()) {
|
if (parent_type->Is<ast::type::ArrayType>()) {
|
||||||
ret = parent_type->As<ast::type::ArrayType>()->type();
|
ret = parent_type->As<ast::type::ArrayType>()->type();
|
||||||
} else if (parent_type->IsVector()) {
|
} else if (parent_type->Is<ast::type::VectorType>()) {
|
||||||
ret = parent_type->AsVector()->type();
|
ret = parent_type->As<ast::type::VectorType>()->type();
|
||||||
} else if (parent_type->Is<ast::type::MatrixType>()) {
|
} else if (parent_type->Is<ast::type::MatrixType>()) {
|
||||||
auto* m = parent_type->As<ast::type::MatrixType>();
|
auto* m = parent_type->As<ast::type::MatrixType>();
|
||||||
ret = mod_->create<ast::type::VectorType>(m->type(), m->rows());
|
ret = mod_->create<ast::type::VectorType>(m->type(), m->rows());
|
||||||
|
@ -540,9 +540,9 @@ bool TypeDeterminer::DetermineIntrinsic(ast::IdentifierExpression* ident,
|
||||||
auto* bool_type = mod_->create<ast::type::BoolType>();
|
auto* bool_type = mod_->create<ast::type::BoolType>();
|
||||||
|
|
||||||
auto* param_type = expr->params()[0]->result_type()->UnwrapPtrIfNeeded();
|
auto* param_type = expr->params()[0]->result_type()->UnwrapPtrIfNeeded();
|
||||||
if (param_type->IsVector()) {
|
if (param_type->Is<ast::type::VectorType>()) {
|
||||||
expr->func()->set_result_type(mod_->create<ast::type::VectorType>(
|
expr->func()->set_result_type(mod_->create<ast::type::VectorType>(
|
||||||
bool_type, param_type->AsVector()->size()));
|
bool_type, param_type->As<ast::type::VectorType>()->size()));
|
||||||
} else {
|
} else {
|
||||||
expr->func()->set_result_type(bool_type);
|
expr->func()->set_result_type(bool_type);
|
||||||
}
|
}
|
||||||
|
@ -703,14 +703,16 @@ bool TypeDeterminer::DetermineIntrinsic(ast::IdentifierExpression* ident,
|
||||||
|
|
||||||
auto* param0_type = expr->params()[0]->result_type()->UnwrapPtrIfNeeded();
|
auto* param0_type = expr->params()[0]->result_type()->UnwrapPtrIfNeeded();
|
||||||
auto* param1_type = expr->params()[1]->result_type()->UnwrapPtrIfNeeded();
|
auto* param1_type = expr->params()[1]->result_type()->UnwrapPtrIfNeeded();
|
||||||
if (!param0_type->IsVector() || !param1_type->IsVector()) {
|
if (!param0_type->Is<ast::type::VectorType>() ||
|
||||||
|
!param1_type->Is<ast::type::VectorType>()) {
|
||||||
set_error(expr->source(), "invalid parameter type for " + ident->name());
|
set_error(expr->source(), "invalid parameter type for " + ident->name());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
expr->func()->set_result_type(mod_->create<ast::type::MatrixType>(
|
expr->func()->set_result_type(mod_->create<ast::type::MatrixType>(
|
||||||
mod_->create<ast::type::F32Type>(), param0_type->AsVector()->size(),
|
mod_->create<ast::type::F32Type>(),
|
||||||
param1_type->AsVector()->size()));
|
param0_type->As<ast::type::VectorType>()->size(),
|
||||||
|
param1_type->As<ast::type::VectorType>()->size()));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (ident->intrinsic() == ast::Intrinsic::kSelect) {
|
if (ident->intrinsic() == ast::Intrinsic::kSelect) {
|
||||||
|
@ -786,7 +788,8 @@ bool TypeDeterminer::DetermineIntrinsic(ast::IdentifierExpression* ident,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (data->vector_size > 0 &&
|
if (data->vector_size > 0 &&
|
||||||
result_types.back()->AsVector()->size() != data->vector_size) {
|
result_types.back()->As<ast::type::VectorType>()->size() !=
|
||||||
|
data->vector_size) {
|
||||||
set_error(expr->source(), "incorrect vector size for " +
|
set_error(expr->source(), "incorrect vector size for " +
|
||||||
ident->name() + ". " + "Requires " +
|
ident->name() + ". " + "Requires " +
|
||||||
std::to_string(data->vector_size) +
|
std::to_string(data->vector_size) +
|
||||||
|
@ -817,9 +820,10 @@ bool TypeDeterminer::DetermineIntrinsic(ast::IdentifierExpression* ident,
|
||||||
// provided.
|
// provided.
|
||||||
if (ident->intrinsic() == ast::Intrinsic::kLength ||
|
if (ident->intrinsic() == ast::Intrinsic::kLength ||
|
||||||
ident->intrinsic() == ast::Intrinsic::kDistance) {
|
ident->intrinsic() == ast::Intrinsic::kDistance) {
|
||||||
expr->func()->set_result_type(result_types[0]->is_float_scalar()
|
expr->func()->set_result_type(
|
||||||
? result_types[0]
|
result_types[0]->is_float_scalar()
|
||||||
: result_types[0]->AsVector()->type());
|
? result_types[0]
|
||||||
|
: result_types[0]->As<ast::type::VectorType>()->type());
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
// The determinant returns the component type of the columns
|
// The determinant returns the component type of the columns
|
||||||
|
@ -1054,8 +1058,8 @@ bool TypeDeterminer::DetermineMemberAccessor(
|
||||||
ret = mod_->create<ast::type::PointerType>(
|
ret = mod_->create<ast::type::PointerType>(
|
||||||
ret, res->As<ast::type::PointerType>()->storage_class());
|
ret, res->As<ast::type::PointerType>()->storage_class());
|
||||||
}
|
}
|
||||||
} else if (data_type->IsVector()) {
|
} else if (data_type->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = data_type->AsVector();
|
auto* vec = data_type->As<ast::type::VectorType>();
|
||||||
|
|
||||||
auto size = expr->member()->name().size();
|
auto size = expr->member()->name().size();
|
||||||
if (size == 1) {
|
if (size == 1) {
|
||||||
|
@ -1103,9 +1107,9 @@ bool TypeDeterminer::DetermineBinary(ast::BinaryExpression* expr) {
|
||||||
expr->IsLessThanEqual() || expr->IsGreaterThanEqual()) {
|
expr->IsLessThanEqual() || expr->IsGreaterThanEqual()) {
|
||||||
auto* bool_type = mod_->create<ast::type::BoolType>();
|
auto* bool_type = mod_->create<ast::type::BoolType>();
|
||||||
auto* param_type = expr->lhs()->result_type()->UnwrapPtrIfNeeded();
|
auto* param_type = expr->lhs()->result_type()->UnwrapPtrIfNeeded();
|
||||||
if (param_type->IsVector()) {
|
if (param_type->Is<ast::type::VectorType>()) {
|
||||||
expr->set_result_type(mod_->create<ast::type::VectorType>(
|
expr->set_result_type(mod_->create<ast::type::VectorType>(
|
||||||
bool_type, param_type->AsVector()->size()));
|
bool_type, param_type->As<ast::type::VectorType>()->size()));
|
||||||
} else {
|
} else {
|
||||||
expr->set_result_type(bool_type);
|
expr->set_result_type(bool_type);
|
||||||
}
|
}
|
||||||
|
@ -1124,11 +1128,13 @@ bool TypeDeterminer::DetermineBinary(ast::BinaryExpression* expr) {
|
||||||
lhs_type->As<ast::type::MatrixType>()->rows(),
|
lhs_type->As<ast::type::MatrixType>()->rows(),
|
||||||
rhs_type->As<ast::type::MatrixType>()->columns()));
|
rhs_type->As<ast::type::MatrixType>()->columns()));
|
||||||
|
|
||||||
} else if (lhs_type->Is<ast::type::MatrixType>() && rhs_type->IsVector()) {
|
} else if (lhs_type->Is<ast::type::MatrixType>() &&
|
||||||
|
rhs_type->Is<ast::type::VectorType>()) {
|
||||||
auto* mat = lhs_type->As<ast::type::MatrixType>();
|
auto* mat = lhs_type->As<ast::type::MatrixType>();
|
||||||
expr->set_result_type(
|
expr->set_result_type(
|
||||||
mod_->create<ast::type::VectorType>(mat->type(), mat->rows()));
|
mod_->create<ast::type::VectorType>(mat->type(), mat->rows()));
|
||||||
} else if (lhs_type->IsVector() && rhs_type->Is<ast::type::MatrixType>()) {
|
} else if (lhs_type->Is<ast::type::VectorType>() &&
|
||||||
|
rhs_type->Is<ast::type::MatrixType>()) {
|
||||||
auto* mat = rhs_type->As<ast::type::MatrixType>();
|
auto* mat = rhs_type->As<ast::type::MatrixType>();
|
||||||
expr->set_result_type(
|
expr->set_result_type(
|
||||||
mod_->create<ast::type::VectorType>(mat->type(), mat->columns()));
|
mod_->create<ast::type::VectorType>(mat->type(), mat->columns()));
|
||||||
|
@ -1138,12 +1144,13 @@ bool TypeDeterminer::DetermineBinary(ast::BinaryExpression* expr) {
|
||||||
} else if (rhs_type->Is<ast::type::MatrixType>()) {
|
} else if (rhs_type->Is<ast::type::MatrixType>()) {
|
||||||
// scalar * matrix
|
// scalar * matrix
|
||||||
expr->set_result_type(rhs_type);
|
expr->set_result_type(rhs_type);
|
||||||
} else if (lhs_type->IsVector() && rhs_type->IsVector()) {
|
} else if (lhs_type->Is<ast::type::VectorType>() &&
|
||||||
|
rhs_type->Is<ast::type::VectorType>()) {
|
||||||
expr->set_result_type(lhs_type);
|
expr->set_result_type(lhs_type);
|
||||||
} else if (lhs_type->IsVector()) {
|
} else if (lhs_type->Is<ast::type::VectorType>()) {
|
||||||
// Vector * scalar
|
// Vector * scalar
|
||||||
expr->set_result_type(lhs_type);
|
expr->set_result_type(lhs_type);
|
||||||
} else if (rhs_type->IsVector()) {
|
} else if (rhs_type->Is<ast::type::VectorType>()) {
|
||||||
// Scalar * vector
|
// Scalar * vector
|
||||||
expr->set_result_type(rhs_type);
|
expr->set_result_type(rhs_type);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -512,8 +512,8 @@ TEST_F(TypeDeterminerTest, Expr_ArrayAccessor_Matrix) {
|
||||||
ASSERT_TRUE(acc.result_type()->Is<ast::type::PointerType>());
|
ASSERT_TRUE(acc.result_type()->Is<ast::type::PointerType>());
|
||||||
|
|
||||||
auto* ptr = acc.result_type()->As<ast::type::PointerType>();
|
auto* ptr = acc.result_type()->As<ast::type::PointerType>();
|
||||||
ASSERT_TRUE(ptr->type()->IsVector());
|
ASSERT_TRUE(ptr->type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_EQ(ptr->type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ptr->type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_ArrayAccessor_Matrix_BothDimensions) {
|
TEST_F(TypeDeterminerTest, Expr_ArrayAccessor_Matrix_BothDimensions) {
|
||||||
|
@ -681,9 +681,12 @@ TEST_F(TypeDeterminerTest, Expr_Constructor_Type) {
|
||||||
|
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&tc));
|
EXPECT_TRUE(td()->DetermineResultType(&tc));
|
||||||
ASSERT_NE(tc.result_type(), nullptr);
|
ASSERT_NE(tc.result_type(), nullptr);
|
||||||
ASSERT_TRUE(tc.result_type()->IsVector());
|
ASSERT_TRUE(tc.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(tc.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(tc.result_type()
|
||||||
EXPECT_EQ(tc.result_type()->AsVector()->size(), 3u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(tc.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_Identifier_GlobalVariable) {
|
TEST_F(TypeDeterminerTest, Expr_Identifier_GlobalVariable) {
|
||||||
|
@ -1029,9 +1032,12 @@ TEST_F(TypeDeterminerTest, Expr_MemberAccessor_VectorSwizzle) {
|
||||||
ast::MemberAccessorExpression mem(ident, swizzle);
|
ast::MemberAccessorExpression mem(ident, swizzle);
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&mem)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&mem)) << td()->error();
|
||||||
ASSERT_NE(mem.result_type(), nullptr);
|
ASSERT_NE(mem.result_type(), nullptr);
|
||||||
ASSERT_TRUE(mem.result_type()->IsVector());
|
ASSERT_TRUE(mem.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(mem.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(mem.result_type()
|
||||||
EXPECT_EQ(mem.result_type()->AsVector()->size(), 2u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(mem.result_type()->As<ast::type::VectorType>()->size(), 2u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_MemberAccessor_VectorSwizzle_SingleElement) {
|
TEST_F(TypeDeterminerTest, Expr_MemberAccessor_VectorSwizzle_SingleElement) {
|
||||||
|
@ -1124,9 +1130,12 @@ TEST_F(TypeDeterminerTest, Expr_Accessor_MultiLevel) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&mem)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&mem)) << td()->error();
|
||||||
|
|
||||||
ASSERT_NE(mem.result_type(), nullptr);
|
ASSERT_NE(mem.result_type(), nullptr);
|
||||||
ASSERT_TRUE(mem.result_type()->IsVector());
|
ASSERT_TRUE(mem.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(mem.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(mem.result_type()
|
||||||
EXPECT_EQ(mem.result_type()->AsVector()->size(), 2u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(mem.result_type()->As<ast::type::VectorType>()->size(), 2u);
|
||||||
}
|
}
|
||||||
|
|
||||||
using Expr_Binary_BitwiseTest = TypeDeterminerTestWithParam<ast::BinaryOp>;
|
using Expr_Binary_BitwiseTest = TypeDeterminerTestWithParam<ast::BinaryOp>;
|
||||||
|
@ -1166,9 +1175,12 @@ TEST_P(Expr_Binary_BitwiseTest, Vector) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::I32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::I32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
||||||
Expr_Binary_BitwiseTest,
|
Expr_Binary_BitwiseTest,
|
||||||
|
@ -1220,10 +1232,12 @@ TEST_P(Expr_Binary_LogicalTest, Vector) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::BoolType>());
|
->As<ast::type::VectorType>()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->type()
|
||||||
|
->Is<ast::type::BoolType>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
||||||
Expr_Binary_LogicalTest,
|
Expr_Binary_LogicalTest,
|
||||||
|
@ -1267,10 +1281,12 @@ TEST_P(Expr_Binary_CompareTest, Vector) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::BoolType>());
|
->As<ast::type::VectorType>()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->type()
|
||||||
|
->Is<ast::type::BoolType>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
||||||
Expr_Binary_CompareTest,
|
Expr_Binary_CompareTest,
|
||||||
|
@ -1319,9 +1335,12 @@ TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Scalar) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Scalar_Vector) {
|
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Scalar_Vector) {
|
||||||
|
@ -1344,9 +1363,12 @@ TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Scalar_Vector) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Vector) {
|
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Vector) {
|
||||||
|
@ -1366,9 +1388,12 @@ TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Vector) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Matrix_Scalar) {
|
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Matrix_Scalar) {
|
||||||
|
@ -1448,9 +1473,12 @@ TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Matrix_Vector) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Matrix) {
|
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Matrix) {
|
||||||
|
@ -1474,9 +1502,12 @@ TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Vector_Matrix) {
|
||||||
|
|
||||||
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
ASSERT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 2u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 2u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Matrix_Matrix) {
|
TEST_F(TypeDeterminerTest, Expr_Binary_Multiply_Matrix_Matrix) {
|
||||||
|
@ -1551,9 +1582,12 @@ TEST_P(IntrinsicDerivativeTest, Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
||||||
|
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 4u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 4u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(IntrinsicDerivativeTest, MissingParam) {
|
TEST_P(IntrinsicDerivativeTest, MissingParam) {
|
||||||
|
@ -1655,10 +1689,12 @@ TEST_P(Intrinsic_FloatMethod, Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
||||||
|
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::BoolType>());
|
->As<ast::type::VectorType>()
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
->type()
|
||||||
|
->Is<ast::type::BoolType>());
|
||||||
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(Intrinsic_FloatMethod, Scalar) {
|
TEST_P(Intrinsic_FloatMethod, Scalar) {
|
||||||
|
@ -1815,18 +1851,24 @@ TEST_P(Intrinsic_StorageTextureOperation, TextureLoadRo) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
||||||
|
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
if (type == TextureType::kF32) {
|
if (type == TextureType::kF32) {
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
} else if (type == TextureType::kI32) {
|
} else if (type == TextureType::kI32) {
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::I32Type>());
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::I32Type>());
|
||||||
} else {
|
} else {
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::U32Type>());
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::U32Type>());
|
||||||
}
|
}
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 4u);
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 4u);
|
||||||
}
|
}
|
||||||
|
|
||||||
INSTANTIATE_TEST_SUITE_P(
|
INSTANTIATE_TEST_SUITE_P(
|
||||||
|
@ -1888,18 +1930,24 @@ TEST_P(Intrinsic_SampledTextureOperation, TextureLoadSampled) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
EXPECT_TRUE(td()->DetermineResultType(&expr));
|
||||||
|
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
ASSERT_TRUE(expr.result_type()->IsVector());
|
ASSERT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
if (type == TextureType::kF32) {
|
if (type == TextureType::kF32) {
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
} else if (type == TextureType::kI32) {
|
} else if (type == TextureType::kI32) {
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::I32Type>());
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::I32Type>());
|
||||||
} else {
|
} else {
|
||||||
EXPECT_TRUE(
|
EXPECT_TRUE(expr.result_type()
|
||||||
expr.result_type()->AsVector()->type()->Is<ast::type::U32Type>());
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::U32Type>());
|
||||||
}
|
}
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 4u);
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 4u);
|
||||||
}
|
}
|
||||||
|
|
||||||
INSTANTIATE_TEST_SUITE_P(
|
INSTANTIATE_TEST_SUITE_P(
|
||||||
|
@ -1956,9 +2004,12 @@ TEST_F(TypeDeterminerTest, Intrinsic_Select) {
|
||||||
EXPECT_TRUE(td()->Determine());
|
EXPECT_TRUE(td()->Determine());
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&expr)) << td()->error();
|
||||||
ASSERT_NE(expr.result_type(), nullptr);
|
ASSERT_NE(expr.result_type(), nullptr);
|
||||||
EXPECT_TRUE(expr.result_type()->IsVector());
|
EXPECT_TRUE(expr.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_EQ(expr.result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(expr.result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
EXPECT_TRUE(expr.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(expr.result_type()
|
||||||
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, Intrinsic_Select_TooFewParams) {
|
TEST_F(TypeDeterminerTest, Intrinsic_Select_TooFewParams) {
|
||||||
|
@ -2093,9 +2144,12 @@ TEST_P(UnaryOpExpressionTest, Expr_UnaryOp) {
|
||||||
ast::UnaryOpExpression der(op, create<ast::IdentifierExpression>("ident"));
|
ast::UnaryOpExpression der(op, create<ast::IdentifierExpression>("ident"));
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&der));
|
EXPECT_TRUE(td()->DetermineResultType(&der));
|
||||||
ASSERT_NE(der.result_type(), nullptr);
|
ASSERT_NE(der.result_type(), nullptr);
|
||||||
ASSERT_TRUE(der.result_type()->IsVector());
|
ASSERT_TRUE(der.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_TRUE(der.result_type()->AsVector()->type()->Is<ast::type::F32Type>());
|
EXPECT_TRUE(der.result_type()
|
||||||
EXPECT_EQ(der.result_type()->AsVector()->size(), 4u);
|
->As<ast::type::VectorType>()
|
||||||
|
->type()
|
||||||
|
->Is<ast::type::F32Type>());
|
||||||
|
EXPECT_EQ(der.result_type()->As<ast::type::VectorType>()->size(), 4u);
|
||||||
}
|
}
|
||||||
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
INSTANTIATE_TEST_SUITE_P(TypeDeterminerTest,
|
||||||
UnaryOpExpressionTest,
|
UnaryOpExpressionTest,
|
||||||
|
@ -2294,7 +2348,7 @@ TEST_P(ImportData_SingleParamTest, Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_SingleParamTest, Error_Integer) {
|
TEST_P(ImportData_SingleParamTest, Error_Integer) {
|
||||||
|
@ -2418,7 +2472,7 @@ TEST_P(ImportData_SingleParam_FloatOrInt_Test, Float_Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_SingleParam_FloatOrInt_Test, Sint_Scalar) {
|
TEST_P(ImportData_SingleParam_FloatOrInt_Test, Sint_Scalar) {
|
||||||
|
@ -2463,7 +2517,7 @@ TEST_P(ImportData_SingleParam_FloatOrInt_Test, Sint_Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_SingleParam_FloatOrInt_Test, Uint_Scalar) {
|
TEST_P(ImportData_SingleParam_FloatOrInt_Test, Uint_Scalar) {
|
||||||
|
@ -2508,7 +2562,7 @@ TEST_P(ImportData_SingleParam_FloatOrInt_Test, Uint_Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_unsigned_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_unsigned_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_SingleParam_FloatOrInt_Test, Error_Bool) {
|
TEST_P(ImportData_SingleParam_FloatOrInt_Test, Error_Bool) {
|
||||||
|
@ -2708,7 +2762,7 @@ TEST_P(ImportData_TwoParamTest, Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_TwoParamTest, Error_Integer) {
|
TEST_P(ImportData_TwoParamTest, Error_Integer) {
|
||||||
|
@ -3045,7 +3099,7 @@ TEST_F(TypeDeterminerTest, ImportData_Cross) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, ImportData_Cross_Error_Scalar) {
|
TEST_F(TypeDeterminerTest, ImportData_Cross_Error_Scalar) {
|
||||||
|
@ -3236,7 +3290,7 @@ TEST_P(ImportData_ThreeParamTest, Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_ThreeParamTest, Error_Integer) {
|
TEST_P(ImportData_ThreeParamTest, Error_Integer) {
|
||||||
|
@ -3476,7 +3530,7 @@ TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Float_Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Sint_Scalar) {
|
TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Sint_Scalar) {
|
||||||
|
@ -3543,7 +3597,7 @@ TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Sint_Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Uint_Scalar) {
|
TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Uint_Scalar) {
|
||||||
|
@ -3610,7 +3664,7 @@ TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Uint_Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_unsigned_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_unsigned_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Error_Bool) {
|
TEST_P(ImportData_ThreeParam_FloatOrInt_Test, Error_Bool) {
|
||||||
|
@ -3824,7 +3878,7 @@ TEST_P(ImportData_Int_SingleParamTest, Vector) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_Int_SingleParamTest, Error_Float) {
|
TEST_P(ImportData_Int_SingleParamTest, Error_Float) {
|
||||||
|
@ -3980,7 +4034,7 @@ TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Signed) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_signed_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Unsigned) {
|
TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Unsigned) {
|
||||||
|
@ -4016,7 +4070,7 @@ TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Unsigned) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_unsigned_integer_vector());
|
EXPECT_TRUE(ident->result_type()->is_unsigned_integer_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Float) {
|
TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Float) {
|
||||||
|
@ -4052,7 +4106,7 @@ TEST_P(ImportData_FloatOrInt_TwoParamTest, Vector_Float) {
|
||||||
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
EXPECT_TRUE(td()->DetermineResultType(&call)) << td()->error();
|
||||||
ASSERT_NE(ident->result_type(), nullptr);
|
ASSERT_NE(ident->result_type(), nullptr);
|
||||||
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
EXPECT_TRUE(ident->result_type()->is_float_vector());
|
||||||
EXPECT_EQ(ident->result_type()->AsVector()->size(), 3u);
|
EXPECT_EQ(ident->result_type()->As<ast::type::VectorType>()->size(), 3u);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_P(ImportData_FloatOrInt_TwoParamTest, Error_Bool) {
|
TEST_P(ImportData_FloatOrInt_TwoParamTest, Error_Bool) {
|
||||||
|
@ -4586,8 +4640,9 @@ TEST_P(TypeDeterminerTextureIntrinsicTest, Call) {
|
||||||
|
|
||||||
switch (param.texture_kind) {
|
switch (param.texture_kind) {
|
||||||
case ast::intrinsic::test::TextureKind::kRegular:
|
case ast::intrinsic::test::TextureKind::kRegular:
|
||||||
ASSERT_TRUE(call.result_type()->IsVector());
|
ASSERT_TRUE(call.result_type()->Is<ast::type::VectorType>());
|
||||||
EXPECT_EQ(call.result_type()->AsVector()->type(), datatype);
|
EXPECT_EQ(call.result_type()->As<ast::type::VectorType>()->type(),
|
||||||
|
datatype);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ast::intrinsic::test::TextureKind::kDepth:
|
case ast::intrinsic::test::TextureKind::kDepth:
|
||||||
|
|
|
@ -377,8 +377,10 @@ bool GeneratorImpl::EmitBinary(std::ostream& pre,
|
||||||
// Multiplying by a matrix requires the use of `mul` in order to get the
|
// Multiplying by a matrix requires the use of `mul` in order to get the
|
||||||
// type of multiply we desire.
|
// type of multiply we desire.
|
||||||
if (expr->op() == ast::BinaryOp::kMultiply &&
|
if (expr->op() == ast::BinaryOp::kMultiply &&
|
||||||
((lhs_type->IsVector() && rhs_type->Is<ast::type::MatrixType>()) ||
|
((lhs_type->Is<ast::type::VectorType>() &&
|
||||||
(lhs_type->Is<ast::type::MatrixType>() && rhs_type->IsVector()) ||
|
rhs_type->Is<ast::type::MatrixType>()) ||
|
||||||
|
(lhs_type->Is<ast::type::MatrixType>() &&
|
||||||
|
rhs_type->Is<ast::type::VectorType>()) ||
|
||||||
(lhs_type->Is<ast::type::MatrixType>() &&
|
(lhs_type->Is<ast::type::MatrixType>() &&
|
||||||
rhs_type->Is<ast::type::MatrixType>()))) {
|
rhs_type->Is<ast::type::MatrixType>()))) {
|
||||||
out << "mul(";
|
out << "mul(";
|
||||||
|
@ -614,13 +616,14 @@ bool GeneratorImpl::EmitCall(std::ostream& pre,
|
||||||
// out << "(";
|
// out << "(";
|
||||||
|
|
||||||
// auto param1_type = params[1]->result_type()->UnwrapPtrIfNeeded();
|
// auto param1_type = params[1]->result_type()->UnwrapPtrIfNeeded();
|
||||||
// if (!param1_type->IsVector()) {
|
// if (!param1_type->Is<ast::type::VectorType>()) {
|
||||||
// error_ = "invalid param type in outer_product got: " +
|
// error_ = "invalid param type in outer_product got: " +
|
||||||
// param1_type->type_name();
|
// param1_type->type_name();
|
||||||
// return false;
|
// return false;
|
||||||
// }
|
// }
|
||||||
|
|
||||||
// for (uint32_t i = 0; i < param1_type->AsVector()->size(); ++i) {
|
// for (uint32_t i = 0; i <
|
||||||
|
// param1_type->As<ast::type::VectorType>()->size(); ++i) {
|
||||||
// if (i > 0) {
|
// if (i > 0) {
|
||||||
// out << ", ";
|
// out << ", ";
|
||||||
// }
|
// }
|
||||||
|
@ -1556,8 +1559,8 @@ bool GeneratorImpl::EmitZeroValue(std::ostream& out, ast::type::Type* type) {
|
||||||
out << "0";
|
out << "0";
|
||||||
} else if (type->Is<ast::type::U32Type>()) {
|
} else if (type->Is<ast::type::U32Type>()) {
|
||||||
out << "0u";
|
out << "0u";
|
||||||
} else if (type->IsVector()) {
|
} else if (type->Is<ast::type::VectorType>()) {
|
||||||
return EmitZeroValue(out, type->AsVector()->type());
|
return EmitZeroValue(out, type->As<ast::type::VectorType>()->type());
|
||||||
} else if (type->Is<ast::type::MatrixType>()) {
|
} else if (type->Is<ast::type::MatrixType>()) {
|
||||||
auto* mat = type->As<ast::type::MatrixType>();
|
auto* mat = type->As<ast::type::MatrixType>();
|
||||||
for (uint32_t i = 0; i < (mat->rows() * mat->columns()); i++) {
|
for (uint32_t i = 0; i < (mat->rows() * mat->columns()); i++) {
|
||||||
|
@ -1697,7 +1700,7 @@ std::string GeneratorImpl::generate_storage_buffer_index_expression(
|
||||||
}
|
}
|
||||||
out << str_member->offset();
|
out << str_member->offset();
|
||||||
|
|
||||||
} else if (res_type->IsVector()) {
|
} else if (res_type->Is<ast::type::VectorType>()) {
|
||||||
// This must be a single element swizzle if we've got a vector at this
|
// This must be a single element swizzle if we've got a vector at this
|
||||||
// point.
|
// point.
|
||||||
if (mem->member()->name().size() != 1) {
|
if (mem->member()->name().size() != 1) {
|
||||||
|
@ -1726,7 +1729,7 @@ std::string GeneratorImpl::generate_storage_buffer_index_expression(
|
||||||
out << "(";
|
out << "(";
|
||||||
if (ary_type->Is<ast::type::ArrayType>()) {
|
if (ary_type->Is<ast::type::ArrayType>()) {
|
||||||
out << ary_type->As<ast::type::ArrayType>()->array_stride();
|
out << ary_type->As<ast::type::ArrayType>()->array_stride();
|
||||||
} else if (ary_type->IsVector()) {
|
} else if (ary_type->Is<ast::type::VectorType>()) {
|
||||||
// TODO(dsinclair): This is a hack. Our vectors can only be f32, i32
|
// TODO(dsinclair): This is a hack. Our vectors can only be f32, i32
|
||||||
// or u32 which are all 4 bytes. When we get f16 or other types we'll
|
// or u32 which are all 4 bytes. When we get f16 or other types we'll
|
||||||
// have to ask the type for the byte size.
|
// have to ask the type for the byte size.
|
||||||
|
@ -1773,8 +1776,9 @@ bool GeneratorImpl::EmitStorageBufferAccessor(std::ostream& pre,
|
||||||
bool is_store = rhs != nullptr;
|
bool is_store = rhs != nullptr;
|
||||||
|
|
||||||
std::string access_method = is_store ? "Store" : "Load";
|
std::string access_method = is_store ? "Store" : "Load";
|
||||||
if (result_type->IsVector()) {
|
if (result_type->Is<ast::type::VectorType>()) {
|
||||||
access_method += std::to_string(result_type->AsVector()->size());
|
access_method +=
|
||||||
|
std::to_string(result_type->As<ast::type::VectorType>()->size());
|
||||||
} else if (result_type->Is<ast::type::MatrixType>()) {
|
} else if (result_type->Is<ast::type::MatrixType>()) {
|
||||||
access_method +=
|
access_method +=
|
||||||
std::to_string(result_type->As<ast::type::MatrixType>()->rows());
|
std::to_string(result_type->As<ast::type::MatrixType>()->rows());
|
||||||
|
@ -1891,7 +1895,8 @@ bool GeneratorImpl::is_storage_buffer_access(
|
||||||
auto* data_type = structure->result_type()->UnwrapAll();
|
auto* data_type = structure->result_type()->UnwrapAll();
|
||||||
// If the data is a multi-element swizzle then we will not load the swizzle
|
// If the data is a multi-element swizzle then we will not load the swizzle
|
||||||
// portion through the Load command.
|
// portion through the Load command.
|
||||||
if (data_type->IsVector() && expr->member()->name().size() > 1) {
|
if (data_type->Is<ast::type::VectorType>() &&
|
||||||
|
expr->member()->name().size() > 1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2127,8 +2132,8 @@ bool GeneratorImpl::EmitType(std::ostream& out,
|
||||||
|
|
||||||
} else if (type->Is<ast::type::U32Type>()) {
|
} else if (type->Is<ast::type::U32Type>()) {
|
||||||
out << "uint";
|
out << "uint";
|
||||||
} else if (type->IsVector()) {
|
} else if (type->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = type->AsVector();
|
auto* vec = type->As<ast::type::VectorType>();
|
||||||
auto size = vec->size();
|
auto size = vec->size();
|
||||||
if (vec->type()->Is<ast::type::F32Type>() && size >= 1 && size <= 4) {
|
if (vec->type()->Is<ast::type::F32Type>() && size >= 1 && size <= 4) {
|
||||||
out << "float" << size;
|
out << "float" << size;
|
||||||
|
|
|
@ -242,8 +242,8 @@ uint32_t GeneratorImpl::calculate_alignment_size(ast::type::Type* type) {
|
||||||
count = adjust_for_alignment(count, largest_alignment);
|
count = adjust_for_alignment(count, largest_alignment);
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
if (type->IsVector()) {
|
if (type->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = type->AsVector();
|
auto* vec = type->As<ast::type::VectorType>();
|
||||||
uint32_t type_size = calculate_alignment_size(vec->type());
|
uint32_t type_size = calculate_alignment_size(vec->type());
|
||||||
if (vec->size() == 2) {
|
if (vec->size() == 2) {
|
||||||
return 2 * type_size;
|
return 2 * type_size;
|
||||||
|
@ -519,13 +519,14 @@ bool GeneratorImpl::EmitCall(ast::CallExpression* expr) {
|
||||||
// out_ << "(";
|
// out_ << "(";
|
||||||
|
|
||||||
// auto param1_type = params[1]->result_type()->UnwrapPtrIfNeeded();
|
// auto param1_type = params[1]->result_type()->UnwrapPtrIfNeeded();
|
||||||
// if (!param1_type->IsVector()) {
|
// if (!param1_type->Is<ast::type::VectorType>()) {
|
||||||
// error_ = "invalid param type in outer_product got: " +
|
// error_ = "invalid param type in outer_product got: " +
|
||||||
// param1_type->type_name();
|
// param1_type->type_name();
|
||||||
// return false;
|
// return false;
|
||||||
// }
|
// }
|
||||||
|
|
||||||
// for (uint32_t i = 0; i < param1_type->AsVector()->size(); ++i) {
|
// for (uint32_t i = 0; i <
|
||||||
|
// param1_type->As<ast::type::VectorType>()->size(); ++i) {
|
||||||
// if (i > 0) {
|
// if (i > 0) {
|
||||||
// out_ << ", ";
|
// out_ << ", ";
|
||||||
// }
|
// }
|
||||||
|
@ -937,8 +938,8 @@ bool GeneratorImpl::EmitZeroValue(ast::type::Type* type) {
|
||||||
out_ << "0";
|
out_ << "0";
|
||||||
} else if (type->Is<ast::type::U32Type>()) {
|
} else if (type->Is<ast::type::U32Type>()) {
|
||||||
out_ << "0u";
|
out_ << "0u";
|
||||||
} else if (type->IsVector()) {
|
} else if (type->Is<ast::type::VectorType>()) {
|
||||||
return EmitZeroValue(type->AsVector()->type());
|
return EmitZeroValue(type->As<ast::type::VectorType>()->type());
|
||||||
} else if (type->Is<ast::type::MatrixType>()) {
|
} else if (type->Is<ast::type::MatrixType>()) {
|
||||||
return EmitZeroValue(type->As<ast::type::MatrixType>()->type());
|
return EmitZeroValue(type->As<ast::type::MatrixType>()->type());
|
||||||
} else if (type->Is<ast::type::ArrayType>()) {
|
} else if (type->Is<ast::type::ArrayType>()) {
|
||||||
|
@ -1911,8 +1912,8 @@ bool GeneratorImpl::EmitType(ast::type::Type* type, const std::string& name) {
|
||||||
|
|
||||||
} else if (type->Is<ast::type::U32Type>()) {
|
} else if (type->Is<ast::type::U32Type>()) {
|
||||||
out_ << "uint";
|
out_ << "uint";
|
||||||
} else if (type->IsVector()) {
|
} else if (type->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = type->AsVector();
|
auto* vec = type->As<ast::type::VectorType>();
|
||||||
if (!EmitType(vec->type(), "")) {
|
if (!EmitType(vec->type(), "")) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,7 +33,7 @@ ast::TypeConstructorExpression* AsVectorConstructor(ast::Expression* expr) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
auto* type_constructor = constructor->AsTypeConstructor();
|
auto* type_constructor = constructor->AsTypeConstructor();
|
||||||
if (!type_constructor->type()->IsVector()) {
|
if (!type_constructor->type()->Is<ast::type::VectorType>()) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return type_constructor;
|
return type_constructor;
|
||||||
|
@ -47,8 +47,8 @@ bool PackCoordAndArrayIndex(
|
||||||
std::function<bool(ast::TypeConstructorExpression*)> callback) {
|
std::function<bool(ast::TypeConstructorExpression*)> callback) {
|
||||||
uint32_t packed_size;
|
uint32_t packed_size;
|
||||||
ast::type::Type* packed_el_ty; // Currenly must be f32.
|
ast::type::Type* packed_el_ty; // Currenly must be f32.
|
||||||
if (coords->result_type()->IsVector()) {
|
if (coords->result_type()->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = coords->result_type()->AsVector();
|
auto* vec = coords->result_type()->As<ast::type::VectorType>();
|
||||||
packed_size = vec->size() + 1;
|
packed_size = vec->size() + 1;
|
||||||
packed_el_ty = vec->type();
|
packed_el_ty = vec->type();
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -881,7 +881,7 @@ bool Builder::GenerateMemberAccessor(ast::MemberAccessorExpression* expr,
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!data_type->IsVector()) {
|
if (!data_type->Is<ast::type::VectorType>()) {
|
||||||
error_ = "Member accessor without a struct or vector. Something is wrong";
|
error_ = "Member accessor without a struct or vector. Something is wrong";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -1191,7 +1191,8 @@ bool Builder::is_constructor_const(ast::Expression* expr, bool is_global_init) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (result_type->IsVector() && !e->AsConstructor()->IsScalarConstructor()) {
|
if (result_type->Is<ast::type::VectorType>() &&
|
||||||
|
!e->AsConstructor()->IsScalarConstructor()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1202,8 +1203,8 @@ bool Builder::is_constructor_const(ast::Expression* expr, bool is_global_init) {
|
||||||
|
|
||||||
auto* sc = e->AsConstructor()->AsScalarConstructor();
|
auto* sc = e->AsConstructor()->AsScalarConstructor();
|
||||||
ast::type::Type* subtype = result_type->UnwrapAll();
|
ast::type::Type* subtype = result_type->UnwrapAll();
|
||||||
if (subtype->IsVector()) {
|
if (subtype->Is<ast::type::VectorType>()) {
|
||||||
subtype = subtype->AsVector()->type()->UnwrapAll();
|
subtype = subtype->As<ast::type::VectorType>()->type()->UnwrapAll();
|
||||||
} else if (subtype->Is<ast::type::MatrixType>()) {
|
} else if (subtype->Is<ast::type::MatrixType>()) {
|
||||||
subtype = subtype->As<ast::type::MatrixType>()->type()->UnwrapAll();
|
subtype = subtype->As<ast::type::MatrixType>()->type()->UnwrapAll();
|
||||||
} else if (subtype->Is<ast::type::ArrayType>()) {
|
} else if (subtype->Is<ast::type::ArrayType>()) {
|
||||||
|
@ -1244,12 +1245,14 @@ uint32_t Builder::GenerateTypeConstructorExpression(
|
||||||
|
|
||||||
bool can_cast_or_copy = result_type->is_scalar();
|
bool can_cast_or_copy = result_type->is_scalar();
|
||||||
|
|
||||||
if (result_type->IsVector() && result_type->AsVector()->type()->is_scalar()) {
|
if (result_type->Is<ast::type::VectorType>() &&
|
||||||
|
result_type->As<ast::type::VectorType>()->type()->is_scalar()) {
|
||||||
auto* value_type = values[0]->result_type()->UnwrapAll();
|
auto* value_type = values[0]->result_type()->UnwrapAll();
|
||||||
can_cast_or_copy =
|
can_cast_or_copy =
|
||||||
(value_type->IsVector() &&
|
(value_type->Is<ast::type::VectorType>() &&
|
||||||
value_type->AsVector()->type()->is_scalar() &&
|
value_type->As<ast::type::VectorType>()->type()->is_scalar() &&
|
||||||
result_type->AsVector()->size() == value_type->AsVector()->size());
|
result_type->As<ast::type::VectorType>()->size() ==
|
||||||
|
value_type->As<ast::type::VectorType>()->size());
|
||||||
}
|
}
|
||||||
if (can_cast_or_copy) {
|
if (can_cast_or_copy) {
|
||||||
return GenerateCastOrCopyOrPassthrough(result_type, values[0]);
|
return GenerateCastOrCopyOrPassthrough(result_type, values[0]);
|
||||||
|
@ -1263,8 +1266,8 @@ uint32_t Builder::GenerateTypeConstructorExpression(
|
||||||
bool result_is_constant_composite = constructor_is_const;
|
bool result_is_constant_composite = constructor_is_const;
|
||||||
bool result_is_spec_composite = false;
|
bool result_is_spec_composite = false;
|
||||||
|
|
||||||
if (result_type->IsVector()) {
|
if (result_type->Is<ast::type::VectorType>()) {
|
||||||
result_type = result_type->AsVector()->type();
|
result_type = result_type->As<ast::type::VectorType>()->type();
|
||||||
}
|
}
|
||||||
|
|
||||||
OperandList ops;
|
OperandList ops;
|
||||||
|
@ -1312,8 +1315,8 @@ uint32_t Builder::GenerateTypeConstructorExpression(
|
||||||
//
|
//
|
||||||
// For cases 1 and 2, if the type is different we also may need to insert
|
// For cases 1 and 2, if the type is different we also may need to insert
|
||||||
// a type cast.
|
// a type cast.
|
||||||
if (value_type->IsVector()) {
|
if (value_type->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = value_type->AsVector();
|
auto* vec = value_type->As<ast::type::VectorType>();
|
||||||
auto* vec_type = vec->type();
|
auto* vec_type = vec->type();
|
||||||
|
|
||||||
auto value_type_id = GenerateTypeIfNeeded(vec_type);
|
auto value_type_id = GenerateTypeIfNeeded(vec_type);
|
||||||
|
@ -1426,7 +1429,8 @@ uint32_t Builder::GenerateCastOrCopyOrPassthrough(ast::type::Type* to_type,
|
||||||
to_type->Is<ast::type::I32Type>()) ||
|
to_type->Is<ast::type::I32Type>()) ||
|
||||||
(from_type->Is<ast::type::F32Type>() &&
|
(from_type->Is<ast::type::F32Type>() &&
|
||||||
to_type->Is<ast::type::F32Type>()) ||
|
to_type->Is<ast::type::F32Type>()) ||
|
||||||
(from_type->IsVector() && (from_type == to_type))) {
|
(from_type->Is<ast::type::VectorType>() &&
|
||||||
|
(from_type == to_type))) {
|
||||||
return val_id;
|
return val_id;
|
||||||
} else if ((from_type->Is<ast::type::I32Type>() &&
|
} else if ((from_type->Is<ast::type::I32Type>() &&
|
||||||
to_type->Is<ast::type::U32Type>()) ||
|
to_type->Is<ast::type::U32Type>()) ||
|
||||||
|
@ -2454,8 +2458,8 @@ uint32_t Builder::GenerateTypeIfNeeded(ast::type::Type* type) {
|
||||||
}
|
}
|
||||||
} else if (type->Is<ast::type::U32Type>()) {
|
} else if (type->Is<ast::type::U32Type>()) {
|
||||||
push_type(spv::Op::OpTypeInt, {result, Operand::Int(32), Operand::Int(0)});
|
push_type(spv::Op::OpTypeInt, {result, Operand::Int(32), Operand::Int(0)});
|
||||||
} else if (type->IsVector()) {
|
} else if (type->Is<ast::type::VectorType>()) {
|
||||||
if (!GenerateVectorType(type->AsVector(), result)) {
|
if (!GenerateVectorType(type->As<ast::type::VectorType>(), result)) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
} else if (type->IsVoid()) {
|
} else if (type->IsVoid()) {
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
#include "src/ast/type/pointer_type.h"
|
#include "src/ast/type/pointer_type.h"
|
||||||
#include "src/ast/type/storage_texture_type.h"
|
#include "src/ast/type/storage_texture_type.h"
|
||||||
#include "src/ast/type/struct_type.h"
|
#include "src/ast/type/struct_type.h"
|
||||||
|
#include "src/ast/type/vector_type.h"
|
||||||
#include "src/ast/type_constructor_expression.h"
|
#include "src/ast/type_constructor_expression.h"
|
||||||
#include "src/context.h"
|
#include "src/context.h"
|
||||||
#include "src/scope_stack.h"
|
#include "src/scope_stack.h"
|
||||||
|
|
|
@ -547,8 +547,8 @@ bool GeneratorImpl::EmitType(ast::type::Type* type) {
|
||||||
|
|
||||||
} else if (type->Is<ast::type::U32Type>()) {
|
} else if (type->Is<ast::type::U32Type>()) {
|
||||||
out_ << "u32";
|
out_ << "u32";
|
||||||
} else if (type->IsVector()) {
|
} else if (type->Is<ast::type::VectorType>()) {
|
||||||
auto* vec = type->AsVector();
|
auto* vec = type->As<ast::type::VectorType>();
|
||||||
out_ << "vec" << vec->size() << "<";
|
out_ << "vec" << vec->size() << "<";
|
||||||
if (!EmitType(vec->type())) {
|
if (!EmitType(vec->type())) {
|
||||||
return false;
|
return false;
|
||||||
|
|
Loading…
Reference in New Issue