orca_wasm/subiterator/
module_subiterator.rs1use crate::ir::id::FunctionID;
4use crate::ir::types::Location;
5use crate::subiterator::function_subiterator::FuncSubIterator;
6
7pub struct ModuleSubIterator {
9 pub(crate) curr_idx: usize,
11 metadata: Vec<(FunctionID, usize)>,
13 pub(crate) func_iterator: FuncSubIterator,
15 skip_funcs: Vec<FunctionID>,
17}
18
19impl ModuleSubIterator {
20 pub fn new(metadata: Vec<(FunctionID, usize)>, skip_funcs: Vec<FunctionID>) -> Self {
22 let curr_idx = 0;
23
24 let (_curr_fid, curr_num_instrs) = metadata[curr_idx];
25 let mut mod_it = ModuleSubIterator {
26 curr_idx,
27 metadata,
28 func_iterator: FuncSubIterator::new(curr_num_instrs),
29 skip_funcs,
30 };
31 mod_it.handle_skips();
32
33 mod_it
34 }
35
36 pub fn get_curr_func(&self) -> (FunctionID, usize) {
37 self.metadata[self.curr_idx]
38 }
39
40 pub fn curr_loc(&self) -> (Location, bool) {
43 let curr_instr = self.func_iterator.curr_instr;
44 (
45 Location::Module {
46 func_idx: self.get_curr_func().0,
47 instr_idx: curr_instr,
48 },
49 self.func_iterator.is_end(curr_instr),
50 )
51 }
52
53 pub(crate) fn reset_from_comp_iterator(&mut self, metadata: Vec<(FunctionID, usize)>) {
55 self.metadata = metadata;
56 self.reset();
57 }
58
59 pub fn reset(&mut self) {
61 self.curr_idx = 0;
62 self.handle_skips();
63 self.func_iterator.reset(self.get_curr_func().1);
64 }
65
66 fn handle_skips(&mut self) {
67 let mut curr_fid = self.get_curr_func().0;
68 while self.skip_funcs.contains(&curr_fid) {
69 self.curr_idx += 1;
70 if self.curr_idx >= self.metadata.len() {
71 break;
72 }
73 curr_fid = self.get_curr_func().0;
74 }
75 }
76
77 fn next_function(&mut self) -> bool {
79 if !self.has_next_function() {
80 return false;
81 }
82 self.curr_idx += 1;
83
84 self.handle_skips();
86 if self.curr_idx < self.metadata.len() {
87 self.func_iterator = FuncSubIterator::new(self.get_curr_func().1);
88 true
89 } else {
90 false
91 }
92 }
93
94 pub fn has_next_function(&self) -> bool {
96 self.curr_idx + 1 < self.metadata.len()
97 }
98
99 pub(crate) fn has_next(&self) -> bool {
101 self.func_iterator.has_next() || self.has_next_function()
102 }
103
104 pub(crate) fn next(&mut self) -> bool {
106 if self.func_iterator.has_next() {
107 self.func_iterator.next()
108 } else {
109 self.next_function()
110 }
111 }
112}