2019-07-04 15:05:17 -05:00
|
|
|
use std::{fmt, iter::FromIterator, sync::Arc};
|
2019-01-25 07:10:34 -06:00
|
|
|
|
2020-11-26 09:48:17 -06:00
|
|
|
use hir::{ExpandResult, MacroFile};
|
2020-10-24 03:39:57 -05:00
|
|
|
use ide_db::base_db::{
|
2020-07-07 03:14:48 -05:00
|
|
|
salsa::debug::{DebugQueryTable, TableEntry},
|
2020-09-29 15:05:18 -05:00
|
|
|
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,
|
|
|
|
};
|
2020-09-29 15:05:18 -05:00
|
|
|
use itertools::Itertools;
|
2020-08-13 09:39:16 -05:00
|
|
|
use profile::{memory_usage, Bytes};
|
2020-07-07 03:14:48 -05:00
|
|
|
use rustc_hash::FxHashMap;
|
2021-06-28 11:50:24 -05:00
|
|
|
use std::env;
|
2020-09-29 15:05:18 -05:00
|
|
|
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
|
|
|
|
2019-09-26 04:31:16 -05: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>()
|
2019-06-02 12:15:10 -05:00
|
|
|
}
|
2019-09-26 04:31:16 -05:00
|
|
|
fn macro_syntax_tree_stats(db: &RootDatabase) -> SyntaxTreeStats {
|
2020-11-24 14:57:51 -06:00
|
|
|
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
|
|
|
|
//
|
|
|
|
// | VS Code | **Rust Analyzer: Status**
|
|
|
|
// |===
|
2021-03-30 18:08:10 -05:00
|
|
|
// image::https://user-images.githubusercontent.com/48062697/113065584-05f34500-91b1-11eb-98cc-5c196f76be7f.gif[]
|
2020-09-29 15:05:18 -05:00
|
|
|
pub(crate) fn status(db: &RootDatabase, file_id: Option<FileId>) -> String {
|
|
|
|
let mut buf = String::new();
|
2021-06-18 13:22:03 -05:00
|
|
|
let count = profile::countme::get_all();
|
2020-09-29 15:05:18 -05:00
|
|
|
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:32:33 -05:00
|
|
|
format_to!(buf, "\nCounts:\n{}", count);
|
2021-06-28 11:50:24 -05:00
|
|
|
}
|
2020-09-29 15:05:18 -05:00
|
|
|
|
|
|
|
if let Some(file_id) = file_id {
|
2021-06-18 13:22:03 -05:00
|
|
|
format_to!(buf, "\nFile info:\n");
|
2020-09-29 15:05:18 -05:00
|
|
|
let krate = crate::parent_module::crate_for(db, file_id).pop();
|
|
|
|
match krate {
|
|
|
|
Some(krate) => {
|
|
|
|
let crate_graph = db.crate_graph();
|
2020-10-20 08:38:11 -05:00
|
|
|
let display_crate = |krate: CrateId| match &crate_graph[krate].display_name {
|
2020-09-29 15:05:18 -05:00
|
|
|
Some(it) => format!("{}({:?})", it, krate),
|
|
|
|
None => format!("{:?}", krate),
|
|
|
|
};
|
2021-06-19 03:44:11 -05:00
|
|
|
format_to!(buf, "Crate: {}\n", display_crate(krate));
|
2020-09-29 15:05:18 -05:00
|
|
|
let deps = crate_graph[krate]
|
|
|
|
.dependencies
|
|
|
|
.iter()
|
|
|
|
.map(|dep| format!("{}={:?}", dep.name, dep.crate_id))
|
|
|
|
.format(", ");
|
2021-06-19 03:44:11 -05:00
|
|
|
format_to!(buf, "Dependencies: {}\n", deps);
|
2020-09-29 15:05:18 -05:00
|
|
|
}
|
2021-06-19 01:51:44 -05:00
|
|
|
None => format_to!(buf, "Does not belong to any crate"),
|
2020-09-29 15:05:18 -05:00
|
|
|
}
|
|
|
|
}
|
2021-01-21 10:04:50 -06:00
|
|
|
|
2021-06-18 17:34:00 -05:00
|
|
|
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 {
|
|
|
|
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 {
|
|
|
|
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 {
|
2019-06-02 12:15:10 -05:00
|
|
|
fn from_iter<T>(iter: T) -> SyntaxTreeStats
|
|
|
|
where
|
2019-07-18 14:29:20 -05:00
|
|
|
T: IntoIterator<Item = TableEntry<FileId, Parse<ast::SourceFile>>>,
|
2019-06-02 12:15:10 -05:00
|
|
|
{
|
|
|
|
let mut res = SyntaxTreeStats::default();
|
|
|
|
for entry in iter {
|
|
|
|
res.total += 1;
|
2019-07-19 11:53:42 -05:00
|
|
|
res.retained += entry.value.is_some() as usize;
|
2019-06-02 12:15:10 -05:00
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-26 09:48:17 -06:00
|
|
|
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
|
2020-11-26 09:48:17 -06:00
|
|
|
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;
|
2019-07-19 11:53:42 -05:00
|
|
|
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 {
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-18 01:29:34 -05:00
|
|
|
impl FromIterator<TableEntry<(), Arc<FxHashMap<SourceRootId, SymbolIndex>>>>
|
|
|
|
for LibrarySymbolsStats
|
|
|
|
{
|
2019-01-25 08:20:52 -06:00
|
|
|
fn from_iter<T>(iter: T) -> LibrarySymbolsStats
|
|
|
|
where
|
2020-06-18 01:29:34 -05:00
|
|
|
T: IntoIterator<Item = TableEntry<(), Arc<FxHashMap<SourceRootId, SymbolIndex>>>>,
|
2019-01-25 08:20:52 -06:00
|
|
|
{
|
|
|
|
let mut res = LibrarySymbolsStats::default();
|
|
|
|
for entry in iter {
|
|
|
|
let value = entry.value.unwrap();
|
2020-06-18 01:29:34 -05:00
|
|
|
for symbols in value.values() {
|
|
|
|
res.total += symbols.len();
|
|
|
|
res.size += symbols.memory_size();
|
|
|
|
}
|
2019-01-25 08:20:52 -06:00
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
|
|
|
}
|