//! Implementation of `std::os` functionality for unix systems #![allow(unused_imports)] // lots of cfg code here use libc::c_char; use crate::os::unix::prelude::*; use crate::error::Error as StdError; use crate::ffi::{CStr, CString, OsStr, OsString}; use crate::fmt; use crate::io::{self, Read, Write}; use crate::iter; use crate::marker::PhantomData; use crate::mem; use crate::memchr; use crate::path::{self, PathBuf}; use crate::ptr; use crate::slice; use crate::str; use crate::sys_common::mutex::Mutex; use crate::sys::{cvt, cvt_libc, fd, syscall}; use crate::vec; extern { #[link_name = "__errno_location"] fn errno_location() -> *mut i32; } /// Returns the platform-specific value of errno pub fn errno() -> i32 { unsafe { (*errno_location()) } } /// Gets a detailed string description for the given error number. pub fn error_string(errno: i32) -> String { if let Some(string) = syscall::STR_ERROR.get(errno as usize) { string.to_string() } else { "unknown error".to_string() } } pub fn getcwd() -> io::Result { let mut buf = [0; 4096]; let count = cvt(syscall::getcwd(&mut buf))?; Ok(PathBuf::from(OsString::from_vec(buf[.. count].to_vec()))) } pub fn chdir(p: &path::Path) -> io::Result<()> { cvt(syscall::chdir(p.to_str().unwrap())).and(Ok(())) } pub struct SplitPaths<'a> { iter: iter::Map bool>, fn(&'a [u8]) -> PathBuf>, } pub fn split_paths(unparsed: &OsStr) -> SplitPaths { fn bytes_to_path(b: &[u8]) -> PathBuf { PathBuf::from(::from_bytes(b)) } fn is_semicolon(b: &u8) -> bool { *b == b';' } let unparsed = unparsed.as_bytes(); SplitPaths { iter: unparsed.split(is_semicolon as fn(&u8) -> bool) .map(bytes_to_path as fn(&[u8]) -> PathBuf) } } impl<'a> Iterator for SplitPaths<'a> { type Item = PathBuf; fn next(&mut self) -> Option { self.iter.next() } fn size_hint(&self) -> (usize, Option) { self.iter.size_hint() } } #[derive(Debug)] pub struct JoinPathsError; pub fn join_paths(paths: I) -> Result where I: Iterator, T: AsRef { let mut joined = Vec::new(); let sep = b';'; for (i, path) in paths.enumerate() { let path = path.as_ref().as_bytes(); if i > 0 { joined.push(sep) } if path.contains(&sep) { return Err(JoinPathsError) } joined.extend_from_slice(path); } Ok(OsStringExt::from_vec(joined)) } impl fmt::Display for JoinPathsError { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { "path segment contains separator `:`".fmt(f) } } impl StdError for JoinPathsError { fn description(&self) -> &str { "failed to join paths" } } pub fn current_exe() -> io::Result { use crate::fs::File; let mut file = File::open("sys:exe")?; let mut path = String::new(); file.read_to_string(&mut path)?; if path.ends_with('\n') { path.pop(); } Ok(PathBuf::from(path)) } pub static ENV_LOCK: Mutex = Mutex::new(); pub struct Env { iter: vec::IntoIter<(OsString, OsString)>, _dont_send_or_sync_me: PhantomData<*mut ()>, } impl Iterator for Env { type Item = (OsString, OsString); fn next(&mut self) -> Option<(OsString, OsString)> { self.iter.next() } fn size_hint(&self) -> (usize, Option) { self.iter.size_hint() } } pub unsafe fn environ() -> *mut *const *const c_char { extern { static mut environ: *const *const c_char; } &mut environ } /// Returns a vector of (variable, value) byte-vector pairs for all the /// environment variables of the current process. pub fn env() -> Env { unsafe { let _guard = ENV_LOCK.lock(); let mut environ = *environ(); if environ == ptr::null() { panic!("os::env() failure getting env string from OS: {}", io::Error::last_os_error()); } let mut result = Vec::new(); while *environ != ptr::null() { if let Some(key_value) = parse(CStr::from_ptr(*environ).to_bytes()) { result.push(key_value); } environ = environ.offset(1); } return Env { iter: result.into_iter(), _dont_send_or_sync_me: PhantomData, } } fn parse(input: &[u8]) -> Option<(OsString, OsString)> { // Strategy (copied from glibc): Variable name and value are separated // by an ASCII equals sign '='. Since a variable name must not be // empty, allow variable names starting with an equals sign. Skip all // malformed lines. if input.is_empty() { return None; } let pos = memchr::memchr(b'=', &input[1..]).map(|p| p + 1); pos.map(|p| ( OsStringExt::from_vec(input[..p].to_vec()), OsStringExt::from_vec(input[p+1..].to_vec()), )) } } pub fn getenv(k: &OsStr) -> io::Result> { // environment variables with a nul byte can't be set, so their value is // always None as well let k = CString::new(k.as_bytes())?; unsafe { let _guard = ENV_LOCK.lock(); let s = libc::getenv(k.as_ptr()) as *const libc::c_char; let ret = if s.is_null() { None } else { Some(OsStringExt::from_vec(CStr::from_ptr(s).to_bytes().to_vec())) }; Ok(ret) } } pub fn setenv(k: &OsStr, v: &OsStr) -> io::Result<()> { let k = CString::new(k.as_bytes())?; let v = CString::new(v.as_bytes())?; unsafe { let _guard = ENV_LOCK.lock(); cvt_libc(libc::setenv(k.as_ptr(), v.as_ptr(), 1)).map(|_| ()) } } pub fn unsetenv(n: &OsStr) -> io::Result<()> { let nbuf = CString::new(n.as_bytes())?; unsafe { let _guard = ENV_LOCK.lock(); cvt_libc(libc::unsetenv(nbuf.as_ptr())).map(|_| ()) } } pub fn page_size() -> usize { 4096 } pub fn temp_dir() -> PathBuf { crate::env::var_os("TMPDIR").map(PathBuf::from).unwrap_or_else(|| { PathBuf::from("/tmp") }) } pub fn home_dir() -> Option { return crate::env::var_os("HOME").map(PathBuf::from); } pub fn exit(code: i32) -> ! { let _ = syscall::exit(code as usize); unreachable!(); } pub fn getpid() -> u32 { syscall::getpid().unwrap() as u32 } pub fn getppid() -> u32 { syscall::getppid().unwrap() as u32 }