use std::borrow::Borrow;
use std::ffi::{CStr, OsStr};
use std::fmt::Write;
use std::os::unix::ffi::OsStrExt;
use std::thread;
use anyhow::Context;
use clap::Parser;
use nix::sys::{personality, resource, signal};
use signal_hook::{consts, iterator::Signals};
use crate::core::configuration::{CliOptions, ConfigFileOptions, ConfigOptions};
use crate::core::controller::Controller;
use crate::core::logger::shadow_logger;
use crate::core::sim_config::SimConfig;
use crate::core::worker;
use crate::cshadow as c;
use crate::utility::shm_cleanup;
use shadow_build_info::{BUILD_TIMESTAMP, GIT_BRANCH, GIT_COMMIT_INFO, GIT_DATE};
const HELP_INFO_STR: &str =
"For more information, visit https://shadow.github.io or https://github.com/shadow";
pub fn run_shadow(args: Vec<&OsStr>) -> anyhow::Result<()> {
let _guard = unsafe { crate::shadow_shmem::allocator::SharedMemAllocatorDropGuard::new() };
verify_glib_version().context("Unsupported GLib version")?;
let mut signals_list = Signals::new([consts::signal::SIGINT, consts::signal::SIGTERM])?;
thread::spawn(move || {
if let Some(signal) = signals_list.forever().next() {
log::info!("Received signal {}. Flushing log and exiting", signal);
log::logger().flush();
std::process::exit(1);
}
log::debug!("Finished waiting for a signal");
});
signal::sigprocmask(
signal::SigmaskHow::SIG_SETMASK,
Some(&signal::SigSet::empty()),
None,
)?;
let options = match CliOptions::try_parse_from(args.clone()) {
Ok(x) => x,
Err(e) => {
e.print().unwrap();
if e.use_stderr() {
std::process::exit(1);
} else {
std::process::exit(0);
}
}
};
if options.show_build_info {
write_build_info(std::io::stderr()).unwrap();
std::process::exit(0);
}
if options.shm_cleanup {
shm_cleanup::shm_cleanup(shm_cleanup::SHM_DIR_PATH)
.context("Cleaning shared memory files")?;
std::process::exit(0);
}
let config_filename: String = match options.config.as_ref().unwrap().as_str() {
"-" => "/dev/stdin",
x => x,
}
.into();
let config_file = load_config_file(&config_filename, true)
.with_context(|| format!("Failed to load configuration file {}", config_filename))?;
let shadow_config = ConfigOptions::new(config_file, options.clone());
if options.show_config {
eprintln!("{:#?}", shadow_config);
return Ok(());
}
if shadow_config.experimental.use_object_counters.unwrap() {
worker::enable_object_counters();
}
let log_level = shadow_config.general.log_level.unwrap();
let log_level: log::Level = log_level.into();
shadow_logger::init(
log_level.to_level_filter(),
shadow_config.experimental.report_errors_to_stderr.unwrap(),
)
.unwrap();
shadow_logger::set_buffering_enabled(false);
if log_level > log::STATIC_MAX_LEVEL {
log::warn!(
"Log level set to {}, but messages higher than {} have been compiled out",
log_level,
log::STATIC_MAX_LEVEL,
);
}
if nix::unistd::getuid().is_root() {
log::warn!(
"Shadow is running as root. Shadow does not emulate Linux permissions, and some
applications may behave differently when running as root. It is recommended to run
Shadow as a non-root user."
);
} else if nix::unistd::geteuid().is_root() {
log::warn!(
"Shadow is running with root privileges. Shadow does not emulate Linux permissions,
and some applications may behave differently when running with root privileges. It
is recommended to run Shadow as a non-root user."
);
}
if let Err(e) = shm_cleanup::shm_cleanup(shm_cleanup::SHM_DIR_PATH) {
log::warn!("Unable to clean up shared memory files: {:?}", e);
}
if shadow_config.experimental.use_cpu_pinning.unwrap() {
#[allow(clippy::collapsible_if)]
if unsafe { c::affinity_initPlatformInfo() } != 0 {
return Err(anyhow::anyhow!("Unable to initialize platform info"));
}
}
raise_rlimit(resource::Resource::RLIMIT_NOFILE).context("Could not raise fd limit")?;
raise_rlimit(resource::Resource::RLIMIT_NPROC).context("Could not raise proc limit")?;
if shadow_config.experimental.use_sched_fifo.unwrap() {
set_sched_fifo().context("Could not set real-time scheduler mode to SCHED_FIFO")?;
log::debug!("Successfully set real-time scheduler mode to SCHED_FIFO");
}
match disable_aslr() {
Ok(()) => log::debug!("ASLR disabled for processes forked from this parent process"),
Err(e) => log::warn!("Could not disable address space layout randomization. This may affect determinism: {e:#}"),
};
if sidechannel_mitigations_enabled().context("Failed to get sidechannel mitigation status")? {
log::warn!(
"Speculative Store Bypass sidechannel mitigation is enabled (perhaps by seccomp?). \
This typically adds ~30% performance overhead."
);
}
eprintln!("** Starting Shadow {}", env!("CARGO_PKG_VERSION"));
let mut build_info = Vec::new();
write_build_info(&mut build_info).unwrap();
for line in std::str::from_utf8(&build_info).unwrap().trim().split('\n') {
log::info!("{line}");
}
log::info!("Logging current startup arguments and environment");
log_environment(args.clone());
if let Err(e) = verify_supported_system() {
log::warn!("Couldn't verify supported system: {e:?}")
}
log::debug!("Startup checks passed, we are ready to start the simulation");
if options.gdb {
pause_for_gdb_attach().context("Could not pause shadow to allow gdb to attach")?;
}
let sim_config = SimConfig::new(&shadow_config, &options.debug_hosts.unwrap_or_default())
.context("Failed to initialize the simulation")?;
let controller = Controller::new(sim_config, &shadow_config);
let buffer_log = !log::log_enabled!(log::Level::Trace);
shadow_logger::set_buffering_enabled(buffer_log);
if buffer_log {
log::info!("Log message buffering is enabled for efficiency");
}
controller.run().context("Failed to run the simulation")?;
shadow_logger::set_buffering_enabled(false);
if buffer_log {
log::info!("Log message buffering is disabled during cleanup");
}
Ok(())
}
pub fn version() -> String {
let mut s = env!("CARGO_PKG_VERSION").to_string();
if let (Some(commit), Some(date)) = (GIT_COMMIT_INFO, GIT_DATE) {
write!(s, " — {commit} {date}").unwrap();
}
s
}
fn write_build_info(mut w: impl std::io::Write) -> std::io::Result<()> {
writeln!(w, "Shadow {}", version())?;
writeln!(
w,
"GLib {}.{}.{}",
c::GLIB_MAJOR_VERSION,
c::GLIB_MINOR_VERSION,
c::GLIB_MICRO_VERSION,
)?;
writeln!(w, "Built on {}", BUILD_TIMESTAMP)?;
writeln!(
w,
"Built from git branch {}",
GIT_BRANCH.unwrap_or("<unknown>"),
)?;
writeln!(w, "{}", env!("SHADOW_BUILD_INFO"))?;
writeln!(w, "{HELP_INFO_STR}")?;
Ok(())
}
fn verify_supported_system() -> anyhow::Result<()> {
let uts_name = nix::sys::utsname::uname()?;
let sysname = uts_name
.sysname()
.to_str()
.with_context(|| "Decoding system name")?;
if sysname != "Linux" {
anyhow::bail!("Unsupported sysname: {sysname}");
}
let version = uts_name
.release()
.to_str()
.with_context(|| "Decoding system release")?;
let mut version_parts = version.split('.');
let Some(major) = version_parts.next() else {
anyhow::bail!("Couldn't find major version in : {version}");
};
let major: i32 = major
.parse()
.with_context(|| format!("Parsing major version number '{major}'"))?;
let Some(minor) = version_parts.next() else {
anyhow::bail!("Couldn't find minor version in : {version}");
};
let minor: i32 = minor
.parse()
.with_context(|| format!("Parsing minor version number '{minor}'"))?;
const MIN_KERNEL_VERSION: (i32, i32) = (5, 4);
if (major, minor) < MIN_KERNEL_VERSION {
anyhow::bail!(
"kernel version {major}.{minor} is older than minimum supported version {}.{}",
MIN_KERNEL_VERSION.0,
MIN_KERNEL_VERSION.1
);
}
Ok(())
}
fn verify_glib_version() -> anyhow::Result<()> {
if c::GLIB_MAJOR_VERSION == 2 && c::GLIB_MINOR_VERSION == 40 {
anyhow::bail!(
"You compiled against GLib version {}.{}.{}, which has bugs known to break \"
Shadow. Please update to a newer version of GLib.",
c::GLIB_MAJOR_VERSION,
c::GLIB_MINOR_VERSION,
c::GLIB_MICRO_VERSION,
);
}
let mismatch = unsafe {
c::glib_check_version(
c::GLIB_MAJOR_VERSION,
c::GLIB_MINOR_VERSION,
c::GLIB_MICRO_VERSION,
)
};
if !mismatch.is_null() {
let mismatch = unsafe { std::ffi::CStr::from_ptr(mismatch) };
anyhow::bail!(
"The version of the run-time GLib library ({}.{}.{}) is not compatible with \
the version against which Shadow was compiled ({}.{}.{}). GLib message: '{}'.",
unsafe { c::glib_major_version },
unsafe { c::glib_minor_version },
unsafe { c::glib_micro_version },
c::GLIB_MAJOR_VERSION,
c::GLIB_MINOR_VERSION,
c::GLIB_MICRO_VERSION,
mismatch.to_string_lossy(),
);
}
Ok(())
}
fn load_config_file(
filename: impl AsRef<std::path::Path>,
extended_yaml: bool,
) -> anyhow::Result<ConfigFileOptions> {
let file = std::fs::File::open(filename).context("Could not open config file")?;
let mut config_file: serde_yaml::Value =
serde_yaml::from_reader(file).context("Could not parse configuration file as yaml")?;
if extended_yaml {
config_file
.apply_merge()
.context("Could not merge '<<' keys")?;
if let serde_yaml::Value::Mapping(ref mut mapping) = &mut config_file {
mapping.retain(|key, _value| {
if let serde_yaml::Value::String(key) = key {
if key.starts_with("x-") {
return false;
}
}
true
});
}
}
serde_yaml::from_value(config_file).context("Could not parse configuration file")
}
fn pause_for_gdb_attach() -> anyhow::Result<()> {
let pid = nix::unistd::getpid();
log::info!(
"Pausing with SIGTSTP to enable debugger attachment (pid {})",
pid
);
eprintln!(
"** Pausing with SIGTSTP to enable debugger attachment (pid {})",
pid
);
signal::raise(signal::Signal::SIGTSTP)?;
log::info!("Resuming now");
Ok(())
}
fn set_sched_fifo() -> anyhow::Result<()> {
let mut param: libc::sched_param = unsafe { std::mem::zeroed() };
param.sched_priority = 1;
let rv = nix::errno::Errno::result(unsafe {
libc::sched_setscheduler(0, libc::SCHED_FIFO, std::ptr::from_ref(¶m))
})
.context("Could not set kernel SCHED_FIFO")?;
assert_eq!(rv, 0);
Ok(())
}
fn raise_rlimit(resource: resource::Resource) -> anyhow::Result<()> {
let (_soft_limit, hard_limit) = resource::getrlimit(resource)?;
resource::setrlimit(resource, hard_limit, hard_limit)?;
Ok(())
}
fn disable_aslr() -> anyhow::Result<()> {
let pers = personality::get().context("Could not get personality")?;
personality::set(pers | personality::Persona::ADDR_NO_RANDOMIZE)
.context("Could not set personality")?;
Ok(())
}
fn sidechannel_mitigations_enabled() -> anyhow::Result<bool> {
let state = nix::errno::Errno::result(unsafe {
libc::prctl(
libc::PR_GET_SPECULATION_CTRL,
libc::PR_SPEC_STORE_BYPASS,
0,
0,
0,
)
})
.context("Failed prctl()")?;
let state = state as u32;
Ok((state & libc::PR_SPEC_DISABLE) != 0)
}
fn log_environment(args: Vec<&OsStr>) {
for arg in args {
log::info!("arg: {}", arg.to_string_lossy());
}
for (key, value) in std::env::vars_os() {
let level = match key.to_string_lossy().borrow() {
"LD_PRELOAD" | "LD_STATIC_TLS_EXTRA" | "G_DEBUG" | "G_SLICE" => log::Level::Info,
_ => log::Level::Trace,
};
log::log!(level, "env: {:?}={:?}", key, value);
}
}
mod export {
use super::*;
#[no_mangle]
pub extern "C-unwind" fn main_runShadow(
argc: libc::c_int,
argv: *const *const libc::c_char,
) -> libc::c_int {
let args = (0..argc).map(|x| unsafe { CStr::from_ptr(*argv.add(x as usize)) });
let args = args.map(|x| OsStr::from_bytes(x.to_bytes()));
let result = run_shadow(args.collect());
log::logger().flush();
if let Err(e) = result {
if log::log_enabled!(log::Level::Error) {
for line in format!("{:?}", e).split('\n') {
log::error!("{}", line);
}
log::logger().flush();
eprintln!("** Shadow did not complete successfully: {}", e);
eprintln!("** {}", e.root_cause());
eprintln!("** See the log for details");
} else {
eprintln!("{:?}", e);
}
return 1;
}
eprintln!("** Shadow completed successfully");
0
}
}