mirror of
https://github.com/rust-lang/rust.git
synced 2025-06-09 05:38:32 +00:00

More ErrorKinds for common errnos From the commit message of the main commit here (as revised): ``` There are a number of IO error situations which it would be very useful for Rust code to be able to recognise without having to resort to OS-specific code. Taking some Unix examples, `ENOTEMPTY` and `EXDEV` have obvious recovery strategies. Recently I was surprised to discover that `ENOSPC` came out as `ErrorKind::Other`. Since I am familiar with Unix I reviwed the list of errno values in https://pubs.opengroup.org/onlinepubs/9699919799/basedefs/errno.h.html Here, I add those that most clearly seem to be needed. `@CraftSpider` provided information about Windows, and references, which I have tried to take into account. This has to be insta-stable because we can't sensibly have a different set of ErrorKinds depending on a std feature flag. I have *not* added these to the mapping tables for any operating systems other than Unix and Windows. I hope that it is OK to add them now for Unix and Windows now, and maybe add them to other OS's mapping tables as and when someone on that OS is able to consider the situation. I adopted the general principle that it was usually a bad idea to map two distinct error values to the same Rust error code. I notice that this principle is already violated in the case of `EACCES` and `EPERM`, which both map to `PermissionDenied`. I think this was probably a mistake but it would be quite hard to change now, so I don't propose to do anything about that. However, for Windows, there are sometimes different error codes for identical situations. Eg there are WSA* versions of some error codes as well as ERROR_* ones. Also Windows seems to have a great many more erorr codes. I don't know precisely what best practice would be for Windows. ``` <strike> ``` Errno values I wasn't sure about so *haven't* included: EMFILE ENFILE ENOBUFS ENOLCK: These are all fairly Unix-specific resource exhaustion situations. In practice it seemed not very likely to me that anyone would want to handle these differently to `Other`. ENOMEM ERANGE EDOM EOVERFLOW Normally these don't get exposed to the Rust callers I hope. They don't tend to come out of filesystem APIs. EILSEQ Hopefully Rust libraries open files in binary mode and do the converstion in Rust. So Rust code ought not to be exposed to EILSEQ. EIO The range of things that could cause this is troublesome. I found it difficult to describe. I do think it would be useful to add this at some point, because EIO on a filesystem operation is much more serious than most other errors. ENETDOWN I wasn't sure if this was useful or, indeed, if any modern systems use it. ENOEXEC It is not clear to me how a Rust program could respond to this. It seems rather niche. EPROTO ENETRESET ENODATA ENOMSG ENOPROTOOPT ENOSR ENOSTR ETIME ENOTRECOVERABLE EOWNERDEAD EBADMSG EPROTONOSUPPORT EPROTOTYPE EIDRM These are network or STREAMS related errors which I have never in my own Unix programming found the need to do anything with. I think someone who understands these better should be the one to try to find good Rust names and descriptions for them. ENOTTY ENXIO ENODEV EOPNOTSUPP ESRCH EALREADY ECANCELED ECHILD EINPROGRESS These are very hard to get unless you're already doing something very Unix-specific, in which case the raw_os_error interface is probably more suitable than relying on the Rust ErrorKind mapping. EFAULT EBADF These would seem to be the result of application UB. ``` </strike> <i>(omitted errnos are discussed below, especially in https://github.com/rust-lang/rust/pull/79965#issuecomment-810468334)
281 lines
9.2 KiB
Rust
281 lines
9.2 KiB
Rust
#![allow(missing_docs, nonstandard_style)]
|
|
|
|
use crate::io::ErrorKind;
|
|
|
|
pub use self::rand::hashmap_random_keys;
|
|
pub use libc::strlen;
|
|
|
|
#[macro_use]
|
|
pub mod weak;
|
|
|
|
pub mod alloc;
|
|
pub mod android;
|
|
pub mod args;
|
|
#[path = "../unix/cmath.rs"]
|
|
pub mod cmath;
|
|
pub mod condvar;
|
|
pub mod env;
|
|
pub mod fd;
|
|
pub mod fs;
|
|
pub mod futex;
|
|
pub mod io;
|
|
#[cfg(any(target_os = "linux", target_os = "android"))]
|
|
pub mod kernel_copy;
|
|
#[cfg(target_os = "l4re")]
|
|
mod l4re;
|
|
pub mod memchr;
|
|
pub mod mutex;
|
|
#[cfg(not(target_os = "l4re"))]
|
|
pub mod net;
|
|
#[cfg(target_os = "l4re")]
|
|
pub use self::l4re::net;
|
|
pub mod os;
|
|
pub mod path;
|
|
pub mod pipe;
|
|
pub mod process;
|
|
pub mod rand;
|
|
pub mod rwlock;
|
|
pub mod stack_overflow;
|
|
pub mod stdio;
|
|
pub mod thread;
|
|
pub mod thread_local_dtor;
|
|
pub mod thread_local_key;
|
|
pub mod time;
|
|
|
|
pub use crate::sys_common::os_str_bytes as os_str;
|
|
|
|
// SAFETY: must be called only once during runtime initialization.
|
|
// NOTE: this is not guaranteed to run, for example when Rust code is called externally.
|
|
pub unsafe fn init(argc: isize, argv: *const *const u8) {
|
|
// The standard streams might be closed on application startup. To prevent
|
|
// std::io::{stdin, stdout,stderr} objects from using other unrelated file
|
|
// resources opened later, we reopen standards streams when they are closed.
|
|
sanitize_standard_fds();
|
|
|
|
// By default, some platforms will send a *signal* when an EPIPE error
|
|
// would otherwise be delivered. This runtime doesn't install a SIGPIPE
|
|
// handler, causing it to kill the program, which isn't exactly what we
|
|
// want!
|
|
//
|
|
// Hence, we set SIGPIPE to ignore when the program starts up in order
|
|
// to prevent this problem.
|
|
reset_sigpipe();
|
|
|
|
stack_overflow::init();
|
|
args::init(argc, argv);
|
|
|
|
unsafe fn sanitize_standard_fds() {
|
|
#[cfg(not(miri))]
|
|
// The standard fds are always available in Miri.
|
|
cfg_if::cfg_if! {
|
|
if #[cfg(not(any(
|
|
target_os = "emscripten",
|
|
target_os = "fuchsia",
|
|
target_os = "vxworks",
|
|
// The poll on Darwin doesn't set POLLNVAL for closed fds.
|
|
target_os = "macos",
|
|
target_os = "ios",
|
|
target_os = "redox",
|
|
)))] {
|
|
use crate::sys::os::errno;
|
|
let pfds: &mut [_] = &mut [
|
|
libc::pollfd { fd: 0, events: 0, revents: 0 },
|
|
libc::pollfd { fd: 1, events: 0, revents: 0 },
|
|
libc::pollfd { fd: 2, events: 0, revents: 0 },
|
|
];
|
|
while libc::poll(pfds.as_mut_ptr(), 3, 0) == -1 {
|
|
if errno() == libc::EINTR {
|
|
continue;
|
|
}
|
|
libc::abort();
|
|
}
|
|
for pfd in pfds {
|
|
if pfd.revents & libc::POLLNVAL == 0 {
|
|
continue;
|
|
}
|
|
if libc::open("/dev/null\0".as_ptr().cast(), libc::O_RDWR, 0) == -1 {
|
|
// If the stream is closed but we failed to reopen it, abort the
|
|
// process. Otherwise we wouldn't preserve the safety of
|
|
// operations on the corresponding Rust object Stdin, Stdout, or
|
|
// Stderr.
|
|
libc::abort();
|
|
}
|
|
}
|
|
} else if #[cfg(any(target_os = "macos", target_os = "ios", target_os = "redox"))] {
|
|
use crate::sys::os::errno;
|
|
for fd in 0..3 {
|
|
if libc::fcntl(fd, libc::F_GETFD) == -1 && errno() == libc::EBADF {
|
|
if libc::open("/dev/null\0".as_ptr().cast(), libc::O_RDWR, 0) == -1 {
|
|
libc::abort();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
unsafe fn reset_sigpipe() {
|
|
#[cfg(not(any(target_os = "emscripten", target_os = "fuchsia")))]
|
|
assert!(signal(libc::SIGPIPE, libc::SIG_IGN) != libc::SIG_ERR);
|
|
}
|
|
}
|
|
|
|
// SAFETY: must be called only once during runtime cleanup.
|
|
// NOTE: this is not guaranteed to run, for example when the program aborts.
|
|
pub unsafe fn cleanup() {
|
|
args::cleanup();
|
|
stack_overflow::cleanup();
|
|
}
|
|
|
|
#[cfg(target_os = "android")]
|
|
pub use crate::sys::android::signal;
|
|
#[cfg(not(target_os = "android"))]
|
|
pub use libc::signal;
|
|
|
|
pub fn decode_error_kind(errno: i32) -> ErrorKind {
|
|
use ErrorKind::*;
|
|
match errno as libc::c_int {
|
|
libc::E2BIG => ArgumentListTooLong,
|
|
libc::EADDRINUSE => AddrInUse,
|
|
libc::EADDRNOTAVAIL => AddrNotAvailable,
|
|
libc::EBUSY => ResourceBusy,
|
|
libc::ECONNABORTED => ConnectionAborted,
|
|
libc::ECONNREFUSED => ConnectionRefused,
|
|
libc::ECONNRESET => ConnectionReset,
|
|
libc::EDEADLK => Deadlock,
|
|
libc::EDQUOT => FilesystemQuotaExceeded,
|
|
libc::EEXIST => AlreadyExists,
|
|
libc::EFBIG => FileTooLarge,
|
|
libc::EHOSTUNREACH => HostUnreachable,
|
|
libc::EINTR => Interrupted,
|
|
libc::EINVAL => InvalidInput,
|
|
libc::EISDIR => IsADirectory,
|
|
libc::ELOOP => FilesystemLoop,
|
|
libc::ENOENT => NotFound,
|
|
libc::ENOMEM => OutOfMemory,
|
|
libc::ENOSPC => StorageFull,
|
|
libc::ENOSYS => Unsupported,
|
|
libc::EMLINK => TooManyLinks,
|
|
libc::ENAMETOOLONG => FilenameTooLong,
|
|
libc::ENETDOWN => NetworkDown,
|
|
libc::ENETUNREACH => NetworkUnreachable,
|
|
libc::ENOTCONN => NotConnected,
|
|
libc::ENOTDIR => NotADirectory,
|
|
libc::ENOTEMPTY => DirectoryNotEmpty,
|
|
libc::EPIPE => BrokenPipe,
|
|
libc::EROFS => ReadOnlyFilesystem,
|
|
libc::ESPIPE => NotSeekable,
|
|
libc::ESTALE => StaleNetworkFileHandle,
|
|
libc::ETIMEDOUT => TimedOut,
|
|
libc::ETXTBSY => ExecutableFileBusy,
|
|
libc::EXDEV => CrossesDevices,
|
|
|
|
libc::EACCES | libc::EPERM => PermissionDenied,
|
|
|
|
// These two constants can have the same value on some systems,
|
|
// but different values on others, so we can't use a match
|
|
// clause
|
|
x if x == libc::EAGAIN || x == libc::EWOULDBLOCK => WouldBlock,
|
|
|
|
_ => Uncategorized,
|
|
}
|
|
}
|
|
|
|
#[doc(hidden)]
|
|
pub trait IsMinusOne {
|
|
fn is_minus_one(&self) -> bool;
|
|
}
|
|
|
|
macro_rules! impl_is_minus_one {
|
|
($($t:ident)*) => ($(impl IsMinusOne for $t {
|
|
fn is_minus_one(&self) -> bool {
|
|
*self == -1
|
|
}
|
|
})*)
|
|
}
|
|
|
|
impl_is_minus_one! { i8 i16 i32 i64 isize }
|
|
|
|
pub fn cvt<T: IsMinusOne>(t: T) -> crate::io::Result<T> {
|
|
if t.is_minus_one() { Err(crate::io::Error::last_os_error()) } else { Ok(t) }
|
|
}
|
|
|
|
pub fn cvt_r<T, F>(mut f: F) -> crate::io::Result<T>
|
|
where
|
|
T: IsMinusOne,
|
|
F: FnMut() -> T,
|
|
{
|
|
loop {
|
|
match cvt(f()) {
|
|
Err(ref e) if e.kind() == ErrorKind::Interrupted => {}
|
|
other => return other,
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn cvt_nz(error: libc::c_int) -> crate::io::Result<()> {
|
|
if error == 0 { Ok(()) } else { Err(crate::io::Error::from_raw_os_error(error)) }
|
|
}
|
|
|
|
// On Unix-like platforms, libc::abort will unregister signal handlers
|
|
// including the SIGABRT handler, preventing the abort from being blocked, and
|
|
// fclose streams, with the side effect of flushing them so libc buffered
|
|
// output will be printed. Additionally the shell will generally print a more
|
|
// understandable error message like "Abort trap" rather than "Illegal
|
|
// instruction" that intrinsics::abort would cause, as intrinsics::abort is
|
|
// implemented as an illegal instruction.
|
|
pub fn abort_internal() -> ! {
|
|
unsafe { libc::abort() }
|
|
}
|
|
|
|
cfg_if::cfg_if! {
|
|
if #[cfg(target_os = "android")] {
|
|
#[link(name = "dl")]
|
|
#[link(name = "log")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "freebsd")] {
|
|
#[link(name = "execinfo")]
|
|
#[link(name = "pthread")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "netbsd")] {
|
|
#[link(name = "pthread")]
|
|
#[link(name = "rt")]
|
|
extern "C" {}
|
|
} else if #[cfg(any(target_os = "dragonfly", target_os = "openbsd"))] {
|
|
#[link(name = "pthread")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "solaris")] {
|
|
#[link(name = "socket")]
|
|
#[link(name = "posix4")]
|
|
#[link(name = "pthread")]
|
|
#[link(name = "resolv")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "illumos")] {
|
|
#[link(name = "socket")]
|
|
#[link(name = "posix4")]
|
|
#[link(name = "pthread")]
|
|
#[link(name = "resolv")]
|
|
#[link(name = "nsl")]
|
|
// Use libumem for the (malloc-compatible) allocator
|
|
#[link(name = "umem")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "macos")] {
|
|
#[link(name = "System")]
|
|
// res_init and friends require -lresolv on macOS/iOS.
|
|
// See #41582 and https://blog.achernya.com/2013/03/os-x-has-silly-libsystem.html
|
|
#[link(name = "resolv")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "ios")] {
|
|
#[link(name = "System")]
|
|
#[link(name = "objc")]
|
|
#[link(name = "Security", kind = "framework")]
|
|
#[link(name = "Foundation", kind = "framework")]
|
|
#[link(name = "resolv")]
|
|
extern "C" {}
|
|
} else if #[cfg(target_os = "fuchsia")] {
|
|
#[link(name = "zircon")]
|
|
#[link(name = "fdio")]
|
|
extern "C" {}
|
|
}
|
|
}
|