rust/crates/ide/src/status.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

165 lines
4.9 KiB
Rust
Raw Normal View History

use std::{fmt, sync::Arc};
2019-01-25 07:10:34 -06:00
use hir::{ExpandResult, MacroFile};
2020-10-24 03:39:57 -05:00
use ide_db::base_db::{
salsa::debug::{DebugQueryTable, TableEntry},
CrateId, FileId, FileTextQuery, SourceDatabase, SourceRootId,
2019-01-22 15:15:03 -06:00
};
2020-08-13 09:39:16 -05:00
use ide_db::{
2020-02-06 05:52:32 -06:00
symbol_index::{LibrarySymbolsQuery, SymbolIndex},
RootDatabase,
};
use itertools::Itertools;
2020-08-13 09:39:16 -05:00
use profile::{memory_usage, Bytes};
2021-06-28 11:50:24 -05:00
use std::env;
use stdx::format_to;
2020-08-12 11:26:51 -05:00
use syntax::{ast, Parse, SyntaxNode};
2019-01-22 15:15:03 -06:00
fn syntax_tree_stats(db: &RootDatabase) -> SyntaxTreeStats {
2020-10-24 03:39:57 -05:00
ide_db::base_db::ParseQuery.in_db(db).entries::<SyntaxTreeStats>()
}
fn macro_syntax_tree_stats(db: &RootDatabase) -> SyntaxTreeStats {
hir::db::ParseMacroExpansionQuery.in_db(db).entries::<SyntaxTreeStats>()
2019-01-26 11:33:33 -06:00
}
2020-05-31 03:14:36 -05:00
// Feature: Status
//
// Shows internal statistic about memory usage of rust-analyzer.
//
// |===
// | Editor | Action Name
//
2022-08-01 06:47:09 -05:00
// | VS Code | **rust-analyzer: Status**
2020-05-31 03:14:36 -05:00
// |===
// image::https://user-images.githubusercontent.com/48062697/113065584-05f34500-91b1-11eb-98cc-5c196f76be7f.gif[]
pub(crate) fn status(db: &RootDatabase, file_id: Option<FileId>) -> String {
let mut buf = String::new();
format_to!(buf, "{}\n", FileTextQuery.in_db(db).entries::<FilesStats>());
format_to!(buf, "{}\n", LibrarySymbolsQuery.in_db(db).entries::<LibrarySymbolsStats>());
format_to!(buf, "{}\n", syntax_tree_stats(db));
2021-06-18 13:22:03 -05:00
format_to!(buf, "{} (Macros)\n", macro_syntax_tree_stats(db));
2021-06-19 01:51:44 -05:00
format_to!(buf, "{} in total\n", memory_usage());
2021-06-28 11:50:24 -05:00
if env::var("RA_COUNT").is_ok() {
2021-06-29 14:34:52 -05:00
format_to!(buf, "\nCounts:\n{}", profile::countme::get_all());
2021-06-28 11:50:24 -05:00
}
if let Some(file_id) = file_id {
2021-06-18 13:22:03 -05:00
format_to!(buf, "\nFile info:\n");
let crates = crate::parent_module::crates_for(db, file_id);
if crates.is_empty() {
format_to!(buf, "Does not belong to any crate");
}
let crate_graph = db.crate_graph();
for krate in crates {
let display_crate = |krate: CrateId| match &crate_graph[krate].display_name {
Some(it) => format!("{it}({krate:?})"),
None => format!("{krate:?}"),
};
format_to!(buf, "Crate: {}\n", display_crate(krate));
let deps = crate_graph[krate]
.dependencies
.iter()
.map(|dep| format!("{}={:?}", dep.name, dep.crate_id))
.format(", ");
format_to!(buf, "Dependencies: {}\n", deps);
}
}
buf.trim().to_string()
2019-01-25 07:10:34 -06:00
}
#[derive(Default)]
struct FilesStats {
total: usize,
2019-01-25 08:20:52 -06:00
size: Bytes,
2019-01-25 07:10:34 -06:00
}
impl fmt::Display for FilesStats {
2022-07-20 08:02:08 -05:00
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
2021-06-19 01:51:44 -05:00
write!(fmt, "{} of files", self.size)
2019-01-25 08:20:52 -06:00
}
}
impl FromIterator<TableEntry<FileId, Arc<String>>> for FilesStats {
fn from_iter<T>(iter: T) -> FilesStats
where
T: IntoIterator<Item = TableEntry<FileId, Arc<String>>>,
{
let mut res = FilesStats::default();
for entry in iter {
res.total += 1;
res.size += entry.value.unwrap().len();
}
res
}
}
#[derive(Default)]
2019-01-26 11:33:33 -06:00
pub(crate) struct SyntaxTreeStats {
2019-01-25 08:20:52 -06:00
total: usize,
2019-01-26 11:33:33 -06:00
pub(crate) retained: usize,
2019-01-25 08:20:52 -06:00
}
impl fmt::Display for SyntaxTreeStats {
2022-07-20 08:02:08 -05:00
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
2021-06-19 04:14:15 -05:00
write!(fmt, "{} trees, {} preserved", self.total, self.retained)
2019-01-25 07:10:34 -06:00
}
}
2019-07-18 14:29:20 -05:00
impl FromIterator<TableEntry<FileId, Parse<ast::SourceFile>>> for SyntaxTreeStats {
fn from_iter<T>(iter: T) -> SyntaxTreeStats
where
2019-07-18 14:29:20 -05:00
T: IntoIterator<Item = TableEntry<FileId, Parse<ast::SourceFile>>>,
{
let mut res = SyntaxTreeStats::default();
for entry in iter {
res.total += 1;
res.retained += entry.value.is_some() as usize;
}
res
}
}
impl<M> FromIterator<TableEntry<MacroFile, ExpandResult<Option<(Parse<SyntaxNode>, M)>>>>
2020-11-24 12:00:23 -06:00
for SyntaxTreeStats
{
2019-01-25 08:20:52 -06:00
fn from_iter<T>(iter: T) -> SyntaxTreeStats
where
T: IntoIterator<Item = TableEntry<MacroFile, ExpandResult<Option<(Parse<SyntaxNode>, M)>>>>,
2019-01-25 08:20:52 -06:00
{
let mut res = SyntaxTreeStats::default();
for entry in iter {
res.total += 1;
res.retained += entry.value.is_some() as usize;
2019-01-25 08:20:52 -06:00
}
res
2019-01-25 07:10:34 -06:00
}
2019-01-25 08:20:52 -06:00
}
#[derive(Default)]
struct LibrarySymbolsStats {
total: usize,
2019-01-25 12:10:28 -06:00
size: Bytes,
2019-01-25 08:20:52 -06:00
}
impl fmt::Display for LibrarySymbolsStats {
2022-07-20 08:02:08 -05:00
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
2021-06-28 01:31:54 -05:00
write!(fmt, "{} of index symbols ({})", self.size, self.total)
2019-01-25 08:20:52 -06:00
}
}
2021-11-29 03:36:00 -06:00
impl FromIterator<TableEntry<SourceRootId, Arc<SymbolIndex>>> for LibrarySymbolsStats {
2019-01-25 08:20:52 -06:00
fn from_iter<T>(iter: T) -> LibrarySymbolsStats
where
2021-11-29 03:36:00 -06:00
T: IntoIterator<Item = TableEntry<SourceRootId, Arc<SymbolIndex>>>,
2019-01-25 08:20:52 -06:00
{
let mut res = LibrarySymbolsStats::default();
for entry in iter {
2021-11-29 03:36:00 -06:00
let symbols = entry.value.unwrap();
res.total += symbols.len();
res.size += symbols.memory_size();
2019-01-25 08:20:52 -06:00
}
res
}
}