buffer_pool/
buffer.rs

1// Copyright (C) 2025, Cloudflare, Inc.
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are
6// met:
7//
8//     * Redistributions of source code must retain the above copyright notice,
9//       this list of conditions and the following disclaimer.
10//
11//     * Redistributions in binary form must reproduce the above copyright
12//       notice, this list of conditions and the following disclaimer in the
13//       documentation and/or other materials provided with the distribution.
14//
15// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
16// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
17// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
19// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
23// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
24// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
27use std::ops::Deref;
28use std::ops::DerefMut;
29
30use crate::Reuse;
31
32/// A convinience wrapper around Vec that allows to "consume" data from the
33/// front *without* shifting.
34///
35/// This is not unlike `VecDeque` but more ergonomic
36/// for the operations we require. Conceptually `VecDeque` is two slices, and
37/// this is one slice. Also there is no `set_len` for `VecDeque`, so it has to
38/// be converted to `Vec` and then back again.
39#[derive(Default, Debug)]
40pub struct ConsumeBuffer {
41    inner: Vec<u8>,
42    head: usize,
43}
44
45impl Deref for ConsumeBuffer {
46    type Target = [u8];
47
48    fn deref(&self) -> &Self::Target {
49        &self.inner[self.head..]
50    }
51}
52
53impl DerefMut for ConsumeBuffer {
54    fn deref_mut(&mut self) -> &mut Self::Target {
55        &mut self.inner[self.head..]
56    }
57}
58
59impl Reuse for ConsumeBuffer {
60    fn reuse(&mut self, val: usize) -> bool {
61        self.inner.clear();
62        self.inner.shrink_to(val);
63        self.head = 0;
64        self.inner.capacity() > 0
65    }
66}
67
68impl ConsumeBuffer {
69    pub fn from_vec(inner: Vec<u8>) -> Self {
70        ConsumeBuffer { inner, head: 0 }
71    }
72
73    pub fn into_vec(self) -> Vec<u8> {
74        let mut inner = self.inner;
75        inner.drain(0..self.head);
76        inner
77    }
78
79    pub fn pop_front(&mut self, count: usize) {
80        assert!(self.head + count <= self.inner.len());
81        self.head += count;
82    }
83
84    pub fn expand(&mut self, count: usize) {
85        self.inner.reserve_exact(count);
86        // SAFETY: u8 is always initialized and we reserved the capacity.
87        unsafe { self.inner.set_len(count) };
88    }
89
90    pub fn truncate(&mut self, count: usize) {
91        self.inner.truncate(self.head + count);
92    }
93
94    pub fn add_prefix(&mut self, prefix: &[u8]) -> bool {
95        if self.head < prefix.len() {
96            return false;
97        }
98
99        self.head -= prefix.len();
100        self.inner[self.head..self.head + prefix.len()].copy_from_slice(prefix);
101
102        true
103    }
104}
105
106impl<'a> Extend<&'a u8> for ConsumeBuffer {
107    fn extend<T: IntoIterator<Item = &'a u8>>(&mut self, iter: T) {
108        self.inner.extend(iter)
109    }
110}