2019-12-12 22:01:06 +01:00
|
|
|
#include "Profile.h"
|
|
|
|
#include "ProfileModel.h"
|
|
|
|
#include <AK/QuickSort.h>
|
|
|
|
#include <LibCore/CFile.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2019-12-16 18:21:05 +01:00
|
|
|
static void sort_profile_nodes(Vector<NonnullRefPtr<ProfileNode>>& nodes)
|
|
|
|
{
|
|
|
|
quick_sort(nodes.begin(), nodes.end(), [](auto& a, auto& b) {
|
|
|
|
return a->sample_count() >= b->sample_count();
|
|
|
|
});
|
|
|
|
|
|
|
|
for (auto& child : nodes)
|
|
|
|
child->sort_children();
|
|
|
|
}
|
|
|
|
|
2019-12-14 19:10:12 +01:00
|
|
|
Profile::Profile(const JsonArray& json)
|
2019-12-12 22:01:06 +01:00
|
|
|
: m_json(json)
|
|
|
|
{
|
2019-12-14 19:10:12 +01:00
|
|
|
m_first_timestamp = m_json.at(0).as_object().get("timestamp").to_number<u64>();
|
|
|
|
m_last_timestamp = m_json.at(m_json.size() - 1).as_object().get("timestamp").to_number<u64>();
|
|
|
|
|
2019-12-12 22:01:06 +01:00
|
|
|
m_model = ProfileModel::create(*this);
|
2019-12-14 19:25:33 +01:00
|
|
|
|
2019-12-15 17:26:54 +01:00
|
|
|
m_samples.ensure_capacity(m_json.size());
|
|
|
|
for (auto& sample_value : m_json.values()) {
|
|
|
|
auto& sample_object = sample_value.as_object();
|
|
|
|
|
|
|
|
Sample sample;
|
|
|
|
sample.timestamp = sample_object.get("timestamp").to_number<u64>();
|
|
|
|
sample.in_kernel = sample_object.get("frames").as_array().at(1).as_object().get("address").to_number<u32>() < (8 * MB);
|
|
|
|
|
|
|
|
auto frames_value = sample_object.get("frames");
|
|
|
|
auto& frames_array = frames_value.as_array();
|
2019-12-16 18:21:05 +01:00
|
|
|
for (int i = frames_array.size() - 1; i >= 1; --i) {
|
2019-12-15 17:26:54 +01:00
|
|
|
auto& frame_value = frames_array.at(i);
|
|
|
|
auto& frame_object = frame_value.as_object();
|
|
|
|
Frame frame;
|
|
|
|
frame.symbol = frame_object.get("symbol").as_string_or({});
|
|
|
|
frame.address = frame_object.get("address").as_u32();
|
|
|
|
frame.offset = frame_object.get("offset").as_u32();
|
|
|
|
sample.frames.append(move(frame));
|
|
|
|
};
|
2019-12-15 18:08:20 +01:00
|
|
|
|
|
|
|
m_deepest_stack_depth = max((u32)frames_array.size(), m_deepest_stack_depth);
|
|
|
|
|
2019-12-15 17:26:54 +01:00
|
|
|
m_samples.append(move(sample));
|
|
|
|
}
|
|
|
|
|
|
|
|
rebuild_tree();
|
2019-12-12 22:01:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Profile::~Profile()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
GModel& Profile::model()
|
|
|
|
{
|
|
|
|
return *m_model;
|
|
|
|
}
|
|
|
|
|
2019-12-14 19:10:12 +01:00
|
|
|
void Profile::rebuild_tree()
|
2019-12-12 22:01:06 +01:00
|
|
|
{
|
2019-12-16 18:21:05 +01:00
|
|
|
Vector<NonnullRefPtr<ProfileNode>> roots;
|
2019-12-12 22:01:06 +01:00
|
|
|
|
2019-12-14 18:44:29 +01:00
|
|
|
auto find_or_create_root = [&roots](const String& symbol, u32 address, u32 offset, u64 timestamp) -> ProfileNode& {
|
2019-12-12 22:01:06 +01:00
|
|
|
for (int i = 0; i < roots.size(); ++i) {
|
|
|
|
auto& root = roots[i];
|
2019-12-16 18:21:05 +01:00
|
|
|
if (root->symbol() == symbol) {
|
2019-12-12 22:01:06 +01:00
|
|
|
return root;
|
|
|
|
}
|
|
|
|
}
|
2019-12-14 18:44:29 +01:00
|
|
|
auto new_root = ProfileNode::create(symbol, address, offset, timestamp);
|
2019-12-12 22:01:06 +01:00
|
|
|
roots.append(new_root);
|
|
|
|
return new_root;
|
|
|
|
};
|
|
|
|
|
2019-12-15 17:26:54 +01:00
|
|
|
for (auto& sample : m_samples) {
|
2019-12-14 19:10:12 +01:00
|
|
|
if (has_timestamp_filter_range()) {
|
2019-12-15 17:26:54 +01:00
|
|
|
auto timestamp = sample.timestamp;
|
2019-12-14 19:10:12 +01:00
|
|
|
if (timestamp < m_timestamp_filter_range_start || timestamp > m_timestamp_filter_range_end)
|
2019-12-15 17:26:54 +01:00
|
|
|
continue;
|
2019-12-14 19:10:12 +01:00
|
|
|
}
|
2019-12-14 18:44:29 +01:00
|
|
|
|
2019-12-12 22:01:06 +01:00
|
|
|
ProfileNode* node = nullptr;
|
|
|
|
|
2019-12-16 18:21:05 +01:00
|
|
|
auto for_each_frame = [&]<typename Callback>(Callback callback)
|
|
|
|
{
|
|
|
|
if (!m_inverted) {
|
|
|
|
for (int i = 0; i < sample.frames.size(); ++i) {
|
|
|
|
if (callback(sample.frames.at(i)) == IterationDecision::Break)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (int i = sample.frames.size() - 1; i >= 0; --i) {
|
|
|
|
if (callback(sample.frames.at(i)) == IterationDecision::Break)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
for_each_frame([&](const Frame& frame) {
|
2019-12-15 17:26:54 +01:00
|
|
|
auto& symbol = frame.symbol;
|
|
|
|
auto& address = frame.address;
|
|
|
|
auto& offset = frame.offset;
|
2019-12-12 22:01:06 +01:00
|
|
|
|
|
|
|
if (symbol.is_empty())
|
2019-12-16 18:21:05 +01:00
|
|
|
return IterationDecision::Break;
|
2019-12-12 22:01:06 +01:00
|
|
|
|
|
|
|
if (!node)
|
2019-12-15 17:26:54 +01:00
|
|
|
node = &find_or_create_root(symbol, address, offset, sample.timestamp);
|
2019-12-12 22:01:06 +01:00
|
|
|
else
|
2019-12-15 17:26:54 +01:00
|
|
|
node = &node->find_or_create_child(symbol, address, offset, sample.timestamp);
|
2019-12-12 22:01:06 +01:00
|
|
|
|
|
|
|
node->increment_sample_count();
|
2019-12-16 18:21:05 +01:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
2019-12-15 17:26:54 +01:00
|
|
|
}
|
2019-12-12 22:01:06 +01:00
|
|
|
|
2019-12-16 18:21:05 +01:00
|
|
|
sort_profile_nodes(roots);
|
2019-12-12 22:01:06 +01:00
|
|
|
|
2019-12-14 19:10:12 +01:00
|
|
|
m_roots = move(roots);
|
|
|
|
m_model->update();
|
|
|
|
}
|
|
|
|
|
|
|
|
OwnPtr<Profile> Profile::load_from_file(const StringView& path)
|
|
|
|
{
|
|
|
|
auto file = CFile::construct(path);
|
|
|
|
if (!file->open(CIODevice::ReadOnly)) {
|
|
|
|
fprintf(stderr, "Unable to open %s, error: %s\n", String(path).characters(), file->error_string());
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto json = JsonValue::from_string(file->read_all());
|
|
|
|
if (!json.is_array()) {
|
|
|
|
fprintf(stderr, "Invalid format (not a JSON array)\n");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto& samples = json.as_array();
|
|
|
|
if (samples.is_empty())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return NonnullOwnPtr<Profile>(NonnullOwnPtr<Profile>::Adopt, *new Profile(move(samples)));
|
2019-12-12 22:01:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileNode::sort_children()
|
|
|
|
{
|
2019-12-16 18:21:05 +01:00
|
|
|
sort_profile_nodes(m_children);
|
2019-12-12 22:01:06 +01:00
|
|
|
}
|
2019-12-14 19:10:12 +01:00
|
|
|
|
|
|
|
void Profile::set_timestamp_filter_range(u64 start, u64 end)
|
|
|
|
{
|
|
|
|
if (m_has_timestamp_filter_range && m_timestamp_filter_range_start == start && m_timestamp_filter_range_end == end)
|
|
|
|
return;
|
|
|
|
m_has_timestamp_filter_range = true;
|
|
|
|
|
|
|
|
m_timestamp_filter_range_start = min(start, end);
|
|
|
|
m_timestamp_filter_range_end = max(start, end);
|
|
|
|
|
|
|
|
rebuild_tree();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Profile::clear_timestamp_filter_range()
|
|
|
|
{
|
|
|
|
if (!m_has_timestamp_filter_range)
|
|
|
|
return;
|
|
|
|
m_has_timestamp_filter_range = false;
|
|
|
|
rebuild_tree();
|
|
|
|
}
|
2019-12-16 18:21:05 +01:00
|
|
|
|
|
|
|
void Profile::set_inverted(bool inverted)
|
|
|
|
{
|
|
|
|
if (m_inverted == inverted)
|
|
|
|
return;
|
|
|
|
m_inverted = inverted;
|
|
|
|
rebuild_tree();
|
|
|
|
}
|