ladybird/Userland/DevTools/Profiler/FilesystemEventModel.cpp
Andreas Kling 34344120f2 AK: Make "foo"_string infallible
Stop worrying about tiny OOMs.

Work towards #20405.
2023-08-07 16:03:27 +02:00

188 lines
4.8 KiB
C++

/*
* Copyright (c) 2022, Jakub Berkop <jakub.berkop@gmail.com>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "FilesystemEventModel.h"
#include "Profile.h"
#include <AK/StringBuilder.h>
#include <LibGUI/FileIconProvider.h>
#include <LibSymbolication/Symbolication.h>
#include <stdio.h>
namespace Profiler {
FileEventModel::FileEventModel(Profile& profile)
: m_profile(profile)
{
}
FileEventModel::~FileEventModel()
{
}
FileEventNode& FileEventNode::find_or_create_node(DeprecatedString const& searched_path)
{
// TODO: Optimize this function.
if (searched_path == ""sv) {
return *this;
}
auto lex_path = LexicalPath(searched_path);
auto parts = lex_path.parts();
auto current = parts.take_first();
StringBuilder sb;
sb.join('/', parts);
auto new_s = sb.to_deprecated_string();
for (auto& child : m_children) {
if (child->m_path == current) {
return child->find_or_create_node(new_s);
}
}
if (m_parent) {
for (auto& child : m_children) {
if (child->m_path == current) {
return child->find_or_create_node(new_s);
}
}
return create_recursively(searched_path);
} else {
if (!searched_path.starts_with("/"sv)) {
m_children.append(create(searched_path, this));
return *m_children.last();
}
return create_recursively(searched_path);
}
}
FileEventNode& FileEventNode::create_recursively(DeprecatedString new_path)
{
auto const lex_path = LexicalPath(new_path);
auto parts = lex_path.parts();
if (parts.size() == 1) {
auto new_node = FileEventNode::create(new_path, this);
m_children.append(new_node);
return *new_node.ptr();
} else {
auto new_node = FileEventNode::create(parts.take_first(), this);
m_children.append(new_node);
StringBuilder sb;
sb.join('/', parts);
return new_node->create_recursively(sb.to_deprecated_string());
}
}
void FileEventNode::for_each_parent_node(Function<void(FileEventNode&)> callback)
{
auto* current = this;
while (current) {
callback(*current);
current = current->m_parent;
}
}
GUI::ModelIndex FileEventModel::index(int row, int column, GUI::ModelIndex const& parent) const
{
if (!parent.is_valid()) {
return create_index(row, column, m_profile.file_event_nodes()->children()[row].ptr());
}
auto& remote_parent = *static_cast<FileEventNode*>(parent.internal_data());
return create_index(row, column, remote_parent.children().at(row).ptr());
}
GUI::ModelIndex FileEventModel::parent_index(GUI::ModelIndex const& index) const
{
if (!index.is_valid())
return {};
auto& node = *static_cast<FileEventNode*>(index.internal_data());
if (!node.parent())
return {};
if (node.parent()->parent()) {
auto const& children = node.parent()->parent()->children();
for (size_t row = 0; row < children.size(); ++row) {
if (children.at(row).ptr() == node.parent()) {
return create_index(row, index.column(), node.parent());
}
}
}
auto const& children = node.parent()->children();
for (size_t row = 0; row < children.size(); ++row) {
if (children.at(row).ptr() == &node) {
return create_index(row, index.column(), node.parent());
}
}
VERIFY_NOT_REACHED();
return {};
}
int FileEventModel::row_count(GUI::ModelIndex const& index) const
{
if (!index.is_valid())
return m_profile.file_event_nodes()->children().size();
auto& node = *static_cast<FileEventNode*>(index.internal_data());
return node.children().size();
}
int FileEventModel::column_count(GUI::ModelIndex const&) const
{
return Column::__Count;
}
ErrorOr<String> FileEventModel::column_name(int column) const
{
switch (column) {
case Column::Path:
return "Path"_short_string;
case Column::Count:
return "Event Count"_string;
case Column::Duration:
return "Duration [ms]"_string;
default:
VERIFY_NOT_REACHED();
}
}
GUI::Variant FileEventModel::data(GUI::ModelIndex const& index, GUI::ModelRole role) const
{
if (role == GUI::ModelRole::TextAlignment) {
if (index.column() == Path)
return Gfx::TextAlignment::CenterLeft;
return Gfx::TextAlignment::CenterRight;
}
auto* node = static_cast<FileEventNode*>(index.internal_data());
if (role == GUI::ModelRole::Display) {
if (index.column() == Column::Count) {
return node->count();
}
if (index.column() == Column::Path) {
return node->path();
}
if (index.column() == Column::Duration) {
return node->duration();
}
return {};
}
return {};
}
}