mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-05-13 19:01:24 +00:00
utils: Add utils::Transform
An element-wise vector transformation utility function. Similar to JS/TS's map() method on arrays. Change-Id: I4baf52daa918f2e7bf5f9b4af13894fe66826f7c Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/69103 Kokoro: Kokoro <noreply+kokoro@google.com> Commit-Queue: Ben Clayton <bclayton@chromium.org> Reviewed-by: Antonio Maiorano <amaiorano@google.com>
This commit is contained in:
parent
e7a94965c1
commit
a539d8d33c
@ -729,6 +729,7 @@ if(${TINT_BUILD_TESTS})
|
|||||||
utils/reverse_test.cc
|
utils/reverse_test.cc
|
||||||
utils/scoped_assignment_test.cc
|
utils/scoped_assignment_test.cc
|
||||||
utils/string_test.cc
|
utils/string_test.cc
|
||||||
|
utils/transform_test.cc
|
||||||
utils/unique_vector_test.cc
|
utils/unique_vector_test.cc
|
||||||
writer/append_vector_test.cc
|
writer/append_vector_test.cc
|
||||||
writer/float_to_string_test.cc
|
writer/float_to_string_test.cc
|
||||||
|
62
src/utils/transform.h
Normal file
62
src/utils/transform.h
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
// Copyright 2021 The Tint Authors
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#ifndef SRC_UTILS_TRANSFORM_H_
|
||||||
|
#define SRC_UTILS_TRANSFORM_H_
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <type_traits>
|
||||||
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "src/traits.h"
|
||||||
|
|
||||||
|
namespace tint {
|
||||||
|
namespace utils {
|
||||||
|
|
||||||
|
/// Transform performs an element-wise transformation of a vector.
|
||||||
|
/// @param in the input vector.
|
||||||
|
/// @param transform the transformation function with signature: `OUT(IN)`
|
||||||
|
/// @returns a new vector with each element of the source vector transformed by
|
||||||
|
/// `transform`.
|
||||||
|
template <typename IN, typename TRANSFORMER>
|
||||||
|
auto Transform(const std::vector<IN>& in, TRANSFORMER&& transform)
|
||||||
|
-> std::vector<decltype(transform(in[0]))> {
|
||||||
|
std::vector<decltype(transform(in[0]))> result(in.size());
|
||||||
|
for (size_t i = 0; i < result.size(); ++i) {
|
||||||
|
result[i] = transform(in[i]);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Transform performs an element-wise transformation of a vector.
|
||||||
|
/// @param in the input vector.
|
||||||
|
/// @param transform the transformation function with signature:
|
||||||
|
/// `OUT(IN, size_t)`
|
||||||
|
/// @returns a new vector with each element of the source vector transformed by
|
||||||
|
/// `transform`.
|
||||||
|
template <typename IN, typename TRANSFORMER>
|
||||||
|
auto Transform(const std::vector<IN>& in, TRANSFORMER&& transform)
|
||||||
|
-> std::vector<decltype(transform(in[0], 1u))> {
|
||||||
|
std::vector<decltype(transform(in[0], 1u))> result(in.size());
|
||||||
|
for (size_t i = 0; i < result.size(); ++i) {
|
||||||
|
result[i] = transform(in[i], i);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace utils
|
||||||
|
} // namespace tint
|
||||||
|
|
||||||
|
#endif // SRC_UTILS_TRANSFORM_H_
|
94
src/utils/transform_test.cc
Normal file
94
src/utils/transform_test.cc
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
// Copyright 2021 The Tint Authors.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#include "src/utils/transform.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
#include "gmock/gmock.h"
|
||||||
|
|
||||||
|
#define CHECK_ELEMENT_TYPE(vector, expected) \
|
||||||
|
static_assert(std::is_same<decltype(vector)::value_type, expected>::value, \
|
||||||
|
"unexpected result vector element type")
|
||||||
|
|
||||||
|
namespace tint {
|
||||||
|
namespace utils {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
TEST(TransformTest, Empty) {
|
||||||
|
const std::vector<int> empty{};
|
||||||
|
{
|
||||||
|
auto transformed = Transform(empty, [](int) -> int {
|
||||||
|
[] { FAIL() << "Transform should not be called for empty vector"; }();
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, int);
|
||||||
|
EXPECT_EQ(transformed.size(), 0u);
|
||||||
|
}
|
||||||
|
{
|
||||||
|
auto transformed = Transform(empty, [](int, size_t) -> int {
|
||||||
|
[] { FAIL() << "Transform should not be called for empty vector"; }();
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, int);
|
||||||
|
EXPECT_EQ(transformed.size(), 0u);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(TransformTest, Identity) {
|
||||||
|
const std::vector<int> input{1, 2, 3, 4};
|
||||||
|
{
|
||||||
|
auto transformed = Transform(input, [](int i) { return i; });
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, int);
|
||||||
|
EXPECT_THAT(transformed, testing::ElementsAre(1, 2, 3, 4));
|
||||||
|
}
|
||||||
|
{
|
||||||
|
auto transformed = Transform(input, [](int i, size_t) { return i; });
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, int);
|
||||||
|
EXPECT_THAT(transformed, testing::ElementsAre(1, 2, 3, 4));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(TransformTest, Index) {
|
||||||
|
const std::vector<int> input{10, 20, 30, 40};
|
||||||
|
{
|
||||||
|
auto transformed = Transform(input, [](int, size_t idx) { return idx; });
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, size_t);
|
||||||
|
EXPECT_THAT(transformed, testing::ElementsAre(0u, 1u, 2u, 3u));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(TransformTest, TransformSameType) {
|
||||||
|
const std::vector<int> input{1, 2, 3, 4};
|
||||||
|
{
|
||||||
|
auto transformed = Transform(input, [](int i) { return i * 10; });
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, int);
|
||||||
|
EXPECT_THAT(transformed, testing::ElementsAre(10, 20, 30, 40));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(TransformTest, TransformDifferentType) {
|
||||||
|
const std::vector<int> input{1, 2, 3, 4};
|
||||||
|
{
|
||||||
|
auto transformed =
|
||||||
|
Transform(input, [](int i) { return std::to_string(i); });
|
||||||
|
CHECK_ELEMENT_TYPE(transformed, std::string);
|
||||||
|
EXPECT_THAT(transformed, testing::ElementsAre("1", "2", "3", "4"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
} // namespace utils
|
||||||
|
} // namespace tint
|
@ -346,6 +346,7 @@ tint_unittests_source_set("tint_unittests_utils_src") {
|
|||||||
"../src/utils/reverse_test.cc",
|
"../src/utils/reverse_test.cc",
|
||||||
"../src/utils/scoped_assignment_test.cc",
|
"../src/utils/scoped_assignment_test.cc",
|
||||||
"../src/utils/string_test.cc",
|
"../src/utils/string_test.cc",
|
||||||
|
"../src/utils/transform_test.cc",
|
||||||
"../src/utils/unique_vector_test.cc",
|
"../src/utils/unique_vector_test.cc",
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user