wasmer-wasix 0.702.0

WASI and WASIX implementation library for Wasmer WebAssembly runtime
use super::*;
use crate::{WasiVForkAsyncify, syscalls::*};

/// ### `proc_exit2()`
/// Similar to `proc_exit()`
///
/// If used in a `proc_fork_env` (vfork) context it will exit the
/// child, switch the process back into the parent process and
/// return. If used for vforking, restoring the control to the
/// place where the vfork happened is the responsibility of the
/// caller.
#[instrument(level = "trace", skip(ctx))]
pub fn proc_exit2<M: MemorySize>(
    mut ctx: FunctionEnvMut<'_, WasiEnv>,
    code: ExitCode,
) -> Result<(), WasiError> {
    WasiEnv::do_pending_operations(&mut ctx)?;

    let Some(mut vfork) = ctx.data_mut().vfork.take() else {
        // Not in a vfork, just exit normally
        return Err(WasiError::Exit(code));
    };

    tracing::debug!(
        parent_pid = %vfork.env.process.pid(),
        child_pid = %ctx.data().process.pid(),
        "proc_exit from vfork, returning control to parent process"
    );

    // Prepare the child env for teardown by closing its FDs
    block_on(
        unsafe { ctx.data().get_memory_and_wasi_state(&ctx, 0) }
            .1
            .fs
            .close_all(),
    );

    // Restore the WasiEnv to the point when we vforked
    let mut parent_env = vfork.env;
    ctx.data_mut().swap_inner(parent_env.as_mut());
    let mut child_env = std::mem::replace(ctx.data_mut(), *parent_env);

    // Terminate the child process
    child_env.owned_handles.push(vfork.handle);
    child_env.process.terminate(code);

    let Some(asyncify_info) = vfork.asyncify else {
        // vfork without asyncify only forks the WasiEnv, which we have restored
        // above. We now return to the guest side in the parent process. Restoring
        // the control flow is done on the guest side.
        // See `proc_fork_env()` for information about this.

        return Ok(());
    };

    // Jump back to the vfork point and continue execution
    let child_pid = child_env.process.pid();
    let rewind_stack = asyncify_info.rewind_stack.freeze();
    let store_data = asyncify_info.store_data;
    unwind::<M, _>(ctx, move |mut ctx, _, _| {
        // Now rewind the previous stack and carry on from where we did the vfork
        match rewind::<M, _>(
            ctx,
            None,
            rewind_stack,
            store_data,
            ForkResult {
                pid: child_pid.raw() as Pid,
                ret: Errno::Success,
            },
        ) {
            Errno::Success => OnCalledAction::InvokeAgain,
            err => {
                warn!("fork failed - could not rewind the stack - errno={}", err);
                OnCalledAction::Trap(Box::new(WasiError::Exit(err.into())))
            }
        }
    })?;
    Ok(())
}