blob: 6017c734bb8fde46c96deefbd825c8b3611f9b98 [file] [log] [blame]
/*
* Copyright (C) 2020 The Android Open Source Project
*
* 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 "perfetto/ext/trace_processor/importers/memory_tracker/graph.h"
#include "perfetto/base/build_config.h"
#include "test/gtest_and_gmock.h"
namespace perfetto {
namespace trace_processor {
namespace {
using Node = GlobalNodeGraph::Node;
using Process = GlobalNodeGraph::Process;
const MemoryAllocatorNodeId kEmptyId;
} // namespace
TEST(GlobalNodeGraphTest, CreateContainerForProcess) {
GlobalNodeGraph global_dump_graph;
Process* dump = global_dump_graph.CreateGraphForProcess(10);
ASSERT_NE(dump, nullptr);
auto* map = global_dump_graph.process_node_graphs().find(10)->second.get();
ASSERT_EQ(dump, map);
}
TEST(GlobalNodeGraphTest, AddNodeOwnershipEdge) {
GlobalNodeGraph global_dump_graph;
Node owner(global_dump_graph.shared_memory_graph(), nullptr);
Node owned(global_dump_graph.shared_memory_graph(), nullptr);
global_dump_graph.AddNodeOwnershipEdge(&owner, &owned, 1);
auto& edges = global_dump_graph.edges();
ASSERT_NE(edges.begin(), edges.end());
auto& edge = *edges.begin();
ASSERT_EQ(edge.source(), &owner);
ASSERT_EQ(edge.target(), &owned);
ASSERT_EQ(edge.priority(), 1);
}
TEST(GlobalNodeGraphTest, VisitInDepthFirstPostOrder) {
GlobalNodeGraph graph;
Process* process_1 = graph.CreateGraphForProcess(1);
Process* process_2 = graph.CreateGraphForProcess(2);
Node* c1 = process_1->CreateNode(kEmptyId, "c1", false);
Node* c2 = process_1->CreateNode(kEmptyId, "c2", false);
Node* c2_c1 = process_1->CreateNode(kEmptyId, "c2/c1", false);
Node* c2_c2 = process_1->CreateNode(kEmptyId, "c2/c2", false);
Node* c3 = process_2->CreateNode(kEmptyId, "c3", false);
Node* c3_c1 = process_2->CreateNode(kEmptyId, "c3/c1", false);
Node* c3_c2 = process_2->CreateNode(kEmptyId, "c3/c2", false);
// |c3_c2| owns |c2_c2|.
graph.AddNodeOwnershipEdge(c3_c2, c2_c2, 1);
// This method should always call owners and then children before the node
// itself.
auto iterator = graph.VisitInDepthFirstPostOrder();
ASSERT_EQ(iterator.next(), graph.shared_memory_graph()->root());
ASSERT_EQ(iterator.next(), c1);
ASSERT_EQ(iterator.next(), c2_c1);
ASSERT_EQ(iterator.next(), c3_c2);
ASSERT_EQ(iterator.next(), c2_c2);
ASSERT_EQ(iterator.next(), c2);
ASSERT_EQ(iterator.next(), process_1->root());
ASSERT_EQ(iterator.next(), c3_c1);
ASSERT_EQ(iterator.next(), c3);
ASSERT_EQ(iterator.next(), process_2->root());
ASSERT_EQ(iterator.next(), nullptr);
}
TEST(GlobalNodeGraphTest, VisitInDepthFirstPreOrder) {
GlobalNodeGraph graph;
Process* process_1 = graph.CreateGraphForProcess(1);
Process* process_2 = graph.CreateGraphForProcess(2);
Node* c1 = process_1->CreateNode(kEmptyId, "c1", false);
Node* c2 = process_1->CreateNode(kEmptyId, "c2", false);
Node* c2_c1 = process_1->CreateNode(kEmptyId, "c2/c1", false);
Node* c2_c2 = process_1->CreateNode(kEmptyId, "c2/c2", false);
Node* c3 = process_2->CreateNode(kEmptyId, "c3", false);
Node* c3_c1 = process_2->CreateNode(kEmptyId, "c3/c1", false);
Node* c3_c2 = process_2->CreateNode(kEmptyId, "c3/c2", false);
// |c2_c2| owns |c3_c2|. Note this is opposite of post-order.
graph.AddNodeOwnershipEdge(c2_c2, c3_c2, 1);
// This method should always call owners and then children after the node
// itself.
auto iterator = graph.VisitInDepthFirstPreOrder();
ASSERT_EQ(iterator.next(), graph.shared_memory_graph()->root());
ASSERT_EQ(iterator.next(), process_1->root());
ASSERT_EQ(iterator.next(), c1);
ASSERT_EQ(iterator.next(), c2);
ASSERT_EQ(iterator.next(), c2_c1);
ASSERT_EQ(iterator.next(), process_2->root());
ASSERT_EQ(iterator.next(), c3);
ASSERT_EQ(iterator.next(), c3_c1);
ASSERT_EQ(iterator.next(), c3_c2);
ASSERT_EQ(iterator.next(), c2_c2);
ASSERT_EQ(iterator.next(), nullptr);
}
TEST(ProcessTest, CreateAndFindNode) {
GlobalNodeGraph global_dump_graph;
Process graph(1, &global_dump_graph);
Node* first =
graph.CreateNode(MemoryAllocatorNodeId(1), "simple/test/1", false);
Node* second =
graph.CreateNode(MemoryAllocatorNodeId(2), "simple/test/2", false);
Node* third =
graph.CreateNode(MemoryAllocatorNodeId(3), "simple/other/1", false);
Node* fourth =
graph.CreateNode(MemoryAllocatorNodeId(4), "complex/path", false);
Node* fifth =
graph.CreateNode(MemoryAllocatorNodeId(5), "complex/path/child/1", false);
ASSERT_EQ(graph.FindNode("simple/test/1"), first);
ASSERT_EQ(graph.FindNode("simple/test/2"), second);
ASSERT_EQ(graph.FindNode("simple/other/1"), third);
ASSERT_EQ(graph.FindNode("complex/path"), fourth);
ASSERT_EQ(graph.FindNode("complex/path/child/1"), fifth);
auto& nodes_by_id = global_dump_graph.nodes_by_id();
ASSERT_EQ(nodes_by_id.find(MemoryAllocatorNodeId(1))->second, first);
ASSERT_EQ(nodes_by_id.find(MemoryAllocatorNodeId(2))->second, second);
ASSERT_EQ(nodes_by_id.find(MemoryAllocatorNodeId(3))->second, third);
ASSERT_EQ(nodes_by_id.find(MemoryAllocatorNodeId(4))->second, fourth);
ASSERT_EQ(nodes_by_id.find(MemoryAllocatorNodeId(5))->second, fifth);
}
TEST(ProcessTest, CreateNodeParent) {
GlobalNodeGraph global_dump_graph;
Process graph(1, &global_dump_graph);
Node* parent = graph.CreateNode(MemoryAllocatorNodeId(1), "simple", false);
Node* child =
graph.CreateNode(MemoryAllocatorNodeId(1), "simple/child", false);
ASSERT_EQ(parent->parent(), graph.root());
ASSERT_EQ(child->parent(), parent);
}
TEST(ProcessTest, WeakAndExplicit) {
GlobalNodeGraph global_dump_graph;
Process graph(1, &global_dump_graph);
Node* first =
graph.CreateNode(MemoryAllocatorNodeId(1), "simple/test/1", true);
Node* second =
graph.CreateNode(MemoryAllocatorNodeId(2), "simple/test/2", false);
ASSERT_TRUE(first->is_weak());
ASSERT_FALSE(second->is_weak());
ASSERT_TRUE(first->is_explicit());
ASSERT_TRUE(second->is_explicit());
Node* parent = graph.FindNode("simple/test");
ASSERT_NE(parent, nullptr);
ASSERT_FALSE(parent->is_weak());
ASSERT_FALSE(parent->is_explicit());
Node* grandparent = graph.FindNode("simple");
ASSERT_NE(grandparent, nullptr);
ASSERT_FALSE(grandparent->is_weak());
ASSERT_FALSE(grandparent->is_explicit());
}
TEST(NodeTest, GetChild) {
GlobalNodeGraph global_dump_graph;
Node node(global_dump_graph.shared_memory_graph(), nullptr);
ASSERT_EQ(node.GetChild("test"), nullptr);
Node child(global_dump_graph.shared_memory_graph(), &node);
node.InsertChild("child", &child);
ASSERT_EQ(node.GetChild("child"), &child);
}
TEST(NodeTest, InsertChild) {
GlobalNodeGraph global_dump_graph;
Node node(global_dump_graph.shared_memory_graph(), nullptr);
ASSERT_EQ(node.GetChild("test"), nullptr);
Node child(global_dump_graph.shared_memory_graph(), &node);
node.InsertChild("child", &child);
ASSERT_EQ(node.GetChild("child"), &child);
}
TEST(NodeTest, AddEntry) {
GlobalNodeGraph global_dump_graph;
Node node(global_dump_graph.shared_memory_graph(), nullptr);
node.AddEntry("scalar", Node::Entry::ScalarUnits::kBytes, 100ul);
ASSERT_EQ(node.entries()->size(), 1ul);
node.AddEntry("string", "data");
ASSERT_EQ(node.entries()->size(), 2ul);
auto scalar = node.entries()->find("scalar");
ASSERT_EQ(scalar->first, "scalar");
ASSERT_EQ(scalar->second.units, Node::Entry::ScalarUnits::kBytes);
ASSERT_EQ(scalar->second.value_uint64, 100ul);
auto string = node.entries()->find("string");
ASSERT_EQ(string->first, "string");
ASSERT_EQ(string->second.value_string, "data");
}
} // namespace trace_processor
} // namespace perfetto