#![cfg(target_os = "linux")]
#![expect(missing_docs)]
use anyhow::Context;
use std::fs::File;
use std::io::ErrorKind;
use std::os::unix::fs::OpenOptionsExt;
use std::path::Path;
use tracing::Level;
const KMSG_NOTE_BYTES: usize = 1024 * 256; pub fn main() -> ! {
if let Err(e) = do_main() {
tracing::error!(?e, "core dump error");
std::process::exit(libc::EXIT_FAILURE)
}
std::process::exit(libc::EXIT_SUCCESS)
}
pub fn do_main() -> anyhow::Result<()> {
let mut args = std::env::args().skip(1).peekable();
let level = if args.peek().is_some_and(|x| x == "-v") {
args.next();
Level::DEBUG
} else {
Level::INFO
};
tracing_subscriber::fmt()
.with_writer(std::io::stderr)
.log_internal_errors(true)
.with_max_level(level)
.with_timer(tracing_subscriber::fmt::time::uptime())
.compact()
.with_ansi(false)
.init();
if underhill_confidentiality::confidential_filtering_enabled() {
tracing::info!("crash reporting disabled due to CVM");
std::process::exit(libc::EXIT_FAILURE);
}
let pid: i32 = args
.next()
.context("missing pid")?
.parse()
.context("failed to parse pid")?;
if args.next().is_some() {
anyhow::bail!("unexpected extra arguments");
}
let mut builder = elfcore::CoreDumpBuilder::new(pid)?;
let mut kmsg_file = NonBlockingFile::new("/dev/kmsg");
match kmsg_file.as_mut() {
Ok(kmsg_file) => _ = builder.add_custom_file_note("KMSG", kmsg_file, KMSG_NOTE_BYTES),
Err(e) => tracing::error!("Failed to open KMSG file: {:?}", e),
}
let n = builder
.write(std::io::stdout().lock())
.context("failed to write core dump")?;
tracing::info!("dump: {} bytes", n);
Ok(())
}
struct NonBlockingFile(File);
impl NonBlockingFile {
fn new<P: AsRef<Path>>(path: P) -> std::io::Result<Self> {
Ok(Self(
std::fs::OpenOptions::new()
.read(true)
.custom_flags(libc::O_NONBLOCK)
.open(path)?,
))
}
}
impl std::io::Read for NonBlockingFile {
fn read(&mut self, buf: &mut [u8]) -> std::io::Result<usize> {
match self.0.read(buf) {
Ok(len) => Ok(len),
Err(ref err) if err.kind() == ErrorKind::WouldBlock => Ok(0),
Err(ref err)
if err.kind() == ErrorKind::Interrupted || err.kind() == ErrorKind::BrokenPipe =>
{
self.read(buf)
}
Err(e) => Err(e),
}
}
}