1use core::mem;
23use crate::ptr;
45/// This replaces the value behind the `v` unique reference by calling the
6/// relevant function.
7///
8/// If a panic occurs in the `change` closure, the entire process will be aborted.
9#[allow(dead_code)] // keep as illustration and for future use
10#[inline]
11pub(crate) fn take_mut<T, E>(v: &mut T, change: impl FnOnce(T) -> Result<T, E>) -> Result<(), E> {
12 replace(v, |value| Ok((change(value)?, ())))
13}
1415/// This replaces the value behind the `v` unique reference by calling the
16/// relevant function, and returns a result obtained along the way.
17///
18/// If a panic occurs in the `change` closure, the entire process will be aborted.
19#[inline]
20pub(crate) fn replace<T, R, E>(
21 v: &mut T,
22 change: impl FnOnce(T) -> Result<(T, R), E>,
23) -> Result<R, E> {
24struct PanicGuard;
2526impl Drop for PanicGuard {
27fn drop(&mut self) {
28crate::abort()
29 }
30 }
3132let guard = PanicGuard;
33let value = unsafe { ptr::read(v) };
34let (new_value, ret) = change(value)?;
35unsafe {
36 ptr::write(v, new_value);
37 }
38 mem::forget(guard);
39Ok(ret)
40}