1
Fork 0

libcore: use unboxed closures in the finally module

This commit is contained in:
Jorge Aparicio 2014-12-05 08:24:40 -05:00
parent e2a362f9bb
commit f18b255bce

View file

@ -30,29 +30,19 @@
#![experimental] #![experimental]
use ops::Drop; use ops::{Drop, FnMut, FnOnce};
/// A trait for executing a destructor unconditionally after a block of code, /// A trait for executing a destructor unconditionally after a block of code,
/// regardless of whether the blocked fails. /// regardless of whether the blocked fails.
pub trait Finally<T> { pub trait Finally<T> {
/// Executes this object, unconditionally running `dtor` after this block of /// Executes this object, unconditionally running `dtor` after this block of
/// code has run. /// code has run.
fn finally(&mut self, dtor: ||) -> T; fn finally<F>(&mut self, dtor: F) -> T where F: FnMut();
} }
impl<'a,T> Finally<T> for ||: 'a -> T { impl<T, F> Finally<T> for F where F: FnMut() -> T {
fn finally(&mut self, dtor: ||) -> T { fn finally<G>(&mut self, mut dtor: G) -> T where G: FnMut() {
try_finally(&mut (), self, try_finally(&mut (), self, |_, f| (*f)(), |_| dtor())
|_, f| (*f)(),
|_| dtor())
}
}
impl<T> Finally<T> for fn() -> T {
fn finally(&mut self, dtor: ||) -> T {
try_finally(&mut (), (),
|_, _| (*self)(),
|_| dtor())
} }
} }
@ -86,11 +76,10 @@ impl<T> Finally<T> for fn() -> T {
/// // use state.buffer, state.len to cleanup /// // use state.buffer, state.len to cleanup
/// }) /// })
/// ``` /// ```
pub fn try_finally<T,U,R>(mutate: &mut T, pub fn try_finally<T, U, R, F, G>(mutate: &mut T, drop: U, try_fn: F, finally_fn: G) -> R where
drop: U, F: FnOnce(&mut T, U) -> R,
try_fn: |&mut T, U| -> R, G: FnMut(&mut T),
finally_fn: |&mut T|) {
-> R {
let f = Finallyalizer { let f = Finallyalizer {
mutate: mutate, mutate: mutate,
dtor: finally_fn, dtor: finally_fn,
@ -98,13 +87,13 @@ pub fn try_finally<T,U,R>(mutate: &mut T,
try_fn(&mut *f.mutate, drop) try_fn(&mut *f.mutate, drop)
} }
struct Finallyalizer<'a,A:'a> { struct Finallyalizer<'a, A:'a, F> where F: FnMut(&mut A) {
mutate: &'a mut A, mutate: &'a mut A,
dtor: |&mut A|: 'a dtor: F,
} }
#[unsafe_destructor] #[unsafe_destructor]
impl<'a,A> Drop for Finallyalizer<'a,A> { impl<'a, A, F> Drop for Finallyalizer<'a, A, F> where F: FnMut(&mut A) {
#[inline] #[inline]
fn drop(&mut self) { fn drop(&mut self) {
(self.dtor)(self.mutate); (self.dtor)(self.mutate);