2021-12-30 00:14:23 +00:00
|
|
|
/*
|
2022-06-02 18:51:06 +00:00
|
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
2021-12-30 00:14:23 +00:00
|
|
|
* 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.
|
|
|
|
*/
|
2022-10-26 19:18:14 +00:00
|
|
|
|
2022-10-27 17:56:09 +00:00
|
|
|
// FIXME: This test does some very x86_64-specific things.
|
|
|
|
#![cfg(target_arch = "x86_64")]
|
2021-12-30 00:14:23 +00:00
|
|
|
|
|
|
|
// when we convert syscall, such as open -> openat, the old syscall
|
|
|
|
// args should not be clobbered, even with the conversion.
|
|
|
|
|
2022-06-30 21:56:20 +00:00
|
|
|
use reverie::syscalls::Syscall;
|
|
|
|
use reverie::Error;
|
|
|
|
use reverie::Guest;
|
|
|
|
use reverie::Tool;
|
2021-12-30 00:14:23 +00:00
|
|
|
|
2022-11-03 19:53:11 +00:00
|
|
|
#[derive(Debug, Default, Clone)]
|
2021-12-30 00:14:23 +00:00
|
|
|
struct LocalStateTailInject;
|
|
|
|
|
2022-11-03 19:53:11 +00:00
|
|
|
#[derive(Debug, Default, Clone)]
|
2021-12-30 00:14:23 +00:00
|
|
|
struct LocalStateInject;
|
|
|
|
|
|
|
|
#[reverie::tool]
|
|
|
|
impl Tool for LocalStateTailInject {
|
2022-11-18 20:36:51 +00:00
|
|
|
type GlobalState = ();
|
|
|
|
type ThreadState = ();
|
|
|
|
|
2021-12-30 00:14:23 +00:00
|
|
|
async fn handle_syscall_event<T: Guest<Self>>(
|
|
|
|
&self,
|
|
|
|
guest: &mut T,
|
|
|
|
syscall: Syscall,
|
|
|
|
) -> Result<i64, Error> {
|
|
|
|
match syscall {
|
|
|
|
Syscall::Open(open_syscall) => {
|
|
|
|
guest
|
|
|
|
.tail_inject(reverie::syscalls::Openat::from(open_syscall))
|
|
|
|
.await
|
|
|
|
}
|
|
|
|
_ => guest.tail_inject(syscall).await,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[reverie::tool]
|
|
|
|
impl Tool for LocalStateInject {
|
2022-11-18 20:36:51 +00:00
|
|
|
type GlobalState = ();
|
|
|
|
type ThreadState = ();
|
|
|
|
|
2021-12-30 00:14:23 +00:00
|
|
|
async fn handle_syscall_event<T: Guest<Self>>(
|
|
|
|
&self,
|
|
|
|
guest: &mut T,
|
|
|
|
syscall: Syscall,
|
|
|
|
) -> Result<i64, Error> {
|
|
|
|
match syscall {
|
|
|
|
Syscall::Open(open_syscall) => Ok(guest
|
|
|
|
.inject(reverie::syscalls::Openat::from(open_syscall))
|
|
|
|
.await?),
|
|
|
|
_ => guest.tail_inject(syscall).await,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(all(not(sanitized), test))]
|
|
|
|
mod tests {
|
|
|
|
use nix::unistd;
|
|
|
|
use reverie_ptrace::testing::check_fn;
|
|
|
|
|
2022-08-06 15:21:44 +00:00
|
|
|
use super::*;
|
|
|
|
|
2021-12-30 00:14:23 +00:00
|
|
|
#[cfg(target_arch = "x86_64")]
|
|
|
|
#[allow(unused_mut)]
|
2022-03-10 20:05:42 +00:00
|
|
|
unsafe fn open_syscall_sanity_check() -> i32 {
|
|
|
|
let path = b"/dev/null\0".as_ptr() as usize;
|
|
|
|
let flags: usize = 0x8000; // O_LARGEFILE
|
|
|
|
let mode: usize = 0o644;
|
|
|
|
|
|
|
|
let mut ret: usize;
|
|
|
|
|
|
|
|
// // The following asm block does this:
|
|
|
|
// let ret = open("/dev/null", 0x8000, 0644);
|
|
|
|
// if ret >= -4095 as u64 { exit_group(1) }
|
|
|
|
// // Sanity check input registers to ensure they didn't change.
|
|
|
|
// if %rsi != 0x8000 { exit_group(1) }
|
|
|
|
// if %rdx != 0644 { exit_roup(1) }
|
|
|
|
// return fd
|
|
|
|
core::arch::asm!(
|
2022-04-12 18:09:06 +00:00
|
|
|
"mov r8, rdi",
|
|
|
|
// Set syscall to `SYS_open`
|
|
|
|
"mov rax, 2",
|
2022-03-10 20:05:42 +00:00
|
|
|
"syscall",
|
|
|
|
// if (ret >= -4095 as u64) goto 1
|
2022-04-12 18:09:06 +00:00
|
|
|
"cmp rax, 0xfffffffffffff001",
|
2022-03-10 20:05:42 +00:00
|
|
|
"jae 2f",
|
|
|
|
// if (rax != r8) goto label1;
|
2022-04-12 18:09:06 +00:00
|
|
|
"cmp r8, rdi",
|
2022-03-10 20:05:42 +00:00
|
|
|
"jne 2f",
|
|
|
|
// if (rsi != 0x8000) goto label1;
|
|
|
|
"cmp rsi, 0x8000",
|
|
|
|
"jne 2f",
|
|
|
|
// if (rdx != 0644) goto label1;
|
|
|
|
"cmp rdx, 0x1a4",
|
|
|
|
"jne 2f",
|
|
|
|
// Otherwise, we're successful.
|
|
|
|
"jmp 3f",
|
|
|
|
"2:",
|
|
|
|
// Set syscall arg1 to label1
|
|
|
|
"mov rdi, 0x1",
|
2022-04-12 18:09:06 +00:00
|
|
|
// Set syscall to SYS_exit_group
|
|
|
|
"mov rax, 231",
|
2022-03-10 20:05:42 +00:00
|
|
|
// Do the syscall
|
|
|
|
"syscall",
|
|
|
|
"3:",
|
|
|
|
lateout("rax") ret,
|
2022-04-12 18:09:06 +00:00
|
|
|
in("rdi") path,
|
2022-03-10 20:05:42 +00:00
|
|
|
in("rsi") flags,
|
|
|
|
in("rdx") mode,
|
|
|
|
out("r8") _, // Clobbered
|
|
|
|
out("rcx") _, // rcx is used to store old rip
|
|
|
|
out("r11") _, // r11 is used to store old rflags
|
|
|
|
);
|
|
|
|
|
2022-04-12 18:09:06 +00:00
|
|
|
ret as i32
|
2022-03-10 20:05:42 +00:00
|
|
|
}
|
|
|
|
|
2021-12-30 00:14:23 +00:00
|
|
|
#[cfg(not(target_arch = "x86_64"))]
|
|
|
|
unsafe fn open_syscall_sanity_check() -> i32 {
|
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn open_into_openat_tail_inject_test() {
|
|
|
|
check_fn::<LocalStateTailInject, _>(move || {
|
|
|
|
let fd = unsafe { open_syscall_sanity_check() };
|
|
|
|
assert!(unistd::close(fd).is_ok());
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn open_into_openat_inject_test() {
|
|
|
|
check_fn::<LocalStateInject, _>(move || {
|
|
|
|
let fd = unsafe { open_syscall_sanity_check() };
|
|
|
|
assert!(unistd::close(fd).is_ok());
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|