From 6314e62cfb06ea7bbe5f530f2824010be0ffa4c7 Mon Sep 17 00:00:00 2001 From: Aleksey Kladov Date: Sun, 16 Jun 2019 19:19:38 +0300 Subject: add analysis-bench to benchmark incremental analysis MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Can be used like this: ``` $ cargo run --release -p ra_cli -- \ analysis-bench ../chalk/ \ --complete ../chalk/chalk-engine/src/logic.rs:94:0 loading: 225.970093ms from scratch: 8.492373325s no change: 445.265µs trivial change: 95.631242ms ``` Or like this: ``` $ cargo run --release -p ra_cli -- \ analysis-bench ../chalk/ \ --highlight ../chalk/chalk-engine/src/logic.rs loading: 209.873484ms from scratch: 9.504916942s no change: 7.731119ms trivial change: 124.984039ms ``` "from scratch" includes initial analysis of the relevant bits of the project "no change" just asks the same question for the second time. It measures overhead on assembling the answer outside of salsa. "trivial change" doesn't do an actual salsa change, it just advances the revision. This test how fast is salsa at validating things. --- crates/ra_cli/src/main.rs | 47 ++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 46 insertions(+), 1 deletion(-) (limited to 'crates/ra_cli/src/main.rs') diff --git a/crates/ra_cli/src/main.rs b/crates/ra_cli/src/main.rs index 1db98aec1..5adf8b096 100644 --- a/crates/ra_cli/src/main.rs +++ b/crates/ra_cli/src/main.rs @@ -1,4 +1,5 @@ mod analysis_stats; +mod analysis_bench; use std::{io::Read, error::Error}; @@ -26,6 +27,27 @@ fn main() -> Result<()> { .arg(Arg::with_name("only").short("o").takes_value(true)) .arg(Arg::with_name("path")), ) + .subcommand( + SubCommand::with_name("analysis-bench") + .arg(Arg::with_name("verbose").short("v").long("verbose")) + .arg( + Arg::with_name("highlight") + .long("highlight") + .takes_value(true) + .conflicts_with("complete") + .value_name("PATH") + .help("highlight this file"), + ) + .arg( + Arg::with_name("complete") + .long("complete") + .takes_value(true) + .conflicts_with("highlight") + .value_name("PATH:LINE:COLUMN") + .help("compute completions at this location"), + ) + .arg(Arg::with_name("path").value_name("PATH").help("project to analyze")), + ) .get_matches(); match matches.subcommand() { ("parse", Some(matches)) => { @@ -51,7 +73,25 @@ fn main() -> Result<()> { let verbose = matches.is_present("verbose"); let path = matches.value_of("path").unwrap_or(""); let only = matches.value_of("only"); - analysis_stats::run(verbose, path, only)?; + analysis_stats::run(verbose, path.as_ref(), only)?; + } + ("analysis-bench", Some(matches)) => { + let verbose = matches.is_present("verbose"); + let path = matches.value_of("path").unwrap_or(""); + let op = if let Some(path) = matches.value_of("highlight") { + analysis_bench::Op::Highlight { path: path.into() } + } else if let Some(path_line_col) = matches.value_of("complete") { + let (path_line, column) = rsplit_at_char(path_line_col, ':')?; + let (path, line) = rsplit_at_char(path_line, ':')?; + analysis_bench::Op::Complete { + path: path.into(), + line: line.parse()?, + column: column.parse()?, + } + } else { + panic!("either --highlight or --complete must be set") + }; + analysis_bench::run(verbose, path.as_ref(), op)?; } _ => unreachable!(), } @@ -68,3 +108,8 @@ fn read_stdin() -> Result { std::io::stdin().read_to_string(&mut buff)?; Ok(buff) } + +fn rsplit_at_char(s: &str, c: char) -> Result<(&str, &str)> { + let idx = s.rfind(":").ok_or_else(|| format!("no `{}` in {}", c, s))?; + Ok((&s[..idx], &s[idx + 1..])) +} -- cgit v1.2.3