mirror of
https://github.com/facebookexperimental/reverie.git
synced 2025-01-23 05:06:23 +00:00
20c7020868
Summary: The OSS license linter doesn't like the word "its". Reviewed By: johnhurt Differential Revision: D36856385 fbshipit-source-id: 909037d96de8976f08004497d28b77838f8c6870
85 lines
2.2 KiB
Rust
85 lines
2.2 KiB
Rust
/*
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
//! Tests surrounding exit logic.
|
|
|
|
use reverie::{
|
|
syscalls::{self, Syscall, SyscallInfo, Sysno},
|
|
Error, ExitStatus, GlobalRPC, GlobalTool, Guest, Pid, Tool,
|
|
};
|
|
use serde::{Deserialize, Serialize};
|
|
use std::sync::Mutex;
|
|
|
|
#[derive(Debug, Serialize, Deserialize, Default)]
|
|
struct GlobalState {
|
|
// FIXME: Can't use (Pid, ExitStatus) types here since they don't implement
|
|
// Serialize/Deserialize.
|
|
exited: Mutex<Vec<(i32, ExitStatus)>>,
|
|
}
|
|
|
|
#[reverie::global_tool]
|
|
impl GlobalTool for GlobalState {
|
|
type Request = ExitStatus;
|
|
type Response = ();
|
|
|
|
async fn receive_rpc(&self, from: Pid, exit_status: ExitStatus) -> Self::Response {
|
|
self.exited
|
|
.lock()
|
|
.unwrap()
|
|
.push((from.as_raw(), exit_status));
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Deserialize, Default, Clone)]
|
|
struct InjectExitTool {}
|
|
|
|
#[reverie::tool]
|
|
impl Tool for InjectExitTool {
|
|
type GlobalState = GlobalState;
|
|
|
|
async fn on_exit_process<G: GlobalRPC<Self::GlobalState>>(
|
|
self,
|
|
_pid: Pid,
|
|
global_state: &G,
|
|
exit_status: ExitStatus,
|
|
) -> Result<(), Error> {
|
|
global_state.send_rpc(exit_status).await?;
|
|
Ok(())
|
|
}
|
|
|
|
async fn handle_syscall_event<T: Guest<Self>>(
|
|
&self,
|
|
guest: &mut T,
|
|
syscall: Syscall,
|
|
) -> Result<i64, Error> {
|
|
if syscall.number() == Sysno::getpid {
|
|
guest
|
|
.tail_inject(syscalls::ExitGroup::new().with_status(42))
|
|
.await
|
|
} else {
|
|
guest.tail_inject(syscall).await
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(not(sanitized))]
|
|
#[test]
|
|
fn smoke() {
|
|
use reverie_ptrace::testing::test_fn;
|
|
|
|
let (output, state) = test_fn::<InjectExitTool, _>(|| unsafe {
|
|
let _ = libc::getpid();
|
|
libc::syscall(libc::SYS_exit_group, 0);
|
|
})
|
|
.unwrap();
|
|
assert_eq!(output.status, ExitStatus::Exited(42));
|
|
let mut mg = state.exited.lock().unwrap();
|
|
assert_eq!(mg.pop().map(|x| x.1), Some(ExitStatus::Exited(42)));
|
|
assert!(mg.is_empty());
|
|
}
|