1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
//===-- MemoryTreeTests.cpp -------------------------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "support/MemoryTree.h"
#include "support/TestTracer.h"
#include "support/Trace.h"
#include "llvm/Support/Allocator.h"
#include "gmock/gmock.h"
#include "gtest/gtest.h"
#include <ostream>
namespace clang {
namespace clangd {
namespace {
using testing::Contains;
using testing::ElementsAre;
using testing::IsEmpty;
using testing::UnorderedElementsAre;
MATCHER_P2(WithNameAndSize, Name, Size, "") {
return arg.first == Name &&
arg.getSecond().total() == static_cast<size_t>(Size);
}
TEST(MemoryTree, Basics) {
MemoryTree MT;
EXPECT_EQ(MT.total(), 0U);
EXPECT_THAT(MT.children(), IsEmpty());
MT.addUsage(42);
EXPECT_EQ(MT.total(), 42U);
EXPECT_THAT(MT.children(), IsEmpty());
MT.child("leaf").addUsage(1);
EXPECT_EQ(MT.total(), 43U);
EXPECT_THAT(MT.children(), UnorderedElementsAre(WithNameAndSize("leaf", 1)));
// child should be idempotent.
MT.child("leaf").addUsage(1);
EXPECT_EQ(MT.total(), 44U);
EXPECT_THAT(MT.children(), UnorderedElementsAre(WithNameAndSize("leaf", 2)));
}
TEST(MemoryTree, DetailedNodesWithoutDetails) {
MemoryTree MT;
MT.detail("should_be_ignored").addUsage(2);
EXPECT_THAT(MT.children(), IsEmpty());
EXPECT_EQ(MT.total(), 2U);
// Make sure children from details are merged.
MT.detail("first_detail").child("leaf").addUsage(1);
MT.detail("second_detail").child("leaf").addUsage(1);
EXPECT_THAT(MT.children(), Contains(WithNameAndSize("leaf", 2)));
}
TEST(MemoryTree, DetailedNodesWithDetails) {
llvm::BumpPtrAllocator Alloc;
MemoryTree MT(&Alloc);
{
auto &Detail = MT.detail("first_detail");
Detail.child("leaf").addUsage(1);
EXPECT_THAT(MT.children(), Contains(WithNameAndSize("first_detail", 1)));
EXPECT_THAT(Detail.children(), Contains(WithNameAndSize("leaf", 1)));
}
{
auto &Detail = MT.detail("second_detail");
Detail.child("leaf").addUsage(1);
EXPECT_THAT(MT.children(), Contains(WithNameAndSize("second_detail", 1)));
EXPECT_THAT(Detail.children(), Contains(WithNameAndSize("leaf", 1)));
}
}
TEST(MemoryTree, Record) {
trace::TestTracer Tracer;
static constexpr llvm::StringLiteral MetricName = "memory_usage";
static constexpr trace::Metric OutMetric(MetricName, trace::Metric::Value,
"component_name");
auto AddNodes = [](MemoryTree Root) {
Root.child("leaf").addUsage(1);
{
auto &Detail = Root.detail("detail");
Detail.addUsage(1);
Detail.child("leaf").addUsage(1);
auto &Child = Detail.child("child");
Child.addUsage(1);
Child.child("leaf").addUsage(1);
}
{
auto &Child = Root.child("child");
Child.addUsage(1);
Child.child("leaf").addUsage(1);
}
return Root;
};
llvm::BumpPtrAllocator Alloc;
record(AddNodes(MemoryTree(&Alloc)), "root", OutMetric);
EXPECT_THAT(Tracer.takeMetric(MetricName, "root"), ElementsAre(7));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.leaf"), ElementsAre(1));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.detail"), ElementsAre(4));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.detail.leaf"),
ElementsAre(1));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.detail.child"),
ElementsAre(2));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.detail.child.leaf"),
ElementsAre(1));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.child"), ElementsAre(2));
EXPECT_THAT(Tracer.takeMetric(MetricName, "root.child.leaf"), ElementsAre(1));
}
} // namespace
} // namespace clangd
} // namespace clang
|