From 6709090fab5de7668ab0e8408b9c5ecbb75a694b Mon Sep 17 00:00:00 2001 From: Nikolay Kim Date: Sat, 20 Feb 2021 11:59:49 +0600 Subject: [PATCH] Do not leak request/response pools --- ntex/CHANGES.md | 2 ++ ntex/src/http/message.rs | 33 ++++++++++++++++++--------------- 2 files changed, 20 insertions(+), 15 deletions(-) diff --git a/ntex/CHANGES.md b/ntex/CHANGES.md index 6870043f..493d6da3 100644 --- a/ntex/CHANGES.md +++ b/ntex/CHANGES.md @@ -4,6 +4,8 @@ * http: Refactor date service +* http: Do not leak request/response pools + * server: Rename ServerBulder::system_exit to stop_runtime * util: Drop Either service, use Variant instead diff --git a/ntex/src/http/message.rs b/ntex/src/http/message.rs index 1610f311..e4663a87 100644 --- a/ntex/src/http/message.rs +++ b/ntex/src/http/message.rs @@ -33,7 +33,9 @@ bitflags! { pub(crate) trait Head: Default + 'static { fn clear(&mut self); - fn pool() -> &'static MessagePool; + fn with_pool(f: F) -> R + where + F: FnOnce(&MessagePool) -> R; } #[derive(Debug)] @@ -68,8 +70,11 @@ impl Head for RequestHead { self.extensions.borrow_mut().clear(); } - fn pool() -> &'static MessagePool { - REQUEST_POOL.with(|p| *p) + fn with_pool(f: F) -> R + where + F: FnOnce(&MessagePool) -> R, + { + REQUEST_POOL.with(|p| f(p)) } } @@ -327,7 +332,7 @@ pub(crate) struct Message { impl Message { /// Get new message from the pool of objects pub(crate) fn new() -> Self { - T::pool().get_message() + T::with_pool(|p| p.get_message()) } } @@ -348,7 +353,7 @@ impl std::ops::DerefMut for Message { impl Drop for Message { fn drop(&mut self) { if Rc::strong_count(&self.head) == 1 { - T::pool().release(self.head.clone()); + T::with_pool(|p| p.release(self.head.clone())); } } } @@ -395,18 +400,17 @@ pub(crate) struct MessagePool(RefCell>>); /// Request's objects pool pub(super) struct BoxedResponsePool(RefCell>>); -thread_local!(static REQUEST_POOL: &'static MessagePool = MessagePool::::create()); -thread_local!(static RESPONSE_POOL: &'static BoxedResponsePool = BoxedResponsePool::create()); +thread_local!(static REQUEST_POOL: MessagePool = MessagePool::::create()); +thread_local!(static RESPONSE_POOL: BoxedResponsePool = BoxedResponsePool::create()); impl MessagePool { - fn create() -> &'static MessagePool { - let pool = MessagePool(RefCell::new(Vec::with_capacity(128))); - Box::leak(Box::new(pool)) + fn create() -> MessagePool { + MessagePool(RefCell::new(Vec::with_capacity(128))) } /// Get message from the pool #[inline] - fn get_message(&'static self) -> Message { + fn get_message(&self) -> Message { if let Some(mut msg) = self.0.borrow_mut().pop() { if let Some(r) = Rc::get_mut(&mut msg) { r.clear(); @@ -430,14 +434,13 @@ impl MessagePool { } impl BoxedResponsePool { - fn create() -> &'static BoxedResponsePool { - let pool = BoxedResponsePool(RefCell::new(Vec::with_capacity(128))); - Box::leak(Box::new(pool)) + fn create() -> BoxedResponsePool { + BoxedResponsePool(RefCell::new(Vec::with_capacity(128))) } /// Get message from the pool #[inline] - fn get_message(&'static self, status: StatusCode) -> BoxedResponseHead { + fn get_message(&self, status: StatusCode) -> BoxedResponseHead { if let Some(mut head) = self.0.borrow_mut().pop() { head.reason = None; head.status = status;