1use crate::alias_analysis::AliasAnalysis;
13use crate::dominator_tree::DominatorTree;
14use crate::egraph::EgraphPass;
15use crate::flowgraph::ControlFlowGraph;
16use crate::ir::Function;
17use crate::isa::TargetIsa;
18use crate::legalizer::simple_legalize;
19use crate::loop_analysis::LoopAnalysis;
20use crate::machinst::{CompiledCode, CompiledCodeStencil};
21use crate::nan_canonicalization::do_nan_canonicalization;
22use crate::remove_constant_phis::do_remove_constant_phis;
23use crate::result::{CodegenResult, CompileResult};
24use crate::settings::{FlagsOrIsa, OptLevel};
25use crate::trace;
26use crate::unreachable_code::eliminate_unreachable_code;
27use crate::verifier::{verify_context, VerifierErrors, VerifierResult};
28use crate::{timing, CompileError};
29#[cfg(feature = "souper-harvest")]
30use alloc::string::String;
31use alloc::vec::Vec;
32use cranelift_control::ControlPlane;
33use target_lexicon::Architecture;
34
35#[cfg(feature = "souper-harvest")]
36use crate::souper_harvest::do_souper_harvest;
37
38pub struct Context {
40 pub func: Function,
42
43 pub cfg: ControlFlowGraph,
45
46 pub domtree: DominatorTree,
48
49 pub loop_analysis: LoopAnalysis,
51
52 pub(crate) compiled_code: Option<CompiledCode>,
54
55 pub want_disasm: bool,
57}
58
59impl Context {
60 pub fn new() -> Self {
65 Self::for_function(Function::new())
66 }
67
68 pub fn for_function(func: Function) -> Self {
73 Self {
74 func,
75 cfg: ControlFlowGraph::new(),
76 domtree: DominatorTree::new(),
77 loop_analysis: LoopAnalysis::new(),
78 compiled_code: None,
79 want_disasm: false,
80 }
81 }
82
83 pub fn clear(&mut self) {
85 self.func.clear();
86 self.cfg.clear();
87 self.domtree.clear();
88 self.loop_analysis.clear();
89 self.compiled_code = None;
90 self.want_disasm = false;
91 }
92
93 pub fn compiled_code(&self) -> Option<&CompiledCode> {
96 self.compiled_code.as_ref()
97 }
98
99 pub fn take_compiled_code(&mut self) -> Option<CompiledCode> {
102 self.compiled_code.take()
103 }
104
105 pub fn set_disasm(&mut self, val: bool) {
108 self.want_disasm = val;
109 }
110
111 #[deprecated = "use Context::compile"]
113 pub fn compile_and_emit(
114 &mut self,
115 isa: &dyn TargetIsa,
116 mem: &mut Vec<u8>,
117 ctrl_plane: &mut ControlPlane,
118 ) -> CompileResult<&CompiledCode> {
119 let compiled_code = self.compile(isa, ctrl_plane)?;
120 mem.extend_from_slice(compiled_code.code_buffer());
121 Ok(compiled_code)
122 }
123
124 pub fn compile_stencil(
128 &mut self,
129 isa: &dyn TargetIsa,
130 ctrl_plane: &mut ControlPlane,
131 ) -> CodegenResult<CompiledCodeStencil> {
132 let result;
133 trace!("****** START compiling {}", self.func.display_spec());
134 {
135 let _tt = timing::compile();
136
137 self.verify_if(isa)?;
138 self.optimize(isa, ctrl_plane)?;
139 result = isa.compile_function(&self.func, &self.domtree, self.want_disasm, ctrl_plane);
140 }
141 trace!("****** DONE compiling {}\n", self.func.display_spec());
142 result
143 }
144
145 pub fn optimize(
151 &mut self,
152 isa: &dyn TargetIsa,
153 ctrl_plane: &mut ControlPlane,
154 ) -> CodegenResult<()> {
155 log::debug!(
156 "Number of CLIF instructions to optimize: {}",
157 self.func.dfg.num_insts()
158 );
159 log::debug!(
160 "Number of CLIF blocks to optimize: {}",
161 self.func.dfg.num_blocks()
162 );
163
164 let opt_level = isa.flags().opt_level();
165 crate::trace!(
166 "Optimizing (opt level {:?}):\n{}",
167 opt_level,
168 self.func.display()
169 );
170
171 self.compute_cfg();
172 if isa.flags().enable_nan_canonicalization() {
173 self.canonicalize_nans(isa)?;
174 }
175
176 self.legalize(isa)?;
177
178 self.compute_domtree();
179 self.eliminate_unreachable_code(isa)?;
180 self.remove_constant_phis(isa)?;
181
182 self.func.dfg.resolve_all_aliases();
183
184 if opt_level != OptLevel::None {
185 self.egraph_pass(isa, ctrl_plane)?;
186 }
187
188 Ok(())
189 }
190
191 pub fn compile(
204 &mut self,
205 isa: &dyn TargetIsa,
206 ctrl_plane: &mut ControlPlane,
207 ) -> CompileResult<&CompiledCode> {
208 let stencil = self
209 .compile_stencil(isa, ctrl_plane)
210 .map_err(|error| CompileError {
211 inner: error,
212 func: &self.func,
213 })?;
214 Ok(self
215 .compiled_code
216 .insert(stencil.apply_params(&self.func.params)))
217 }
218
219 #[deprecated = "use CompiledCode::get_code_bb_layout"]
223 pub fn get_code_bb_layout(&self) -> Option<(Vec<usize>, Vec<(usize, usize)>)> {
224 self.compiled_code().map(CompiledCode::get_code_bb_layout)
225 }
226
227 #[cfg(feature = "unwind")]
231 #[deprecated = "use CompiledCode::create_unwind_info"]
232 pub fn create_unwind_info(
233 &self,
234 isa: &dyn TargetIsa,
235 ) -> CodegenResult<Option<crate::isa::unwind::UnwindInfo>> {
236 self.compiled_code().unwrap().create_unwind_info(isa)
237 }
238
239 pub fn verify<'a, FOI: Into<FlagsOrIsa<'a>>>(&self, fisa: FOI) -> VerifierResult<()> {
245 let mut errors = VerifierErrors::default();
246 let _ = verify_context(&self.func, &self.cfg, &self.domtree, fisa, &mut errors);
247
248 if errors.is_empty() {
249 Ok(())
250 } else {
251 Err(errors)
252 }
253 }
254
255 pub fn verify_if<'a, FOI: Into<FlagsOrIsa<'a>>>(&self, fisa: FOI) -> CodegenResult<()> {
257 let fisa = fisa.into();
258 if fisa.flags.enable_verifier() {
259 self.verify(fisa)?;
260 }
261 Ok(())
262 }
263
264 pub fn remove_constant_phis<'a, FOI: Into<FlagsOrIsa<'a>>>(
266 &mut self,
267 fisa: FOI,
268 ) -> CodegenResult<()> {
269 do_remove_constant_phis(&mut self.func, &mut self.domtree);
270 self.verify_if(fisa)?;
271 Ok(())
272 }
273
274 pub fn canonicalize_nans(&mut self, isa: &dyn TargetIsa) -> CodegenResult<()> {
276 let has_vector_support = match isa.triple().architecture {
278 Architecture::Riscv64(_) => match isa.isa_flags().iter().find(|f| f.name == "has_v") {
279 Some(value) => value.as_bool().unwrap_or(false),
280 None => false,
281 },
282 _ => true,
283 };
284 do_nan_canonicalization(&mut self.func, has_vector_support);
285 self.verify_if(isa)
286 }
287
288 pub fn legalize(&mut self, isa: &dyn TargetIsa) -> CodegenResult<()> {
290 self.domtree.clear();
293 self.loop_analysis.clear();
294
295 simple_legalize(&mut self.func, isa);
297 self.verify_if(isa)
298 }
299
300 pub fn compute_cfg(&mut self) {
302 self.cfg.compute(&self.func)
303 }
304
305 pub fn compute_domtree(&mut self) {
307 self.domtree.compute(&self.func, &self.cfg)
308 }
309
310 pub fn compute_loop_analysis(&mut self) {
312 self.loop_analysis
313 .compute(&self.func, &self.cfg, &self.domtree)
314 }
315
316 pub fn flowgraph(&mut self) {
318 self.compute_cfg();
319 self.compute_domtree()
320 }
321
322 pub fn eliminate_unreachable_code<'a, FOI>(&mut self, fisa: FOI) -> CodegenResult<()>
324 where
325 FOI: Into<FlagsOrIsa<'a>>,
326 {
327 eliminate_unreachable_code(&mut self.func, &mut self.cfg, &self.domtree);
328 self.verify_if(fisa)
329 }
330
331 pub fn replace_redundant_loads(&mut self) -> CodegenResult<()> {
337 let mut analysis = AliasAnalysis::new(&self.func, &self.domtree);
338 analysis.compute_and_update_aliases(&mut self.func);
339 Ok(())
340 }
341
342 #[cfg(feature = "souper-harvest")]
344 pub fn souper_harvest(
345 &mut self,
346 out: &mut std::sync::mpsc::Sender<String>,
347 ) -> CodegenResult<()> {
348 do_souper_harvest(&self.func, out);
349 Ok(())
350 }
351
352 pub fn egraph_pass<'a, FOI>(
354 &mut self,
355 fisa: FOI,
356 ctrl_plane: &mut ControlPlane,
357 ) -> CodegenResult<()>
358 where
359 FOI: Into<FlagsOrIsa<'a>>,
360 {
361 let _tt = timing::egraph();
362
363 trace!(
364 "About to optimize with egraph phase:\n{}",
365 self.func.display()
366 );
367 let fisa = fisa.into();
368 self.compute_loop_analysis();
369 let mut alias_analysis = AliasAnalysis::new(&self.func, &self.domtree);
370 let mut pass = EgraphPass::new(
371 &mut self.func,
372 &self.domtree,
373 &self.loop_analysis,
374 &mut alias_analysis,
375 &fisa.flags,
376 ctrl_plane,
377 );
378 pass.run();
379 log::debug!("egraph stats: {:?}", pass.stats);
380 trace!("After egraph optimization:\n{}", self.func.display());
381
382 self.verify_if(fisa)
383 }
384}