[tint][constant] Add constant::Manager
Constructs deduplicated constants, similarly to type::Manager The constant::Manager owns the type::Manager so they can be std::move()'d together without having to risk having the constant::Manager hold a stale pointer to a moved type::Manager. Not currently used. That comes next. Also un-inline type::Manager scalar helpers. Reduces transitive includes. Bug: tint:1935 Change-Id: I28fc74a712f19a171850df5e84433e2d60cba256 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/134360 Reviewed-by: Dan Sinclair <dsinclair@chromium.org> Kokoro: Kokoro <noreply+kokoro@google.com> Reviewed-by: James Price <jrprice@google.com> Commit-Queue: Ben Clayton <bclayton@google.com>
This commit is contained in:
parent
56e5fb57ef
commit
9aa8012d91
|
@ -894,6 +894,8 @@ libtint_source_set("libtint_constant_src") {
|
||||||
"constant/clone_context.h",
|
"constant/clone_context.h",
|
||||||
"constant/composite.cc",
|
"constant/composite.cc",
|
||||||
"constant/composite.h",
|
"constant/composite.h",
|
||||||
|
"constant/manager.cc",
|
||||||
|
"constant/manager.h",
|
||||||
"constant/node.cc",
|
"constant/node.cc",
|
||||||
"constant/node.h",
|
"constant/node.h",
|
||||||
"constant/scalar.cc",
|
"constant/scalar.cc",
|
||||||
|
@ -1681,6 +1683,7 @@ if (tint_build_unittests) {
|
||||||
tint_unittests_source_set("tint_unittests_constant_src") {
|
tint_unittests_source_set("tint_unittests_constant_src") {
|
||||||
sources = [
|
sources = [
|
||||||
"constant/composite_test.cc",
|
"constant/composite_test.cc",
|
||||||
|
"constant/manager_test.cc",
|
||||||
"constant/scalar_test.cc",
|
"constant/scalar_test.cc",
|
||||||
"constant/splat_test.cc",
|
"constant/splat_test.cc",
|
||||||
]
|
]
|
||||||
|
|
|
@ -237,6 +237,8 @@ list(APPEND TINT_LIB_SRCS
|
||||||
constant/clone_context.h
|
constant/clone_context.h
|
||||||
constant/composite.cc
|
constant/composite.cc
|
||||||
constant/composite.h
|
constant/composite.h
|
||||||
|
constant/manager.cc
|
||||||
|
constant/manager.h
|
||||||
constant/scalar.cc
|
constant/scalar.cc
|
||||||
constant/scalar.h
|
constant/scalar.h
|
||||||
constant/splat.cc
|
constant/splat.cc
|
||||||
|
@ -923,6 +925,7 @@ if(TINT_BUILD_TESTS)
|
||||||
ast/workgroup_attribute_test.cc
|
ast/workgroup_attribute_test.cc
|
||||||
clone_context_test.cc
|
clone_context_test.cc
|
||||||
constant/composite_test.cc
|
constant/composite_test.cc
|
||||||
|
constant/manager_test.cc
|
||||||
constant/scalar_test.cc
|
constant/scalar_test.cc
|
||||||
constant/splat_test.cc
|
constant/splat_test.cc
|
||||||
debug_test.cc
|
debug_test.cc
|
||||||
|
|
|
@ -0,0 +1,105 @@
|
||||||
|
// Copyright 2023 The Tint Authors.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#include "src/tint/constant/manager.h"
|
||||||
|
|
||||||
|
#include "src/tint/constant/composite.h"
|
||||||
|
#include "src/tint/constant/scalar.h"
|
||||||
|
#include "src/tint/constant/splat.h"
|
||||||
|
#include "src/tint/type/abstract_float.h"
|
||||||
|
#include "src/tint/type/abstract_int.h"
|
||||||
|
#include "src/tint/type/bool.h"
|
||||||
|
#include "src/tint/type/f16.h"
|
||||||
|
#include "src/tint/type/f32.h"
|
||||||
|
#include "src/tint/type/i32.h"
|
||||||
|
#include "src/tint/type/manager.h"
|
||||||
|
#include "src/tint/type/u32.h"
|
||||||
|
#include "src/tint/utils/predicates.h"
|
||||||
|
|
||||||
|
namespace tint::constant {
|
||||||
|
|
||||||
|
Manager::Manager() = default;
|
||||||
|
|
||||||
|
Manager::Manager(Manager&&) = default;
|
||||||
|
|
||||||
|
Manager& Manager::operator=(Manager&& rhs) = default;
|
||||||
|
|
||||||
|
Manager::~Manager() = default;
|
||||||
|
|
||||||
|
const constant::Value* Manager::Composite(const type::Type* type,
|
||||||
|
utils::VectorRef<const constant::Value*> elements) {
|
||||||
|
if (elements.IsEmpty()) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool any_zero = false;
|
||||||
|
bool all_zero = true;
|
||||||
|
bool all_equal = true;
|
||||||
|
auto* first = elements.Front();
|
||||||
|
for (auto* el : elements) {
|
||||||
|
if (!el) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
if (!any_zero && el->AnyZero()) {
|
||||||
|
any_zero = true;
|
||||||
|
}
|
||||||
|
if (all_zero && !el->AllZero()) {
|
||||||
|
all_zero = false;
|
||||||
|
}
|
||||||
|
if (all_equal && el != first) {
|
||||||
|
all_equal = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (all_equal) {
|
||||||
|
return Splat(type, elements.Front(), elements.Length());
|
||||||
|
}
|
||||||
|
|
||||||
|
return Get<constant::Composite>(type, std::move(elements), all_zero, any_zero);
|
||||||
|
}
|
||||||
|
|
||||||
|
const constant::Splat* Manager::Splat(const type::Type* type,
|
||||||
|
const constant::Value* element,
|
||||||
|
size_t n) {
|
||||||
|
return Get<constant::Splat>(type, element, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<i32>* Manager::Get(i32 value) {
|
||||||
|
return Get<Scalar<i32>>(types.i32(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<u32>* Manager::Get(u32 value) {
|
||||||
|
return Get<Scalar<u32>>(types.u32(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<f32>* Manager::Get(f32 value) {
|
||||||
|
return Get<Scalar<f32>>(types.f32(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<f16>* Manager::Get(f16 value) {
|
||||||
|
return Get<Scalar<f16>>(types.f16(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<bool>* Manager::Get(bool value) {
|
||||||
|
return Get<Scalar<bool>>(types.bool_(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<AFloat>* Manager::Get(AFloat value) {
|
||||||
|
return Get<Scalar<AFloat>>(types.AFloat(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Scalar<AInt>* Manager::Get(AInt value) {
|
||||||
|
return Get<Scalar<AInt>>(types.AInt(), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace tint::constant
|
|
@ -0,0 +1,157 @@
|
||||||
|
// Copyright 2023 The Tint Authors.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#ifndef SRC_TINT_CONSTANT_MANAGER_H_
|
||||||
|
#define SRC_TINT_CONSTANT_MANAGER_H_
|
||||||
|
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#include "src/tint/constant/value.h"
|
||||||
|
#include "src/tint/number.h"
|
||||||
|
#include "src/tint/type/manager.h"
|
||||||
|
#include "src/tint/utils/hash.h"
|
||||||
|
#include "src/tint/utils/unique_allocator.h"
|
||||||
|
|
||||||
|
namespace tint::constant {
|
||||||
|
class Splat;
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
class Scalar;
|
||||||
|
} // namespace tint::constant
|
||||||
|
|
||||||
|
namespace tint::constant {
|
||||||
|
|
||||||
|
/// The constant manager holds a type manager and all the pointers to the known constant values.
|
||||||
|
class Manager final {
|
||||||
|
public:
|
||||||
|
/// Iterator is the type returned by begin() and end()
|
||||||
|
using TypeIterator = utils::BlockAllocator<Value>::ConstIterator;
|
||||||
|
|
||||||
|
/// Constructor
|
||||||
|
Manager();
|
||||||
|
|
||||||
|
/// Move constructor
|
||||||
|
Manager(Manager&&);
|
||||||
|
|
||||||
|
/// Move assignment operator
|
||||||
|
/// @param rhs the Manager to move
|
||||||
|
/// @return this Manager
|
||||||
|
Manager& operator=(Manager&& rhs);
|
||||||
|
|
||||||
|
/// Destructor
|
||||||
|
~Manager();
|
||||||
|
|
||||||
|
/// Wrap returns a new Manager created with the constants and types of `inner`.
|
||||||
|
/// The Manager returned by Wrap is intended to temporarily extend the constants and types of an
|
||||||
|
/// existing immutable Manager. As the copied constants and types are owned by `inner`, `inner`
|
||||||
|
/// must not be destructed or assigned while using the returned Manager.
|
||||||
|
/// TODO(bclayton) - Evaluate whether there are safer alternatives to this
|
||||||
|
/// function. See crbug.com/tint/460.
|
||||||
|
/// @param inner the immutable Manager to extend
|
||||||
|
/// @return the Manager that wraps `inner`
|
||||||
|
static Manager Wrap(const Manager& inner) {
|
||||||
|
Manager out;
|
||||||
|
out.values_.Wrap(inner.values_);
|
||||||
|
out.types = type::Manager::Wrap(inner.types);
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @param args the arguments used to construct the type, unique node or node.
|
||||||
|
/// @return a pointer to an instance of `T` with the provided arguments.
|
||||||
|
/// If NODE derives from UniqueNode and an existing instance of `T` has been
|
||||||
|
/// constructed, then the same pointer is returned.
|
||||||
|
template <typename NODE, typename... ARGS>
|
||||||
|
NODE* Get(ARGS&&... args) {
|
||||||
|
return values_.Get<NODE>(std::forward<ARGS>(args)...);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @returns an iterator to the beginning of the types
|
||||||
|
TypeIterator begin() const { return values_.begin(); }
|
||||||
|
/// @returns an iterator to the end of the types
|
||||||
|
TypeIterator end() const { return values_.end(); }
|
||||||
|
|
||||||
|
/// Constructs a constant of a vector, matrix or array type.
|
||||||
|
///
|
||||||
|
/// Examines the element values and will return either a constant::Composite or a
|
||||||
|
/// constant::Splat, depending on the element types and values.
|
||||||
|
///
|
||||||
|
/// @param type the composite type
|
||||||
|
/// @param elements the composite elements
|
||||||
|
/// @returns the value pointer
|
||||||
|
const constant::Value* Composite(const type::Type* type,
|
||||||
|
utils::VectorRef<const constant::Value*> elements);
|
||||||
|
|
||||||
|
/// Constructs a splat constant.
|
||||||
|
/// @param type the splat type
|
||||||
|
/// @param element the splat element
|
||||||
|
/// @param n the number of elements
|
||||||
|
/// @returns the value pointer
|
||||||
|
const constant::Splat* Splat(const type::Type* type, const constant::Value* element, size_t n);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the i32 value @p value
|
||||||
|
const Scalar<i32>* Get(i32 value);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the u32 value @p value
|
||||||
|
const Scalar<u32>* Get(u32 value);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the f32 value @p value
|
||||||
|
const Scalar<f32>* Get(f32 value);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the f16 value @p value
|
||||||
|
const Scalar<f16>* Get(f16 value);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the bool value @p value
|
||||||
|
const Scalar<bool>* Get(bool value);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the AFloat value @p value
|
||||||
|
const Scalar<AFloat>* Get(AFloat value);
|
||||||
|
|
||||||
|
/// @param value the constant value
|
||||||
|
/// @return a Scalar holding the AInt value @p value
|
||||||
|
const Scalar<AInt>* Get(AInt value);
|
||||||
|
|
||||||
|
/// The type manager
|
||||||
|
type::Manager types;
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// A specialization of utils::Hasher for constant::Value
|
||||||
|
struct Hasher {
|
||||||
|
/// @param value the value to hash
|
||||||
|
/// @returns a hash of the value
|
||||||
|
size_t operator()(const constant::Value& value) const { return value.Hash(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
/// An equality helper for constant::Value
|
||||||
|
struct Equal {
|
||||||
|
/// @param a the LHS value
|
||||||
|
/// @param b the RHS value
|
||||||
|
/// @returns true if the two constants are equal
|
||||||
|
bool operator()(const constant::Value& a, const constant::Value& b) const {
|
||||||
|
return a.Equal(&b);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Unique types owned by the manager
|
||||||
|
utils::UniqueAllocator<Value, Hasher, Equal> values_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace tint::constant
|
||||||
|
|
||||||
|
#endif // SRC_TINT_CONSTANT_MANAGER_H_
|
|
@ -0,0 +1,187 @@
|
||||||
|
// Copyright 2023 The Tint Authors.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#include "src/tint/constant/manager.h"
|
||||||
|
|
||||||
|
#include "gtest/gtest.h"
|
||||||
|
#include "src/tint/constant/scalar.h"
|
||||||
|
#include "src/tint/type/abstract_float.h"
|
||||||
|
#include "src/tint/type/abstract_int.h"
|
||||||
|
#include "src/tint/type/bool.h"
|
||||||
|
#include "src/tint/type/f16.h"
|
||||||
|
#include "src/tint/type/f32.h"
|
||||||
|
#include "src/tint/type/i32.h"
|
||||||
|
#include "src/tint/type/manager.h"
|
||||||
|
#include "src/tint/type/u32.h"
|
||||||
|
|
||||||
|
namespace tint::constant {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
using namespace tint::number_suffixes; // NOLINT
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
size_t count(const T& range_loopable) {
|
||||||
|
size_t n = 0;
|
||||||
|
for (auto it : range_loopable) {
|
||||||
|
(void)it;
|
||||||
|
n++;
|
||||||
|
}
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
using ManagerTest = testing::Test;
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, GetUnregistered) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c)>);
|
||||||
|
ASSERT_NE(c, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, GetSameConstantReturnsSamePtr) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c1 = cm.Get(1_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c1)>);
|
||||||
|
ASSERT_NE(c1, nullptr);
|
||||||
|
|
||||||
|
auto* c2 = cm.Get(1_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c2)>);
|
||||||
|
EXPECT_EQ(c1, c2);
|
||||||
|
EXPECT_EQ(c1->Type(), c2->Type());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, GetDifferentTypeReturnsDifferentPtr) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c1 = cm.Get(1_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c1)>);
|
||||||
|
ASSERT_NE(c1, nullptr);
|
||||||
|
|
||||||
|
auto* c2 = cm.Get(1_u);
|
||||||
|
static_assert(std::is_same_v<const Scalar<u32>*, decltype(c2)>);
|
||||||
|
EXPECT_NE(static_cast<const Value*>(c1), static_cast<const Value*>(c2));
|
||||||
|
EXPECT_NE(c1->Type(), c2->Type());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, GetDifferentValueReturnsDifferentPtr) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c1 = cm.Get(1_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c1)>);
|
||||||
|
ASSERT_NE(c1, nullptr);
|
||||||
|
|
||||||
|
auto* c2 = cm.Get(2_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c2)>);
|
||||||
|
ASSERT_NE(c2, nullptr);
|
||||||
|
EXPECT_NE(c1, c2);
|
||||||
|
EXPECT_EQ(c1->Type(), c2->Type());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_i32) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1_i);
|
||||||
|
static_assert(std::is_same_v<const Scalar<i32>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::I32>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, 1_i);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_u32) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1_u);
|
||||||
|
static_assert(std::is_same_v<const Scalar<u32>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::U32>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, 1_u);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_f32) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1_f);
|
||||||
|
static_assert(std::is_same_v<const Scalar<f32>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::F32>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, 1_f);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_f16) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1_h);
|
||||||
|
static_assert(std::is_same_v<const Scalar<f16>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::F16>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, 1_h);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_bool) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(true);
|
||||||
|
static_assert(std::is_same_v<const Scalar<bool>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::Bool>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_AFloat) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1._a);
|
||||||
|
static_assert(std::is_same_v<const Scalar<AFloat>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::AbstractFloat>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, 1._a);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, Get_AInt) {
|
||||||
|
constant::Manager cm;
|
||||||
|
|
||||||
|
auto* c = cm.Get(1_a);
|
||||||
|
static_assert(std::is_same_v<const Scalar<AInt>*, decltype(c)>);
|
||||||
|
ASSERT_TRUE(Is<type::AbstractInt>(c->Type()));
|
||||||
|
EXPECT_EQ(c->value, 1_a);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, WrapDoesntAffectInner_Constant) {
|
||||||
|
Manager inner;
|
||||||
|
Manager outer = Manager::Wrap(inner);
|
||||||
|
|
||||||
|
inner.Get(1_i);
|
||||||
|
|
||||||
|
EXPECT_EQ(count(inner), 1u);
|
||||||
|
EXPECT_EQ(count(outer), 0u);
|
||||||
|
|
||||||
|
outer.Get(1_i);
|
||||||
|
|
||||||
|
EXPECT_EQ(count(inner), 1u);
|
||||||
|
EXPECT_EQ(count(outer), 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ManagerTest, WrapDoesntAffectInner_Types) {
|
||||||
|
Manager inner;
|
||||||
|
Manager outer = Manager::Wrap(inner);
|
||||||
|
|
||||||
|
inner.types.Get<type::I32>();
|
||||||
|
|
||||||
|
EXPECT_EQ(count(inner.types), 1u);
|
||||||
|
EXPECT_EQ(count(outer.types), 0u);
|
||||||
|
|
||||||
|
outer.types.Get<type::U32>();
|
||||||
|
|
||||||
|
EXPECT_EQ(count(inner.types), 1u);
|
||||||
|
EXPECT_EQ(count(outer.types), 1u);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
} // namespace tint::constant
|
|
@ -14,6 +14,18 @@
|
||||||
|
|
||||||
#include "src/tint/type/manager.h"
|
#include "src/tint/type/manager.h"
|
||||||
|
|
||||||
|
#include "src/tint/type/abstract_float.h"
|
||||||
|
#include "src/tint/type/abstract_int.h"
|
||||||
|
#include "src/tint/type/bool.h"
|
||||||
|
#include "src/tint/type/f16.h"
|
||||||
|
#include "src/tint/type/f32.h"
|
||||||
|
#include "src/tint/type/i32.h"
|
||||||
|
#include "src/tint/type/matrix.h"
|
||||||
|
#include "src/tint/type/type.h"
|
||||||
|
#include "src/tint/type/u32.h"
|
||||||
|
#include "src/tint/type/vector.h"
|
||||||
|
#include "src/tint/type/void.h"
|
||||||
|
|
||||||
namespace tint::type {
|
namespace tint::type {
|
||||||
|
|
||||||
Manager::Manager() = default;
|
Manager::Manager() = default;
|
||||||
|
@ -24,4 +36,91 @@ Manager& Manager::operator=(Manager&& rhs) = default;
|
||||||
|
|
||||||
Manager::~Manager() = default;
|
Manager::~Manager() = default;
|
||||||
|
|
||||||
|
const type::Void* Manager::void_() {
|
||||||
|
return Get<type::Void>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Bool* Manager::bool_() {
|
||||||
|
return Get<type::Bool>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::I32* Manager::i32() {
|
||||||
|
return Get<type::I32>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::U32* Manager::u32() {
|
||||||
|
return Get<type::U32>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::F32* Manager::f32() {
|
||||||
|
return Get<type::F32>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::F16* Manager::f16() {
|
||||||
|
return Get<type::F16>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::AbstractFloat* Manager::AFloat() {
|
||||||
|
return Get<type::AbstractFloat>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::AbstractInt* Manager::AInt() {
|
||||||
|
return Get<type::AbstractInt>();
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Vector* Manager::vec(const type::Type* inner, uint32_t size) {
|
||||||
|
return Get<type::Vector>(inner, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Vector* Manager::vec2(const type::Type* inner) {
|
||||||
|
return vec(inner, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Vector* Manager::vec3(const type::Type* inner) {
|
||||||
|
return vec(inner, 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Vector* Manager::vec4(const type::Type* inner) {
|
||||||
|
return vec(inner, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat(const type::Type* inner, uint32_t cols, uint32_t rows) {
|
||||||
|
return Get<type::Matrix>(vec(inner, rows), cols);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat2x2(const type::Type* inner) {
|
||||||
|
return mat(inner, 2, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat2x3(const type::Type* inner) {
|
||||||
|
return mat(inner, 2, 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat2x4(const type::Type* inner) {
|
||||||
|
return mat(inner, 2, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat3x2(const type::Type* inner) {
|
||||||
|
return mat(inner, 3, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat3x3(const type::Type* inner) {
|
||||||
|
return mat(inner, 3, 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat3x4(const type::Type* inner) {
|
||||||
|
return mat(inner, 3, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat4x2(const type::Type* inner) {
|
||||||
|
return mat(inner, 4, 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat4x3(const type::Type* inner) {
|
||||||
|
return mat(inner, 4, 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
const type::Matrix* Manager::mat4x4(const type::Type* inner) {
|
||||||
|
return mat(inner, 4, 4);
|
||||||
|
}
|
||||||
} // namespace tint::type
|
} // namespace tint::type
|
||||||
|
|
|
@ -17,18 +17,25 @@
|
||||||
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include "src/tint/type/bool.h"
|
|
||||||
#include "src/tint/type/f16.h"
|
|
||||||
#include "src/tint/type/f32.h"
|
|
||||||
#include "src/tint/type/i32.h"
|
|
||||||
#include "src/tint/type/matrix.h"
|
|
||||||
#include "src/tint/type/type.h"
|
#include "src/tint/type/type.h"
|
||||||
#include "src/tint/type/u32.h"
|
#include "src/tint/type/unique_node.h"
|
||||||
#include "src/tint/type/vector.h"
|
|
||||||
#include "src/tint/type/void.h"
|
|
||||||
#include "src/tint/utils/hash.h"
|
#include "src/tint/utils/hash.h"
|
||||||
#include "src/tint/utils/unique_allocator.h"
|
#include "src/tint/utils/unique_allocator.h"
|
||||||
|
|
||||||
|
// Forward declarations
|
||||||
|
namespace tint::type {
|
||||||
|
class AbstractFloat;
|
||||||
|
class AbstractInt;
|
||||||
|
class Bool;
|
||||||
|
class F16;
|
||||||
|
class F32;
|
||||||
|
class I32;
|
||||||
|
class Matrix;
|
||||||
|
class U32;
|
||||||
|
class Vector;
|
||||||
|
class Void;
|
||||||
|
} // namespace tint::type
|
||||||
|
|
||||||
namespace tint::type {
|
namespace tint::type {
|
||||||
|
|
||||||
/// The type manager holds all the pointers to the known types.
|
/// The type manager holds all the pointers to the known types.
|
||||||
|
@ -93,85 +100,87 @@ class Manager final {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @returns a void type
|
/// @returns a void type
|
||||||
const type::Type* void_() { return Get<type::Void>(); }
|
const type::Void* void_();
|
||||||
|
|
||||||
/// @returns a bool type
|
/// @returns a bool type
|
||||||
const type::Type* bool_() { return Get<type::Bool>(); }
|
const type::Bool* bool_();
|
||||||
|
|
||||||
/// @returns an i32 type
|
/// @returns an i32 type
|
||||||
const type::Type* i32() { return Get<type::I32>(); }
|
const type::I32* i32();
|
||||||
|
|
||||||
/// @returns a u32 type
|
/// @returns a u32 type
|
||||||
const type::Type* u32() { return Get<type::U32>(); }
|
const type::U32* u32();
|
||||||
|
|
||||||
/// @returns an f32 type
|
/// @returns an f32 type
|
||||||
const type::Type* f32() { return Get<type::F32>(); }
|
const type::F32* f32();
|
||||||
|
|
||||||
/// @returns an f16 type
|
/// @returns an f16 type
|
||||||
const type::Type* f16() { return Get<type::F16>(); }
|
const type::F16* f16();
|
||||||
|
|
||||||
|
/// @returns a abstract-float type
|
||||||
|
const type::AbstractFloat* AFloat();
|
||||||
|
|
||||||
|
/// @returns a abstract-int type
|
||||||
|
const type::AbstractInt* AInt();
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @param size the vector size
|
/// @param size the vector size
|
||||||
/// @returns the vector type
|
/// @returns the vector type
|
||||||
const type::Vector* vec(const type::Type* inner, uint32_t size) {
|
const type::Vector* vec(const type::Type* inner, uint32_t size);
|
||||||
return Get<type::Vector>(inner, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the vector type
|
/// @returns the vector type
|
||||||
const type::Vector* vec2(const type::Type* inner) { return vec(inner, 2); }
|
const type::Vector* vec2(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the vector type
|
/// @returns the vector type
|
||||||
const type::Vector* vec3(const type::Type* inner) { return vec(inner, 3); }
|
const type::Vector* vec3(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the vector type
|
/// @returns the vector type
|
||||||
const type::Vector* vec4(const type::Type* inner) { return vec(inner, 4); }
|
const type::Vector* vec4(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @param cols the number of columns
|
/// @param cols the number of columns
|
||||||
/// @param rows the number of rows
|
/// @param rows the number of rows
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat(const type::Type* inner, uint32_t cols, uint32_t rows) {
|
const type::Matrix* mat(const type::Type* inner, uint32_t cols, uint32_t rows);
|
||||||
return Get<type::Matrix>(vec(inner, rows), cols);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat2x2(const type::Type* inner) { return mat(inner, 2, 2); }
|
const type::Matrix* mat2x2(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat2x3(const type::Type* inner) { return mat(inner, 2, 3); }
|
const type::Matrix* mat2x3(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat2x4(const type::Type* inner) { return mat(inner, 2, 4); }
|
const type::Matrix* mat2x4(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat3x2(const type::Type* inner) { return mat(inner, 3, 2); }
|
const type::Matrix* mat3x2(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat3x3(const type::Type* inner) { return mat(inner, 3, 3); }
|
const type::Matrix* mat3x3(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat3x4(const type::Type* inner) { return mat(inner, 3, 4); }
|
const type::Matrix* mat3x4(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat4x2(const type::Type* inner) { return mat(inner, 4, 2); }
|
const type::Matrix* mat4x2(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat4x3(const type::Type* inner) { return mat(inner, 4, 3); }
|
const type::Matrix* mat4x3(const type::Type* inner);
|
||||||
|
|
||||||
/// @param inner the inner type
|
/// @param inner the inner type
|
||||||
/// @returns the matrix type
|
/// @returns the matrix type
|
||||||
const type::Matrix* mat4x4(const type::Type* inner) { return mat(inner, 4, 4); }
|
const type::Matrix* mat4x4(const type::Type* inner);
|
||||||
|
|
||||||
/// @returns an iterator to the beginning of the types
|
/// @returns an iterator to the beginning of the types
|
||||||
TypeIterator begin() const { return types_.begin(); }
|
TypeIterator begin() const { return types_.begin(); }
|
||||||
|
|
Loading…
Reference in New Issue