97 lines
2.0 KiB
Rust
97 lines
2.0 KiB
Rust
use std::cell::RefCell;
|
|
use std::mem::{replace, swap};
|
|
use std::ops::{Deref, DerefMut, Drop};
|
|
use std::rc::Rc;
|
|
|
|
type BufferStack = Rc<RefCell<Vec<Vec<u8>>>>;
|
|
|
|
pub struct Buffer {
|
|
buffer: Vec<u8>,
|
|
stack: BufferStack,
|
|
}
|
|
|
|
impl Buffer {
|
|
pub fn resize(&mut self, new_len: usize) {
|
|
self.buffer.resize(new_len, 0u8);
|
|
}
|
|
|
|
pub fn move_out(&mut self) -> Buffer {
|
|
Buffer {
|
|
buffer: replace(&mut self.buffer, vec![]),
|
|
stack: Rc::clone(&self.stack),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl AsRef<[u8]> for Buffer {
|
|
fn as_ref(&self) -> &[u8] {
|
|
&self.buffer
|
|
}
|
|
}
|
|
|
|
impl AsMut<[u8]> for Buffer {
|
|
fn as_mut(&mut self) -> &mut [u8] {
|
|
&mut self.buffer
|
|
}
|
|
}
|
|
|
|
impl Deref for Buffer {
|
|
type Target = [u8];
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.buffer
|
|
}
|
|
}
|
|
|
|
impl DerefMut for Buffer {
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
&mut self.buffer
|
|
}
|
|
}
|
|
|
|
impl Drop for Buffer {
|
|
fn drop(&mut self) {
|
|
if self.buffer.capacity() > 0 {
|
|
let mut tmp = vec![];
|
|
swap(&mut tmp, &mut self.buffer);
|
|
{
|
|
let mut stack = self.stack.borrow_mut();
|
|
stack.push(tmp);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct BufferPool {
|
|
buffers_size: usize,
|
|
stack: BufferStack,
|
|
}
|
|
|
|
impl BufferPool {
|
|
pub fn new(buffers_size: usize) -> BufferPool {
|
|
BufferPool {
|
|
buffers_size,
|
|
stack: Rc::new(RefCell::new(vec![])),
|
|
}
|
|
}
|
|
|
|
pub fn get_buffer(&mut self) -> Buffer {
|
|
let buffer = match self.stack.borrow_mut().pop() {
|
|
None => vec![0u8; self.buffers_size],
|
|
Some(mut v) => {
|
|
// memsetting the buffer with `resize` would be a waste of time
|
|
let capacity = v.capacity();
|
|
unsafe {
|
|
v.set_len(capacity);
|
|
}
|
|
v
|
|
}
|
|
};
|
|
|
|
Buffer {
|
|
buffer,
|
|
stack: Rc::clone(&self.stack),
|
|
}
|
|
}
|
|
}
|