Skip to content

Commit

Permalink
automata/nfa/backtrack: fix memory usage
Browse files Browse the repository at this point in the history
This fixes a memory usage regression where the backtracker would eagerly
allocate its entire capacity up-front. In this case, it meant a minimum
of 256KB for every regex.

Prior to regex 1.9, 256KB was treated as a *maximum*, and we only
allocated what we needed. We migrate that strategy to regex-automata now
as well. This probably does come with a latency cost (I'll run rebar to
be sure it isn't horrendous), but we definitely can't be eagerly
allocating 256KB for every regex. If the latency ends up being an issue,
we can investigate fixing that in other ways.

Fixes #1027
  • Loading branch information
BurntSushi committed Jul 7, 2023
1 parent c51486d commit 4e89cbf
Showing 1 changed file with 19 additions and 29 deletions.
48 changes: 19 additions & 29 deletions regex-automata/src/nfa/thompson/backtrack.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1376,7 +1376,7 @@ impl BoundedBacktracker {
for slot in slots.iter_mut() {
*slot = None;
}
cache.setup_search(&self.nfa, input)?;
cache.setup_search(&self, input)?;
if input.is_done() {
return Ok(None);
}
Expand Down Expand Up @@ -1750,11 +1750,11 @@ impl Cache {
/// in the BoundedBacktracker.
fn setup_search(
&mut self,
nfa: &NFA,
re: &BoundedBacktracker,
input: &Input<'_>,
) -> Result<(), MatchError> {
self.stack.clear();
self.visited.setup_search(nfa, input)?;
self.visited.setup_search(re, input)?;
Ok(())
}
}
Expand Down Expand Up @@ -1836,23 +1836,9 @@ impl Visited {
true
}

/// Returns the capacity of this visited set in terms of the number of bits
/// it has to track (StateID, offset) pairs.
fn capacity(&self) -> usize {
self.bitset.len() * Visited::BLOCK_SIZE
}

/// Reset this visited set to work with the given bounded backtracker.
fn reset(&mut self, re: &BoundedBacktracker) {
// The capacity given in the config is "bytes of heap memory," but the
// capacity we use here is "number of bits." So convert the capacity in
// bytes to the capacity in bits.
let capacity = 8 * re.get_config().get_visited_capacity();
let blocks = div_ceil(capacity, Visited::BLOCK_SIZE);
self.bitset.resize(blocks, 0);
// N.B. 'stride' is set in 'setup_search', since it isn't known until
// we know the length of the haystack. (That is also when we return an
// error if the haystack is too big.)
fn reset(&mut self, _: &BoundedBacktracker) {
self.bitset.truncate(0);
}

/// Setup this visited set to work for a search using the given NFA
Expand All @@ -1861,7 +1847,7 @@ impl Visited {
/// result in panics or silently incorrect search behavior.
fn setup_search(
&mut self,
nfa: &NFA,
re: &BoundedBacktracker,
input: &Input<'_>,
) -> Result<(), MatchError> {
// Our haystack length is only the length of the span of the entire
Expand All @@ -1872,19 +1858,23 @@ impl Visited {
// search loop includes the position at input.end(). (And it does this
// because matches are delayed by one byte to account for look-around.)
self.stride = haylen + 1;
let capacity = match nfa.states().len().checked_mul(self.stride) {
None => return Err(err()),
Some(capacity) => capacity,
};
if capacity > self.capacity() {
let needed_capacity =
match re.get_nfa().states().len().checked_mul(self.stride) {
None => return Err(err()),
Some(capacity) => capacity,
};
let max_capacity = 8 * re.get_config().get_visited_capacity();
if needed_capacity > max_capacity {
return Err(err());
}
// We only need to zero out our desired capacity, not our total
// capacity in this set.
let blocks = div_ceil(capacity, Visited::BLOCK_SIZE);
for block in self.bitset.iter_mut().take(blocks) {
let needed_blocks = div_ceil(needed_capacity, Visited::BLOCK_SIZE);
self.bitset.truncate(needed_blocks);
for block in self.bitset.iter_mut() {
*block = 0;
}
if needed_blocks > self.bitset.len() {
self.bitset.resize(needed_blocks, 0);
}
Ok(())
}

Expand Down

0 comments on commit 4e89cbf

Please sign in to comment.