cranelift_codegen/cursor.rs
1//! Cursor library.
2//!
3//! This module defines cursor data types that can be used for inserting instructions.
4
5use crate::ir;
6
7/// The possible positions of a cursor.
8#[derive(Clone, Copy, PartialEq, Eq, Debug)]
9pub enum CursorPosition {
10    /// Cursor is not pointing anywhere. No instructions can be inserted.
11    Nowhere,
12    /// Cursor is pointing at an existing instruction.
13    /// New instructions will be inserted *before* the current instruction.
14    At(ir::Inst),
15    /// Cursor is before the beginning of a block. No instructions can be inserted. Calling
16    /// `next_inst()` will move to the first instruction in the block.
17    Before(ir::Block),
18    /// Cursor is pointing after the end of a block.
19    /// New instructions will be appended to the block.
20    After(ir::Block),
21}
22
23/// All cursor types implement the `Cursor` which provides common navigation operations.
24pub trait Cursor {
25    /// Get the current cursor position.
26    fn position(&self) -> CursorPosition;
27
28    /// Set the current position.
29    fn set_position(&mut self, pos: CursorPosition);
30
31    /// Get the source location that should be assigned to new instructions.
32    fn srcloc(&self) -> ir::SourceLoc;
33
34    /// Set the source location that should be assigned to new instructions.
35    fn set_srcloc(&mut self, srcloc: ir::SourceLoc);
36
37    /// Borrow a reference to the function layout that this cursor is navigating.
38    fn layout(&self) -> &ir::Layout;
39
40    /// Borrow a mutable reference to the function layout that this cursor is navigating.
41    fn layout_mut(&mut self) -> &mut ir::Layout;
42
43    /// Exchange this cursor for one with a set source location.
44    ///
45    /// This is intended to be used as a builder method:
46    ///
47    /// ```
48    /// # use cranelift_codegen::ir::{Function, Block, SourceLoc};
49    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
50    /// fn edit_func(func: &mut Function, srcloc: SourceLoc) {
51    ///     let mut pos = FuncCursor::new(func).with_srcloc(srcloc);
52    ///
53    ///     // Use `pos`...
54    /// }
55    /// ```
56    fn with_srcloc(mut self, srcloc: ir::SourceLoc) -> Self
57    where
58        Self: Sized,
59    {
60        self.set_srcloc(srcloc);
61        self
62    }
63
64    /// Rebuild this cursor positioned at `pos`.
65    fn at_position(mut self, pos: CursorPosition) -> Self
66    where
67        Self: Sized,
68    {
69        self.set_position(pos);
70        self
71    }
72
73    /// Rebuild this cursor positioned at `inst`.
74    ///
75    /// This is intended to be used as a builder method:
76    ///
77    /// ```
78    /// # use cranelift_codegen::ir::{Function, Block, Inst};
79    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
80    /// fn edit_func(func: &mut Function, inst: Inst) {
81    ///     let mut pos = FuncCursor::new(func).at_inst(inst);
82    ///
83    ///     // Use `pos`...
84    /// }
85    /// ```
86    fn at_inst(mut self, inst: ir::Inst) -> Self
87    where
88        Self: Sized,
89    {
90        self.goto_inst(inst);
91        self
92    }
93
94    /// Rebuild this cursor positioned at the first insertion point for `block`.
95    /// This differs from `at_first_inst` in that it doesn't assume that any
96    /// instructions have been inserted into `block` yet.
97    ///
98    /// This is intended to be used as a builder method:
99    ///
100    /// ```
101    /// # use cranelift_codegen::ir::{Function, Block, Inst};
102    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
103    /// fn edit_func(func: &mut Function, block: Block) {
104    ///     let mut pos = FuncCursor::new(func).at_first_insertion_point(block);
105    ///
106    ///     // Use `pos`...
107    /// }
108    /// ```
109    fn at_first_insertion_point(mut self, block: ir::Block) -> Self
110    where
111        Self: Sized,
112    {
113        self.goto_first_insertion_point(block);
114        self
115    }
116
117    /// Rebuild this cursor positioned at the first instruction in `block`.
118    ///
119    /// This is intended to be used as a builder method:
120    ///
121    /// ```
122    /// # use cranelift_codegen::ir::{Function, Block, Inst};
123    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
124    /// fn edit_func(func: &mut Function, block: Block) {
125    ///     let mut pos = FuncCursor::new(func).at_first_inst(block);
126    ///
127    ///     // Use `pos`...
128    /// }
129    /// ```
130    fn at_first_inst(mut self, block: ir::Block) -> Self
131    where
132        Self: Sized,
133    {
134        self.goto_first_inst(block);
135        self
136    }
137
138    /// Rebuild this cursor positioned at the last instruction in `block`.
139    ///
140    /// This is intended to be used as a builder method:
141    ///
142    /// ```
143    /// # use cranelift_codegen::ir::{Function, Block, Inst};
144    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
145    /// fn edit_func(func: &mut Function, block: Block) {
146    ///     let mut pos = FuncCursor::new(func).at_last_inst(block);
147    ///
148    ///     // Use `pos`...
149    /// }
150    /// ```
151    fn at_last_inst(mut self, block: ir::Block) -> Self
152    where
153        Self: Sized,
154    {
155        self.goto_last_inst(block);
156        self
157    }
158
159    /// Rebuild this cursor positioned after `inst`.
160    ///
161    /// This is intended to be used as a builder method:
162    ///
163    /// ```
164    /// # use cranelift_codegen::ir::{Function, Block, Inst};
165    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
166    /// fn edit_func(func: &mut Function, inst: Inst) {
167    ///     let mut pos = FuncCursor::new(func).after_inst(inst);
168    ///
169    ///     // Use `pos`...
170    /// }
171    /// ```
172    fn after_inst(mut self, inst: ir::Inst) -> Self
173    where
174        Self: Sized,
175    {
176        self.goto_after_inst(inst);
177        self
178    }
179
180    /// Rebuild this cursor positioned at the top of `block`.
181    ///
182    /// This is intended to be used as a builder method:
183    ///
184    /// ```
185    /// # use cranelift_codegen::ir::{Function, Block, Inst};
186    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
187    /// fn edit_func(func: &mut Function, block: Block) {
188    ///     let mut pos = FuncCursor::new(func).at_top(block);
189    ///
190    ///     // Use `pos`...
191    /// }
192    /// ```
193    fn at_top(mut self, block: ir::Block) -> Self
194    where
195        Self: Sized,
196    {
197        self.goto_top(block);
198        self
199    }
200
201    /// Rebuild this cursor positioned at the bottom of `block`.
202    ///
203    /// This is intended to be used as a builder method:
204    ///
205    /// ```
206    /// # use cranelift_codegen::ir::{Function, Block, Inst};
207    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
208    /// fn edit_func(func: &mut Function, block: Block) {
209    ///     let mut pos = FuncCursor::new(func).at_bottom(block);
210    ///
211    ///     // Use `pos`...
212    /// }
213    /// ```
214    fn at_bottom(mut self, block: ir::Block) -> Self
215    where
216        Self: Sized,
217    {
218        self.goto_bottom(block);
219        self
220    }
221
222    /// Get the block corresponding to the current position.
223    fn current_block(&self) -> Option<ir::Block> {
224        use self::CursorPosition::*;
225        match self.position() {
226            Nowhere => None,
227            At(inst) => self.layout().inst_block(inst),
228            Before(block) | After(block) => Some(block),
229        }
230    }
231
232    /// Get the instruction corresponding to the current position, if any.
233    fn current_inst(&self) -> Option<ir::Inst> {
234        use self::CursorPosition::*;
235        match self.position() {
236            At(inst) => Some(inst),
237            _ => None,
238        }
239    }
240
241    /// Go to the position after a specific instruction, which must be inserted
242    /// in the layout. New instructions will be inserted after `inst`.
243    fn goto_after_inst(&mut self, inst: ir::Inst) {
244        debug_assert!(self.layout().inst_block(inst).is_some());
245        let new_pos = if let Some(next) = self.layout().next_inst(inst) {
246            CursorPosition::At(next)
247        } else {
248            CursorPosition::After(
249                self.layout()
250                    .inst_block(inst)
251                    .expect("current instruction removed?"),
252            )
253        };
254        self.set_position(new_pos);
255    }
256
257    /// Go to a specific instruction which must be inserted in the layout.
258    /// New instructions will be inserted before `inst`.
259    fn goto_inst(&mut self, inst: ir::Inst) {
260        debug_assert!(self.layout().inst_block(inst).is_some());
261        self.set_position(CursorPosition::At(inst));
262    }
263
264    /// Go to the position for inserting instructions at the beginning of `block`,
265    /// which unlike `goto_first_inst` doesn't assume that any instructions have
266    /// been inserted into `block` yet.
267    fn goto_first_insertion_point(&mut self, block: ir::Block) {
268        if let Some(inst) = self.layout().first_inst(block) {
269            self.goto_inst(inst);
270        } else {
271            self.goto_bottom(block);
272        }
273    }
274
275    /// Go to the first instruction in `block`.
276    fn goto_first_inst(&mut self, block: ir::Block) {
277        let inst = self.layout().first_inst(block).expect("Empty block");
278        self.goto_inst(inst);
279    }
280
281    /// Go to the last instruction in `block`.
282    fn goto_last_inst(&mut self, block: ir::Block) {
283        let inst = self.layout().last_inst(block).expect("Empty block");
284        self.goto_inst(inst);
285    }
286
287    /// Go to the top of `block` which must be inserted into the layout.
288    /// At this position, instructions cannot be inserted, but `next_inst()` will move to the first
289    /// instruction in `block`.
290    fn goto_top(&mut self, block: ir::Block) {
291        debug_assert!(self.layout().is_block_inserted(block));
292        self.set_position(CursorPosition::Before(block));
293    }
294
295    /// Go to the bottom of `block` which must be inserted into the layout.
296    /// At this position, inserted instructions will be appended to `block`.
297    fn goto_bottom(&mut self, block: ir::Block) {
298        debug_assert!(self.layout().is_block_inserted(block));
299        self.set_position(CursorPosition::After(block));
300    }
301
302    /// Go to the top of the next block in layout order and return it.
303    ///
304    /// - If the cursor wasn't pointing at anything, go to the top of the first block in the
305    ///   function.
306    /// - If there are no more blocks, leave the cursor pointing at nothing and return `None`.
307    ///
308    /// # Examples
309    ///
310    /// The `next_block()` method is intended for iterating over the blocks in layout order:
311    ///
312    /// ```
313    /// # use cranelift_codegen::ir::{Function, Block};
314    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
315    /// fn edit_func(func: &mut Function) {
316    ///     let mut cursor = FuncCursor::new(func);
317    ///     while let Some(block) = cursor.next_block() {
318    ///         // Edit block.
319    ///     }
320    /// }
321    /// ```
322    fn next_block(&mut self) -> Option<ir::Block> {
323        let next = if let Some(block) = self.current_block() {
324            self.layout().next_block(block)
325        } else {
326            self.layout().entry_block()
327        };
328        self.set_position(match next {
329            Some(block) => CursorPosition::Before(block),
330            None => CursorPosition::Nowhere,
331        });
332        next
333    }
334
335    /// Go to the bottom of the previous block in layout order and return it.
336    ///
337    /// - If the cursor wasn't pointing at anything, go to the bottom of the last block in the
338    ///   function.
339    /// - If there are no more blocks, leave the cursor pointing at nothing and return `None`.
340    ///
341    /// # Examples
342    ///
343    /// The `prev_block()` method is intended for iterating over the blocks in backwards layout order:
344    ///
345    /// ```
346    /// # use cranelift_codegen::ir::{Function, Block};
347    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
348    /// fn edit_func(func: &mut Function) {
349    ///     let mut cursor = FuncCursor::new(func);
350    ///     while let Some(block) = cursor.prev_block() {
351    ///         // Edit block.
352    ///     }
353    /// }
354    /// ```
355    fn prev_block(&mut self) -> Option<ir::Block> {
356        let prev = if let Some(block) = self.current_block() {
357            self.layout().prev_block(block)
358        } else {
359            self.layout().last_block()
360        };
361        self.set_position(match prev {
362            Some(block) => CursorPosition::After(block),
363            None => CursorPosition::Nowhere,
364        });
365        prev
366    }
367
368    /// Move to the next instruction in the same block and return it.
369    ///
370    /// - If the cursor was positioned before a block, go to the first instruction in that block.
371    /// - If there are no more instructions in the block, go to the `After(block)` position and return
372    ///   `None`.
373    /// - If the cursor wasn't pointing anywhere, keep doing that.
374    ///
375    /// This method will never move the cursor to a different block.
376    ///
377    /// # Examples
378    ///
379    /// The `next_inst()` method is intended for iterating over the instructions in a block like
380    /// this:
381    ///
382    /// ```
383    /// # use cranelift_codegen::ir::{Function, Block};
384    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
385    /// fn edit_block(func: &mut Function, block: Block) {
386    ///     let mut cursor = FuncCursor::new(func).at_top(block);
387    ///     while let Some(inst) = cursor.next_inst() {
388    ///         // Edit instructions...
389    ///     }
390    /// }
391    /// ```
392    /// The loop body can insert and remove instructions via the cursor.
393    ///
394    /// Iterating over all the instructions in a function looks like this:
395    ///
396    /// ```
397    /// # use cranelift_codegen::ir::{Function, Block};
398    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
399    /// fn edit_func(func: &mut Function) {
400    ///     let mut cursor = FuncCursor::new(func);
401    ///     while let Some(block) = cursor.next_block() {
402    ///         while let Some(inst) = cursor.next_inst() {
403    ///             // Edit instructions...
404    ///         }
405    ///     }
406    /// }
407    /// ```
408    fn next_inst(&mut self) -> Option<ir::Inst> {
409        use self::CursorPosition::*;
410        match self.position() {
411            Nowhere | After(..) => None,
412            At(inst) => {
413                if let Some(next) = self.layout().next_inst(inst) {
414                    self.set_position(At(next));
415                    Some(next)
416                } else {
417                    let pos = After(
418                        self.layout()
419                            .inst_block(inst)
420                            .expect("current instruction removed?"),
421                    );
422                    self.set_position(pos);
423                    None
424                }
425            }
426            Before(block) => {
427                if let Some(next) = self.layout().first_inst(block) {
428                    self.set_position(At(next));
429                    Some(next)
430                } else {
431                    self.set_position(After(block));
432                    None
433                }
434            }
435        }
436    }
437
438    /// Move to the previous instruction in the same block and return it.
439    ///
440    /// - If the cursor was positioned after a block, go to the last instruction in that block.
441    /// - If there are no more instructions in the block, go to the `Before(block)` position and return
442    ///   `None`.
443    /// - If the cursor wasn't pointing anywhere, keep doing that.
444    ///
445    /// This method will never move the cursor to a different block.
446    ///
447    /// # Examples
448    ///
449    /// The `prev_inst()` method is intended for iterating backwards over the instructions in an
450    /// block like this:
451    ///
452    /// ```
453    /// # use cranelift_codegen::ir::{Function, Block};
454    /// # use cranelift_codegen::cursor::{Cursor, FuncCursor};
455    /// fn edit_block(func: &mut Function, block: Block) {
456    ///     let mut cursor = FuncCursor::new(func).at_bottom(block);
457    ///     while let Some(inst) = cursor.prev_inst() {
458    ///         // Edit instructions...
459    ///     }
460    /// }
461    /// ```
462    fn prev_inst(&mut self) -> Option<ir::Inst> {
463        use self::CursorPosition::*;
464        match self.position() {
465            Nowhere | Before(..) => None,
466            At(inst) => {
467                if let Some(prev) = self.layout().prev_inst(inst) {
468                    self.set_position(At(prev));
469                    Some(prev)
470                } else {
471                    let pos = Before(
472                        self.layout()
473                            .inst_block(inst)
474                            .expect("current instruction removed?"),
475                    );
476                    self.set_position(pos);
477                    None
478                }
479            }
480            After(block) => {
481                if let Some(prev) = self.layout().last_inst(block) {
482                    self.set_position(At(prev));
483                    Some(prev)
484                } else {
485                    self.set_position(Before(block));
486                    None
487                }
488            }
489        }
490    }
491
492    /// Insert an instruction at the current position.
493    ///
494    /// - If pointing at an instruction, the new instruction is inserted before the current
495    ///   instruction.
496    /// - If pointing at the bottom of a block, the new instruction is appended to the block.
497    /// - Otherwise panic.
498    ///
499    /// In either case, the cursor is not moved, such that repeated calls to `insert_inst()` causes
500    /// instructions to appear in insertion order in the block.
501    fn insert_inst(&mut self, inst: ir::Inst) {
502        use self::CursorPosition::*;
503        match self.position() {
504            Nowhere | Before(..) => panic!("Invalid insert_inst position"),
505            At(cur) => self.layout_mut().insert_inst(inst, cur),
506            After(block) => self.layout_mut().append_inst(inst, block),
507        }
508    }
509
510    /// Remove the instruction under the cursor.
511    ///
512    /// The cursor is left pointing at the position following the current instruction.
513    ///
514    /// Return the instruction that was removed.
515    fn remove_inst(&mut self) -> ir::Inst {
516        let inst = self.current_inst().expect("No instruction to remove");
517        self.next_inst();
518        self.layout_mut().remove_inst(inst);
519        inst
520    }
521
522    /// Remove the instruction under the cursor.
523    ///
524    /// The cursor is left pointing at the position preceding the current instruction.
525    ///
526    /// Return the instruction that was removed.
527    fn remove_inst_and_step_back(&mut self) -> ir::Inst {
528        let inst = self.current_inst().expect("No instruction to remove");
529        self.prev_inst();
530        self.layout_mut().remove_inst(inst);
531        inst
532    }
533
534    /// Insert a block at the current position and switch to it.
535    ///
536    /// As far as possible, this method behaves as if the block header were an instruction inserted
537    /// at the current position.
538    ///
539    /// - If the cursor is pointing at an existing instruction, *the current block is split in two*
540    ///   and the current instruction becomes the first instruction in the inserted block.
541    /// - If the cursor points at the bottom of a block, the new block is inserted after the current
542    ///   one, and moved to the bottom of the new block where instructions can be appended.
543    /// - If the cursor points to the top of a block, the new block is inserted above the current one.
544    /// - If the cursor is not pointing at anything, the new block is placed last in the layout.
545    ///
546    /// This means that it is always valid to call this method, and it always leaves the cursor in
547    /// a state that will insert instructions into the new block.
548    fn insert_block(&mut self, new_block: ir::Block) {
549        use self::CursorPosition::*;
550        match self.position() {
551            At(inst) => {
552                self.layout_mut().split_block(new_block, inst);
553                // All other cases move to `After(block)`, but in this case we'll stay `At(inst)`.
554                return;
555            }
556            Nowhere => self.layout_mut().append_block(new_block),
557            Before(block) => self.layout_mut().insert_block(new_block, block),
558            After(block) => self.layout_mut().insert_block_after(new_block, block),
559        }
560        // For everything but `At(inst)` we end up appending to the new block.
561        self.set_position(After(new_block));
562    }
563}
564
565/// Function cursor.
566///
567/// A `FuncCursor` holds a mutable reference to a whole `ir::Function` while keeping a position
568/// too. The function can be re-borrowed by accessing the public `cur.func` member.
569///
570/// This cursor is for use before legalization. The inserted instructions are not given an
571/// encoding.
572pub struct FuncCursor<'f> {
573    pos: CursorPosition,
574    srcloc: ir::SourceLoc,
575
576    /// The referenced function.
577    pub func: &'f mut ir::Function,
578}
579
580impl<'f> FuncCursor<'f> {
581    /// Create a new `FuncCursor` pointing nowhere.
582    pub fn new(func: &'f mut ir::Function) -> Self {
583        Self {
584            pos: CursorPosition::Nowhere,
585            srcloc: Default::default(),
586            func,
587        }
588    }
589
590    /// Use the source location of `inst` for future instructions.
591    pub fn use_srcloc(&mut self, inst: ir::Inst) {
592        self.srcloc = self.func.srcloc(inst);
593    }
594
595    /// Create an instruction builder that inserts an instruction at the current position.
596    pub fn ins(&mut self) -> ir::InsertBuilder<'_, &mut FuncCursor<'f>> {
597        ir::InsertBuilder::new(self)
598    }
599}
600
601impl<'f> Cursor for FuncCursor<'f> {
602    fn position(&self) -> CursorPosition {
603        self.pos
604    }
605
606    fn set_position(&mut self, pos: CursorPosition) {
607        self.pos = pos
608    }
609
610    fn srcloc(&self) -> ir::SourceLoc {
611        self.srcloc
612    }
613
614    fn set_srcloc(&mut self, srcloc: ir::SourceLoc) {
615        self.func.params.ensure_base_srcloc(srcloc);
616        self.srcloc = srcloc;
617    }
618
619    fn layout(&self) -> &ir::Layout {
620        &self.func.layout
621    }
622
623    fn layout_mut(&mut self) -> &mut ir::Layout {
624        &mut self.func.layout
625    }
626}
627
628impl<'c, 'f> ir::InstInserterBase<'c> for &'c mut FuncCursor<'f> {
629    fn data_flow_graph(&self) -> &ir::DataFlowGraph {
630        &self.func.dfg
631    }
632
633    fn data_flow_graph_mut(&mut self) -> &mut ir::DataFlowGraph {
634        &mut self.func.dfg
635    }
636
637    fn insert_built_inst(self, inst: ir::Inst) -> &'c mut ir::DataFlowGraph {
638        self.insert_inst(inst);
639        if !self.srcloc.is_default() {
640            self.func.set_srcloc(inst, self.srcloc);
641        }
642        &mut self.func.dfg
643    }
644}