/build/cargo-vendor-dir/regex-syntax-0.8.2/src/hir/visitor.rs
Line | Count | Source (jump to first uncovered line) |
1 | | use alloc::{vec, vec::Vec}; |
2 | | |
3 | | use crate::hir::{self, Hir, HirKind}; |
4 | | |
5 | | /// A trait for visiting the high-level IR (HIR) in depth first order. |
6 | | /// |
7 | | /// The principle aim of this trait is to enable callers to perform case |
8 | | /// analysis on a high-level intermediate representation of a regular |
9 | | /// expression without necessarily using recursion. In particular, this permits |
10 | | /// callers to do case analysis with constant stack usage, which can be |
11 | | /// important since the size of an HIR may be proportional to end user input. |
12 | | /// |
13 | | /// Typical usage of this trait involves providing an implementation and then |
14 | | /// running it using the [`visit`] function. |
15 | | pub trait Visitor { |
16 | | /// The result of visiting an HIR. |
17 | | type Output; |
18 | | /// An error that visiting an HIR might return. |
19 | | type Err; |
20 | | |
21 | | /// All implementors of `Visitor` must provide a `finish` method, which |
22 | | /// yields the result of visiting the HIR or an error. |
23 | | fn finish(self) -> Result<Self::Output, Self::Err>; |
24 | | |
25 | | /// This method is called before beginning traversal of the HIR. |
26 | 0 | fn start(&mut self) {} |
27 | | |
28 | | /// This method is called on an `Hir` before descending into child `Hir` |
29 | | /// nodes. |
30 | 0 | fn visit_pre(&mut self, _hir: &Hir) -> Result<(), Self::Err> { |
31 | 0 | Ok(()) |
32 | 0 | } |
33 | | |
34 | | /// This method is called on an `Hir` after descending all of its child |
35 | | /// `Hir` nodes. |
36 | 0 | fn visit_post(&mut self, _hir: &Hir) -> Result<(), Self::Err> { |
37 | 0 | Ok(()) |
38 | 0 | } |
39 | | |
40 | | /// This method is called between child nodes of an alternation. |
41 | 0 | fn visit_alternation_in(&mut self) -> Result<(), Self::Err> { |
42 | 0 | Ok(()) |
43 | 0 | } |
44 | | |
45 | | /// This method is called between child nodes of a concatenation. |
46 | 0 | fn visit_concat_in(&mut self) -> Result<(), Self::Err> { |
47 | 0 | Ok(()) |
48 | 0 | } |
49 | | } |
50 | | |
51 | | /// Executes an implementation of `Visitor` in constant stack space. |
52 | | /// |
53 | | /// This function will visit every node in the given `Hir` while calling |
54 | | /// appropriate methods provided by the [`Visitor`] trait. |
55 | | /// |
56 | | /// The primary use case for this method is when one wants to perform case |
57 | | /// analysis over an `Hir` without using a stack size proportional to the depth |
58 | | /// of the `Hir`. Namely, this method will instead use constant stack space, |
59 | | /// but will use heap space proportional to the size of the `Hir`. This may be |
60 | | /// desirable in cases where the size of `Hir` is proportional to end user |
61 | | /// input. |
62 | | /// |
63 | | /// If the visitor returns an error at any point, then visiting is stopped and |
64 | | /// the error is returned. |
65 | 0 | pub fn visit<V: Visitor>(hir: &Hir, visitor: V) -> Result<V::Output, V::Err> { |
66 | 0 | HeapVisitor::new().visit(hir, visitor) |
67 | 0 | } |
68 | | |
69 | | /// HeapVisitor visits every item in an `Hir` recursively using constant stack |
70 | | /// size and a heap size proportional to the size of the `Hir`. |
71 | | struct HeapVisitor<'a> { |
72 | | /// A stack of `Hir` nodes. This is roughly analogous to the call stack |
73 | | /// used in a typical recursive visitor. |
74 | | stack: Vec<(&'a Hir, Frame<'a>)>, |
75 | | } |
76 | | |
77 | | /// Represents a single stack frame while performing structural induction over |
78 | | /// an `Hir`. |
79 | | enum Frame<'a> { |
80 | | /// A stack frame allocated just before descending into a repetition |
81 | | /// operator's child node. |
82 | | Repetition(&'a hir::Repetition), |
83 | | /// A stack frame allocated just before descending into a capture's child |
84 | | /// node. |
85 | | Capture(&'a hir::Capture), |
86 | | /// The stack frame used while visiting every child node of a concatenation |
87 | | /// of expressions. |
88 | | Concat { |
89 | | /// The child node we are currently visiting. |
90 | | head: &'a Hir, |
91 | | /// The remaining child nodes to visit (which may be empty). |
92 | | tail: &'a [Hir], |
93 | | }, |
94 | | /// The stack frame used while visiting every child node of an alternation |
95 | | /// of expressions. |
96 | | Alternation { |
97 | | /// The child node we are currently visiting. |
98 | | head: &'a Hir, |
99 | | /// The remaining child nodes to visit (which may be empty). |
100 | | tail: &'a [Hir], |
101 | | }, |
102 | | } |
103 | | |
104 | | impl<'a> HeapVisitor<'a> { |
105 | 0 | fn new() -> HeapVisitor<'a> { |
106 | 0 | HeapVisitor { stack: vec![] } |
107 | 0 | } |
108 | | |
109 | 0 | fn visit<V: Visitor>( |
110 | 0 | &mut self, |
111 | 0 | mut hir: &'a Hir, |
112 | 0 | mut visitor: V, |
113 | 0 | ) -> Result<V::Output, V::Err> { |
114 | 0 | self.stack.clear(); |
115 | 0 |
|
116 | 0 | visitor.start(); |
117 | | loop { |
118 | 0 | visitor.visit_pre(hir)?; |
119 | 0 | if let Some(x) = self.induct(hir) { |
120 | 0 | let child = x.child(); |
121 | 0 | self.stack.push((hir, x)); |
122 | 0 | hir = child; |
123 | 0 | continue; |
124 | 0 | } |
125 | 0 | // No induction means we have a base case, so we can post visit |
126 | 0 | // it now. |
127 | 0 | visitor.visit_post(hir)?; |
128 | | |
129 | | // At this point, we now try to pop our call stack until it is |
130 | | // either empty or we hit another inductive case. |
131 | | loop { |
132 | 0 | let (post_hir, frame) = match self.stack.pop() { |
133 | 0 | None => return visitor.finish(), |
134 | 0 | Some((post_hir, frame)) => (post_hir, frame), |
135 | | }; |
136 | | // If this is a concat/alternate, then we might have additional |
137 | | // inductive steps to process. |
138 | 0 | if let Some(x) = self.pop(frame) { |
139 | 0 | match x { |
140 | | Frame::Alternation { .. } => { |
141 | 0 | visitor.visit_alternation_in()?; |
142 | | } |
143 | | Frame::Concat { .. } => { |
144 | 0 | visitor.visit_concat_in()?; |
145 | | } |
146 | 0 | _ => {} |
147 | | } |
148 | 0 | hir = x.child(); |
149 | 0 | self.stack.push((post_hir, x)); |
150 | 0 | break; |
151 | 0 | } |
152 | 0 | // Otherwise, we've finished visiting all the child nodes for |
153 | 0 | // this HIR, so we can post visit it now. |
154 | 0 | visitor.visit_post(post_hir)?; |
155 | | } |
156 | | } |
157 | 0 | } |
158 | | |
159 | | /// Build a stack frame for the given HIR if one is needed (which occurs if |
160 | | /// and only if there are child nodes in the HIR). Otherwise, return None. |
161 | 0 | fn induct(&mut self, hir: &'a Hir) -> Option<Frame<'a>> { |
162 | 0 | match *hir.kind() { |
163 | 0 | HirKind::Repetition(ref x) => Some(Frame::Repetition(x)), |
164 | 0 | HirKind::Capture(ref x) => Some(Frame::Capture(x)), |
165 | 0 | HirKind::Concat(ref x) if x.is_empty() => None, |
166 | 0 | HirKind::Concat(ref x) => { |
167 | 0 | Some(Frame::Concat { head: &x[0], tail: &x[1..] }) |
168 | | } |
169 | 0 | HirKind::Alternation(ref x) if x.is_empty() => None, |
170 | 0 | HirKind::Alternation(ref x) => { |
171 | 0 | Some(Frame::Alternation { head: &x[0], tail: &x[1..] }) |
172 | | } |
173 | 0 | _ => None, |
174 | | } |
175 | 0 | } |
176 | | |
177 | | /// Pops the given frame. If the frame has an additional inductive step, |
178 | | /// then return it, otherwise return `None`. |
179 | 0 | fn pop(&self, induct: Frame<'a>) -> Option<Frame<'a>> { |
180 | 0 | match induct { |
181 | 0 | Frame::Repetition(_) => None, |
182 | 0 | Frame::Capture(_) => None, |
183 | 0 | Frame::Concat { tail, .. } => { |
184 | 0 | if tail.is_empty() { |
185 | 0 | None |
186 | | } else { |
187 | 0 | Some(Frame::Concat { head: &tail[0], tail: &tail[1..] }) |
188 | | } |
189 | | } |
190 | 0 | Frame::Alternation { tail, .. } => { |
191 | 0 | if tail.is_empty() { |
192 | 0 | None |
193 | | } else { |
194 | 0 | Some(Frame::Alternation { |
195 | 0 | head: &tail[0], |
196 | 0 | tail: &tail[1..], |
197 | 0 | }) |
198 | | } |
199 | | } |
200 | | } |
201 | 0 | } |
202 | | } |
203 | | |
204 | | impl<'a> Frame<'a> { |
205 | | /// Perform the next inductive step on this frame and return the next |
206 | | /// child HIR node to visit. |
207 | 0 | fn child(&self) -> &'a Hir { |
208 | 0 | match *self { |
209 | 0 | Frame::Repetition(rep) => &rep.sub, |
210 | 0 | Frame::Capture(capture) => &capture.sub, |
211 | 0 | Frame::Concat { head, .. } => head, |
212 | 0 | Frame::Alternation { head, .. } => head, |
213 | | } |
214 | 0 | } |
215 | | } |