blob: 0a9f173b4e394b90aac261403d396f8379debb3e [file] [log] [blame]
Yabin Cuiecb9a302015-07-01 10:00:52 -07001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef SIMPLE_PERF_CALLCHAIN_H_
18#define SIMPLE_PERF_CALLCHAIN_H_
19
Yabin Cuib64a8632016-05-24 18:23:33 -070020#include <string.h>
21
22#include <algorithm>
Yabin Cui9970a232016-06-29 12:18:11 -070023#include <functional>
Yabin Cuiecb9a302015-07-01 10:00:52 -070024#include <memory>
Yabin Cuib64a8632016-05-24 18:23:33 -070025#include <queue>
Yabin Cuiecb9a302015-07-01 10:00:52 -070026#include <vector>
27
Yabin Cuib64a8632016-05-24 18:23:33 -070028#include <android-base/logging.h>
Yabin Cuiecb9a302015-07-01 10:00:52 -070029
Yabin Cuifaa7b922021-01-11 17:35:57 -080030namespace simpleperf {
31
Yabin Cuib64a8632016-05-24 18:23:33 -070032template <typename EntryT>
Yabin Cuiecb9a302015-07-01 10:00:52 -070033struct CallChainNode {
34 uint64_t period;
35 uint64_t children_period;
Yabin Cuib64a8632016-05-24 18:23:33 -070036 std::vector<EntryT*> chain;
Yabin Cuiecb9a302015-07-01 10:00:52 -070037 std::vector<std::unique_ptr<CallChainNode>> children;
38};
39
Yabin Cuib64a8632016-05-24 18:23:33 -070040template <typename EntryT>
Yabin Cuiecb9a302015-07-01 10:00:52 -070041struct CallChainRoot {
Yabin Cuib64a8632016-05-24 18:23:33 -070042 typedef CallChainNode<EntryT> NodeT;
Yabin Cui0c093f32017-04-19 11:48:44 -070043 // If duplicated = true, this call tree is part of another call tree.
44 // And we don't need to show it in brief callgraph report mode.
45 bool duplicated;
Yabin Cuiecb9a302015-07-01 10:00:52 -070046 uint64_t children_period;
Yabin Cuib64a8632016-05-24 18:23:33 -070047 std::vector<std::unique_ptr<NodeT>> children;
Yabin Cuiecb9a302015-07-01 10:00:52 -070048
Yabin Cui0c093f32017-04-19 11:48:44 -070049 CallChainRoot() : duplicated(false), children_period(0) {}
Yabin Cuib64a8632016-05-24 18:23:33 -070050
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +020051 void AddCallChain(const std::vector<EntryT*>& callchain, uint64_t period,
52 std::function<bool(const EntryT*, const EntryT*)> is_same_sample) {
Yabin Cuib64a8632016-05-24 18:23:33 -070053 children_period += period;
Yabin Cui9970a232016-06-29 12:18:11 -070054 NodeT* p = FindMatchingNode(children, callchain[0], is_same_sample);
Yabin Cuib64a8632016-05-24 18:23:33 -070055 if (p == nullptr) {
56 std::unique_ptr<NodeT> new_node = AllocateNode(callchain, 0, period, 0);
57 children.push_back(std::move(new_node));
58 return;
59 }
60 size_t callchain_pos = 0;
61 while (true) {
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +020062 size_t match_length = GetMatchingLengthInNode(p, callchain, callchain_pos, is_same_sample);
Yabin Cuib64a8632016-05-24 18:23:33 -070063 CHECK_GT(match_length, 0u);
64 callchain_pos += match_length;
65 bool find_child = true;
66 if (match_length < p->chain.size()) {
67 SplitNode(p, match_length);
68 find_child = false; // No need to find matching node in p->children.
69 }
70 if (callchain_pos == callchain.size()) {
71 p->period += period;
72 return;
73 }
74 p->children_period += period;
75 if (find_child) {
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +020076 NodeT* np = FindMatchingNode(p->children, callchain[callchain_pos], is_same_sample);
Yabin Cuib64a8632016-05-24 18:23:33 -070077 if (np != nullptr) {
78 p = np;
79 continue;
80 }
81 }
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +020082 std::unique_ptr<NodeT> new_node = AllocateNode(callchain, callchain_pos, period, 0);
Yabin Cuib64a8632016-05-24 18:23:33 -070083 p->children.push_back(std::move(new_node));
84 break;
85 }
Yabin Cuiecb9a302015-07-01 10:00:52 -070086 }
87
Yabin Cuib64a8632016-05-24 18:23:33 -070088 void SortByPeriod() {
89 std::queue<std::vector<std::unique_ptr<NodeT>>*> queue;
90 queue.push(&children);
91 while (!queue.empty()) {
92 std::vector<std::unique_ptr<NodeT>>* v = queue.front();
93 queue.pop();
94 std::sort(v->begin(), v->end(), CallChainRoot::CompareNodeByPeriod);
95 for (auto& node : *v) {
96 if (!node->children.empty()) {
97 queue.push(&node->children);
98 }
99 }
100 }
101 }
102
103 private:
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +0200104 NodeT* FindMatchingNode(const std::vector<std::unique_ptr<NodeT>>& nodes, const EntryT* sample,
105 std::function<bool(const EntryT*, const EntryT*)> is_same_sample) {
Yabin Cuib64a8632016-05-24 18:23:33 -0700106 for (auto& node : nodes) {
Yabin Cui9970a232016-06-29 12:18:11 -0700107 if (is_same_sample(node->chain.front(), sample)) {
Yabin Cuib64a8632016-05-24 18:23:33 -0700108 return node.get();
109 }
110 }
111 return nullptr;
112 }
113
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +0200114 size_t GetMatchingLengthInNode(NodeT* node, const std::vector<EntryT*>& chain, size_t chain_start,
115 std::function<bool(const EntryT*, const EntryT*)> is_same_sample) {
Yabin Cuib64a8632016-05-24 18:23:33 -0700116 size_t i, j;
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +0200117 for (i = 0, j = chain_start; i < node->chain.size() && j < chain.size(); ++i, ++j) {
Yabin Cui9970a232016-06-29 12:18:11 -0700118 if (!is_same_sample(node->chain[i], chain[j])) {
Yabin Cuib64a8632016-05-24 18:23:33 -0700119 break;
120 }
121 }
122 return i;
123 }
124
Yabin Cuib64a8632016-05-24 18:23:33 -0700125 void SplitNode(NodeT* parent, size_t parent_length) {
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +0200126 std::unique_ptr<NodeT> child =
127 AllocateNode(parent->chain, parent_length, parent->period, parent->children_period);
Yabin Cuib64a8632016-05-24 18:23:33 -0700128 child->children = std::move(parent->children);
129 parent->period = 0;
130 parent->children_period = child->period + child->children_period;
131 parent->chain.resize(parent_length);
132 parent->children.clear();
133 parent->children.push_back(std::move(child));
134 }
135
Thiébaud Weksteen4848ee02020-10-23 16:06:59 +0200136 std::unique_ptr<NodeT> AllocateNode(const std::vector<EntryT*>& chain, size_t chain_start,
137 uint64_t period, uint64_t children_period) {
Yabin Cuib64a8632016-05-24 18:23:33 -0700138 std::unique_ptr<NodeT> node(new NodeT);
139 for (size_t i = chain_start; i < chain.size(); ++i) {
140 node->chain.push_back(chain[i]);
141 }
142 node->period = period;
143 node->children_period = children_period;
144 return node;
145 }
146
147 static bool CompareNodeByPeriod(const std::unique_ptr<NodeT>& n1,
148 const std::unique_ptr<NodeT>& n2) {
149 uint64_t period1 = n1->period + n1->children_period;
150 uint64_t period2 = n2->period + n2->children_period;
151 return period1 > period2;
152 }
Yabin Cuiecb9a302015-07-01 10:00:52 -0700153};
154
Yabin Cuifaa7b922021-01-11 17:35:57 -0800155} // namespace simpleperf
156
Yabin Cuiecb9a302015-07-01 10:00:52 -0700157#endif // SIMPLE_PERF_CALLCHAIN_H_