rust/crates/ra_ide_api/src/status.rs

192 lines
5.1 KiB
Rust
Raw Normal View History

2019-01-25 08:20:52 -06:00
use std::{
fmt,
iter::FromIterator,
sync::Arc,
};
2019-01-25 07:10:34 -06:00
use ra_syntax::{TreeArc, SyntaxNode, Parse, AstNode};
2019-01-22 15:15:03 -06:00
use ra_db::{
FileTextQuery, SourceRootId,
2019-01-25 08:20:52 -06:00
salsa::{Database, debug::{DebugQueryTable, TableEntry}},
2019-01-22 15:15:03 -06:00
};
use hir::MacroFile;
2019-01-22 15:15:03 -06:00
2019-01-25 08:20:52 -06:00
use crate::{
FileId, db::RootDatabase,
symbol_index::{SymbolIndex, LibrarySymbolsQuery},
};
2019-01-22 15:15:03 -06:00
2019-01-26 11:33:33 -06:00
pub(crate) fn syntax_tree_stats(db: &RootDatabase) -> SyntaxTreeStats {
db.query(ra_db::ParseQuery).entries::<SyntaxTreeStats>()
}
pub(crate) fn macro_syntax_tree_stats(db: &RootDatabase) -> SyntaxTreeStats {
db.query(hir::db::ParseMacroQuery).entries::<SyntaxTreeStats>()
2019-01-26 11:33:33 -06:00
}
2019-01-22 15:15:03 -06:00
pub(crate) fn status(db: &RootDatabase) -> String {
2019-01-25 08:20:52 -06:00
let files_stats = db.query(FileTextQuery).entries::<FilesStats>();
2019-01-26 11:33:33 -06:00
let syntax_tree_stats = syntax_tree_stats(db);
let macro_syntax_tree_stats = macro_syntax_tree_stats(db);
2019-02-08 05:49:43 -06:00
let symbols_stats = db.query(LibrarySymbolsQuery).entries::<LibrarySymbolsStats>();
2019-01-25 08:20:52 -06:00
format!(
"{}\n{}\n{}\n{} (macros)\n\n\nmemory:\n{}\ngc {:?} seconds ago",
2019-01-26 11:33:33 -06:00
files_stats,
symbols_stats,
syntax_tree_stats,
macro_syntax_tree_stats,
2019-01-26 12:12:16 -06:00
MemoryStats::current(),
2019-01-26 11:33:33 -06:00
db.last_gc.elapsed().as_secs(),
2019-01-25 08:20:52 -06:00
)
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 {
2019-01-25 08:20:52 -06:00
write!(fmt, "{} ({}) files", self.total, self.size)
}
}
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
retained_size: Bytes,
}
impl fmt::Display for SyntaxTreeStats {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
2019-02-08 05:49:43 -06:00
write!(fmt, "{} trees, {} ({}) retained", self.total, self.retained, self.retained_size,)
2019-01-25 07:10:34 -06:00
}
}
impl FromIterator<TableEntry<FileId, Parse>> for SyntaxTreeStats {
fn from_iter<T>(iter: T) -> SyntaxTreeStats
where
T: IntoIterator<Item = TableEntry<FileId, Parse>>,
{
let mut res = SyntaxTreeStats::default();
for entry in iter {
res.total += 1;
if let Some(tree) = entry.value.as_ref().map(|it| &it.tree) {
res.retained += 1;
res.retained_size += tree.syntax().memory_size_of_subtree();
}
}
res
}
}
impl FromIterator<TableEntry<MacroFile, Option<TreeArc<SyntaxNode>>>> for SyntaxTreeStats {
2019-01-25 08:20:52 -06:00
fn from_iter<T>(iter: T) -> SyntaxTreeStats
where
T: IntoIterator<Item = TableEntry<MacroFile, Option<TreeArc<SyntaxNode>>>>,
2019-01-25 08:20:52 -06:00
{
let mut res = SyntaxTreeStats::default();
for entry in iter {
res.total += 1;
if let Some(tree) = entry.value.and_then(|it| it) {
2019-01-25 08:20:52 -06:00
res.retained += 1;
res.retained_size += tree.memory_size_of_subtree();
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 {
2019-01-25 12:10:28 -06:00
write!(fmt, "{} ({}) symbols", self.total, self.size,)
2019-01-25 08:20:52 -06:00
}
}
impl FromIterator<TableEntry<SourceRootId, Arc<SymbolIndex>>> for LibrarySymbolsStats {
fn from_iter<T>(iter: T) -> LibrarySymbolsStats
where
T: IntoIterator<Item = TableEntry<SourceRootId, Arc<SymbolIndex>>>,
{
let mut res = LibrarySymbolsStats::default();
for entry in iter {
let value = entry.value.unwrap();
res.total += value.len();
2019-01-25 12:10:28 -06:00
res.size += value.memory_size();
2019-01-25 08:20:52 -06:00
}
res
}
}
2019-01-26 12:12:16 -06:00
struct MemoryStats {
allocated: Bytes,
resident: Bytes,
}
impl MemoryStats {
2019-01-28 06:52:21 -06:00
#[cfg(feature = "jemalloc")]
2019-01-26 12:12:16 -06:00
fn current() -> MemoryStats {
jemalloc_ctl::epoch().unwrap();
MemoryStats {
allocated: Bytes(jemalloc_ctl::stats::allocated().unwrap()),
resident: Bytes(jemalloc_ctl::stats::resident().unwrap()),
}
}
2019-01-28 06:52:21 -06:00
#[cfg(not(feature = "jemalloc"))]
fn current() -> MemoryStats {
2019-02-08 05:49:43 -06:00
MemoryStats { allocated: Bytes(0), resident: Bytes(0) }
2019-01-28 06:52:21 -06:00
}
2019-01-26 12:12:16 -06:00
}
impl fmt::Display for MemoryStats {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
2019-02-08 05:49:43 -06:00
write!(fmt, "{} allocated {} resident", self.allocated, self.resident,)
2019-01-26 12:12:16 -06:00
}
}
2019-01-25 08:20:52 -06:00
#[derive(Default)]
struct Bytes(usize);
impl fmt::Display for Bytes {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let bytes = self.0;
if bytes < 4096 {
return write!(f, "{} bytes", bytes);
}
let kb = bytes / 1024;
if kb < 4096 {
return write!(f, "{}kb", kb);
}
let mb = kb / 1024;
write!(f, "{}mb", mb)
}
}
impl std::ops::AddAssign<usize> for Bytes {
fn add_assign(&mut self, x: usize) {
self.0 += x;
2019-01-25 07:10:34 -06:00
}
2019-01-22 15:15:03 -06:00
}