1 | // $ cargo bench --features full,test --bench rust |
2 | // |
3 | // Syn only, useful for profiling: |
4 | // $ RUSTFLAGS='--cfg syn_only' cargo build --release --features full,test --bench rust |
5 | |
6 | #![cfg_attr (not(syn_only), feature(rustc_private))] |
7 | #![recursion_limit = "1024" ] |
8 | #![allow (clippy::cast_lossless, clippy::unnecessary_wraps)] |
9 | |
10 | #[macro_use ] |
11 | #[path = "../tests/macros/mod.rs" ] |
12 | mod macros; |
13 | |
14 | #[path = "../tests/common/mod.rs" ] |
15 | mod common; |
16 | #[path = "../tests/repo/mod.rs" ] |
17 | mod repo; |
18 | |
19 | use std::fs; |
20 | use std::time::{Duration, Instant}; |
21 | |
22 | #[cfg (not(syn_only))] |
23 | mod tokenstream_parse { |
24 | use proc_macro2::TokenStream; |
25 | use std::str::FromStr; |
26 | |
27 | pub fn bench(content: &str) -> Result<(), ()> { |
28 | TokenStream::from_str(content).map(drop).map_err(drop) |
29 | } |
30 | } |
31 | |
32 | mod syn_parse { |
33 | pub fn bench(content: &str) -> Result<(), ()> { |
34 | syn::parse_file(content).map(drop).map_err(drop) |
35 | } |
36 | } |
37 | |
38 | #[cfg (not(syn_only))] |
39 | mod librustc_parse { |
40 | extern crate rustc_data_structures; |
41 | extern crate rustc_error_messages; |
42 | extern crate rustc_errors; |
43 | extern crate rustc_parse; |
44 | extern crate rustc_session; |
45 | extern crate rustc_span; |
46 | |
47 | use rustc_data_structures::sync::Lrc; |
48 | use rustc_error_messages::FluentBundle; |
49 | use rustc_errors::{emitter::Emitter, translation::Translate, Diagnostic, Handler}; |
50 | use rustc_session::parse::ParseSess; |
51 | use rustc_span::source_map::{FilePathMapping, SourceMap}; |
52 | use rustc_span::{edition::Edition, FileName}; |
53 | |
54 | pub fn bench(content: &str) -> Result<(), ()> { |
55 | struct SilentEmitter; |
56 | |
57 | impl Emitter for SilentEmitter { |
58 | fn emit_diagnostic(&mut self, _diag: &Diagnostic) {} |
59 | fn source_map(&self) -> Option<&Lrc<SourceMap>> { |
60 | None |
61 | } |
62 | } |
63 | |
64 | impl Translate for SilentEmitter { |
65 | fn fluent_bundle(&self) -> Option<&Lrc<FluentBundle>> { |
66 | None |
67 | } |
68 | fn fallback_fluent_bundle(&self) -> &FluentBundle { |
69 | panic!("silent emitter attempted to translate a diagnostic" ); |
70 | } |
71 | } |
72 | |
73 | rustc_span::create_session_if_not_set_then(Edition::Edition2018, |_| { |
74 | let cm = Lrc::new(SourceMap::new(FilePathMapping::empty())); |
75 | let emitter = Box::new(SilentEmitter); |
76 | let handler = Handler::with_emitter(false, None, emitter); |
77 | let sess = ParseSess::with_span_handler(handler, cm); |
78 | if let Err(diagnostic) = rustc_parse::parse_crate_from_source_str( |
79 | FileName::Custom("bench" .to_owned()), |
80 | content.to_owned(), |
81 | &sess, |
82 | ) { |
83 | diagnostic.cancel(); |
84 | return Err(()); |
85 | }; |
86 | Ok(()) |
87 | }) |
88 | } |
89 | } |
90 | |
91 | #[cfg (not(syn_only))] |
92 | mod read_from_disk { |
93 | pub fn bench(content: &str) -> Result<(), ()> { |
94 | _ = content; |
95 | Ok(()) |
96 | } |
97 | } |
98 | |
99 | fn exec(mut codepath: impl FnMut(&str) -> Result<(), ()>) -> Duration { |
100 | let begin = Instant::now(); |
101 | let mut success = 0; |
102 | let mut total = 0; |
103 | |
104 | walkdir::WalkDir::new("tests/rust/src" ) |
105 | .into_iter() |
106 | .filter_entry(repo::base_dir_filter) |
107 | .for_each(|entry| { |
108 | let entry = entry.unwrap(); |
109 | let path = entry.path(); |
110 | if path.is_dir() { |
111 | return; |
112 | } |
113 | let content = fs::read_to_string(path).unwrap(); |
114 | let ok = codepath(&content).is_ok(); |
115 | success += ok as usize; |
116 | total += 1; |
117 | if !ok { |
118 | eprintln!("FAIL {}" , path.display()); |
119 | } |
120 | }); |
121 | |
122 | assert_eq!(success, total); |
123 | begin.elapsed() |
124 | } |
125 | |
126 | fn main() { |
127 | repo::clone_rust(); |
128 | |
129 | macro_rules! testcases { |
130 | ($($(#[$cfg:meta])* $name:ident,)*) => { |
131 | [ |
132 | $( |
133 | $(#[$cfg])* |
134 | (stringify!($name), $name::bench as fn(&str) -> Result<(), ()>), |
135 | )* |
136 | ] |
137 | }; |
138 | } |
139 | |
140 | #[cfg (not(syn_only))] |
141 | { |
142 | let mut lines = 0; |
143 | let mut files = 0; |
144 | exec(|content| { |
145 | lines += content.lines().count(); |
146 | files += 1; |
147 | Ok(()) |
148 | }); |
149 | eprintln!(" \n{} lines in {} files" , lines, files); |
150 | } |
151 | |
152 | for (name, f) in testcases!( |
153 | #[cfg (not(syn_only))] |
154 | read_from_disk, |
155 | #[cfg (not(syn_only))] |
156 | tokenstream_parse, |
157 | syn_parse, |
158 | #[cfg (not(syn_only))] |
159 | librustc_parse, |
160 | ) { |
161 | eprint!("{:20}" , format!("{}:" , name)); |
162 | let elapsed = exec(f); |
163 | eprintln!( |
164 | "elapsed={}.{:03}s" , |
165 | elapsed.as_secs(), |
166 | elapsed.subsec_millis(), |
167 | ); |
168 | } |
169 | eprintln!(); |
170 | } |
171 | |