blob: 6a04a71ff4771ba62bb777cd3d4731517fb03b4f [file] [log] [blame]
/*
* Copyright 2024 WebAssembly Community Group participants
*
* 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 <cstddef>
#include <optional>
#include <vector>
#include "support/topological_sort.h"
#include "gtest/gtest.h"
using namespace wasm;
using Graph = TopologicalSort::Graph;
TEST(TopologicalSortTest, Empty) {
Graph graph;
TopologicalOrders orders(graph);
EXPECT_EQ(orders.begin(), orders.end());
}
TEST(TopologicalSortTest, Singleton) {
Graph graph(1);
TopologicalOrders orders(graph);
auto it = orders.begin();
ASSERT_NE(it, orders.end());
EXPECT_EQ(*it, std::vector<Index>{0});
++it;
EXPECT_EQ(it, orders.end());
}
TEST(TopologicalSortTest, Permutations) {
Graph graph(3);
TopologicalOrders orders(graph);
std::set<std::vector<Index>> results(orders.begin(), orders.end());
std::set<std::vector<Index>> expected{
{0, 1, 2},
{0, 2, 1},
{1, 0, 2},
{1, 2, 0},
{2, 0, 1},
{2, 1, 0},
};
EXPECT_EQ(results, expected);
}
TEST(TopologicalSortTest, Chain) {
constexpr Index n = 10;
Graph graph(n);
for (Index i = 1; i < n; ++i) {
graph[i].push_back(i - 1);
}
TopologicalOrders orders(graph);
std::set<std::vector<Index>> results(orders.begin(), orders.end());
std::set<std::vector<Index>> expected{{9, 8, 7, 6, 5, 4, 3, 2, 1, 0}};
EXPECT_EQ(results, expected);
}
TEST(TopologicalSortTest, TwoChains) {
Graph graph(4);
graph[0].push_back(2);
graph[1].push_back(3);
TopologicalOrders orders(graph);
std::set<std::vector<Index>> results(orders.begin(), orders.end());
std::set<std::vector<Index>> expected{
{0, 1, 2, 3},
{0, 1, 3, 2},
{0, 2, 1, 3},
{1, 0, 2, 3},
{1, 0, 3, 2},
{1, 3, 0, 2},
};
EXPECT_EQ(results, expected);
}
TEST(TopologicalSortTest, Diamond) {
Graph graph(4);
graph[0].push_back(1);
graph[0].push_back(2);
graph[1].push_back(3);
graph[2].push_back(3);
TopologicalOrders orders(graph);
std::set<std::vector<Index>> results(orders.begin(), orders.end());
std::set<std::vector<Index>> expected{
{0, 1, 2, 3},
{0, 2, 1, 3},
};
EXPECT_EQ(results, expected);
}
TEST(MinTopologicalSortTest, SortStrings) {
std::map<std::string, std::vector<std::string>> graph{
{"animal", {"mammal"}},
{"cat", {}},
{"dog", {}},
{"mammal", {"cat", "dog"}}};
std::vector<std::string> expected{"animal", "mammal", "cat", "dog"};
EXPECT_EQ(TopologicalSort::sortOf(graph.begin(), graph.end()), expected);
}
TEST(MinTopologicalSortTest, EmptyMinSort) {
Graph graph(0);
EXPECT_EQ(TopologicalSort::minSort<>(graph), std::vector<Index>{});
}
TEST(MinTopologicalSortTest, UnconstrainedMinSort) {
Graph graph(3);
std::vector<Index> expected{0, 1, 2};
EXPECT_EQ(TopologicalSort::minSort(graph), expected);
}
TEST(MinTopologicalSortTest, ReversedMinSort) {
Graph graph(3);
graph[2].push_back(1);
graph[1].push_back(0);
std::vector<Index> expected{2, 1, 0};
EXPECT_EQ(TopologicalSort::minSort(graph), expected);
}
TEST(MinTopologicalSortTest, OneBeforeZeroMinSort) {
Graph graph(3);
graph[1].push_back(0);
// 2 last because it is greater than 1 and 0
std::vector<Index> expected{1, 0, 2};
EXPECT_EQ(TopologicalSort::minSort(graph), expected);
}
TEST(MinTopologicalSortTest, TwoBeforeOneMinSort) {
Graph graph(3);
graph[2].push_back(1);
// 0 first because it is less than 2 and 1
std::vector<Index> expected{0, 2, 1};
EXPECT_EQ(TopologicalSort::minSort(graph), expected);
}
TEST(MinTopologicalSortTest, TwoBeforeZeroMinSort) {
Graph graph(3);
graph[2].push_back(0);
// 1 first because it is less than 2 and zero is not eligible.
std::vector<Index> expected{1, 2, 0};
EXPECT_EQ(TopologicalSort::minSort(graph), expected);
}