mirror of
https://github.com/topjohnwu/Magisk.git
synced 2025-04-29 22:14:27 +02:00
406 lines
10 KiB
Rust
406 lines
10 KiB
Rust
use crate::logging::Formatter;
|
|
use crate::{LogLevel, errno, log_with_args, log_with_formatter};
|
|
use std::fmt::Display;
|
|
use std::panic::Location;
|
|
use std::ptr::NonNull;
|
|
use std::{fmt, io};
|
|
use thiserror::Error;
|
|
|
|
// Error handling throughout the Rust codebase in Magisk:
|
|
//
|
|
// All errors should be logged and consumed as soon as possible and converted into LoggedError.
|
|
// For `Result` with errors that implement the `Display` trait, use the `?` operator to
|
|
// log and convert to LoggedResult.
|
|
//
|
|
// To log an error with more information, use `ResultExt::log_with_msg()`.
|
|
//
|
|
// The "cxx" method variants in `CxxResultExt` are only used for C++ interop and
|
|
// should not be used directly in any Rust code.
|
|
|
|
#[derive(Default)]
|
|
pub struct LoggedError {}
|
|
pub type LoggedResult<T> = Result<T, LoggedError>;
|
|
|
|
#[macro_export]
|
|
macro_rules! log_err {
|
|
($($args:tt)+) => {{
|
|
$crate::log_with_args($crate::LogLevel::Error, format_args_nl!($($args)+));
|
|
$crate::LoggedError::default()
|
|
}};
|
|
}
|
|
|
|
// Any result or option can be silenced
|
|
pub trait SilentResultExt<T> {
|
|
fn silent(self) -> LoggedResult<T>;
|
|
}
|
|
|
|
impl<T, E> SilentResultExt<T> for Result<T, E> {
|
|
fn silent(self) -> LoggedResult<T> {
|
|
match self {
|
|
Ok(v) => Ok(v),
|
|
Err(_) => Err(LoggedError::default()),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T> SilentResultExt<T> for Option<T> {
|
|
fn silent(self) -> LoggedResult<T> {
|
|
match self {
|
|
Some(v) => Ok(v),
|
|
None => Err(LoggedError::default()),
|
|
}
|
|
}
|
|
}
|
|
|
|
// Public API for logging results
|
|
pub trait ResultExt<T> {
|
|
fn log(self) -> LoggedResult<T>;
|
|
fn log_with_msg<F: FnOnce(Formatter) -> fmt::Result>(self, f: F) -> LoggedResult<T>;
|
|
fn log_ok(self);
|
|
}
|
|
|
|
// Internal C++ bridging logging routines
|
|
pub(crate) trait CxxResultExt<T> {
|
|
fn log_cxx(self) -> LoggedResult<T>;
|
|
}
|
|
|
|
trait Loggable<T> {
|
|
fn do_log(self, level: LogLevel, caller: Option<&'static Location>) -> LoggedResult<T>;
|
|
fn do_log_msg<F: FnOnce(Formatter) -> fmt::Result>(
|
|
self,
|
|
level: LogLevel,
|
|
caller: Option<&'static Location>,
|
|
f: F,
|
|
) -> LoggedResult<T>;
|
|
}
|
|
|
|
impl<T, R: Loggable<T>> CxxResultExt<T> for R {
|
|
fn log_cxx(self) -> LoggedResult<T> {
|
|
self.do_log(LogLevel::ErrorCxx, None)
|
|
}
|
|
}
|
|
|
|
impl<T, R: Loggable<T>> ResultExt<T> for R {
|
|
#[cfg(not(debug_assertions))]
|
|
fn log(self) -> LoggedResult<T> {
|
|
self.do_log(LogLevel::Error, None)
|
|
}
|
|
|
|
#[track_caller]
|
|
#[cfg(debug_assertions)]
|
|
fn log(self) -> LoggedResult<T> {
|
|
self.do_log(LogLevel::Error, Some(Location::caller()))
|
|
}
|
|
|
|
#[cfg(not(debug_assertions))]
|
|
fn log_with_msg<F: FnOnce(Formatter) -> fmt::Result>(self, f: F) -> LoggedResult<T> {
|
|
self.do_log_msg(LogLevel::Error, None, f)
|
|
}
|
|
|
|
#[track_caller]
|
|
#[cfg(debug_assertions)]
|
|
fn log_with_msg<F: FnOnce(Formatter) -> fmt::Result>(self, f: F) -> LoggedResult<T> {
|
|
self.do_log_msg(LogLevel::Error, Some(Location::caller()), f)
|
|
}
|
|
|
|
#[cfg(not(debug_assertions))]
|
|
fn log_ok(self) {
|
|
self.log().ok();
|
|
}
|
|
|
|
#[track_caller]
|
|
#[cfg(debug_assertions)]
|
|
fn log_ok(self) {
|
|
self.do_log(LogLevel::Error, Some(Location::caller())).ok();
|
|
}
|
|
}
|
|
|
|
impl<T> Loggable<T> for LoggedResult<T> {
|
|
fn do_log(self, _: LogLevel, _: Option<&'static Location>) -> LoggedResult<T> {
|
|
self
|
|
}
|
|
|
|
fn do_log_msg<F: FnOnce(Formatter) -> fmt::Result>(
|
|
self,
|
|
level: LogLevel,
|
|
caller: Option<&'static Location>,
|
|
f: F,
|
|
) -> LoggedResult<T> {
|
|
match self {
|
|
Ok(v) => Ok(v),
|
|
Err(_) => {
|
|
log_with_formatter(level, |w| {
|
|
if let Some(caller) = caller {
|
|
write!(w, "[{}:{}] ", caller.file(), caller.line())?;
|
|
}
|
|
f(w)?;
|
|
w.write_char('\n')
|
|
});
|
|
Err(LoggedError::default())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T, E: Display> Loggable<T> for Result<T, E> {
|
|
fn do_log(self, level: LogLevel, caller: Option<&'static Location>) -> LoggedResult<T> {
|
|
match self {
|
|
Ok(v) => Ok(v),
|
|
Err(e) => {
|
|
if let Some(caller) = caller {
|
|
log_with_args(
|
|
level,
|
|
format_args_nl!("[{}:{}] {:#}", caller.file(), caller.line(), e),
|
|
);
|
|
} else {
|
|
log_with_args(level, format_args_nl!("{:#}", e));
|
|
}
|
|
Err(LoggedError::default())
|
|
}
|
|
}
|
|
}
|
|
|
|
fn do_log_msg<F: FnOnce(Formatter) -> fmt::Result>(
|
|
self,
|
|
level: LogLevel,
|
|
caller: Option<&'static Location>,
|
|
f: F,
|
|
) -> LoggedResult<T> {
|
|
match self {
|
|
Ok(v) => Ok(v),
|
|
Err(e) => {
|
|
log_with_formatter(level, |w| {
|
|
if let Some(caller) = caller {
|
|
write!(w, "[{}:{}] ", caller.file(), caller.line())?;
|
|
}
|
|
f(w)?;
|
|
writeln!(w, ": {:#}", e)
|
|
});
|
|
Err(LoggedError::default())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Automatically convert all printable errors to LoggedError to support `?` operator
|
|
impl<T: Display> From<T> for LoggedError {
|
|
#[cfg(not(debug_assertions))]
|
|
fn from(e: T) -> Self {
|
|
log_with_args(LogLevel::Error, format_args_nl!("{:#}", e));
|
|
LoggedError::default()
|
|
}
|
|
|
|
#[track_caller]
|
|
#[cfg(debug_assertions)]
|
|
fn from(e: T) -> Self {
|
|
let caller = Location::caller();
|
|
log_with_args(
|
|
LogLevel::Error,
|
|
format_args_nl!("[{}:{}] {:#}", caller.file(), caller.line(), e),
|
|
);
|
|
LoggedError::default()
|
|
}
|
|
}
|
|
|
|
// Check libc return value and map to Result
|
|
pub trait LibcReturn
|
|
where
|
|
Self: Copy,
|
|
{
|
|
type Value;
|
|
|
|
fn is_error(&self) -> bool;
|
|
fn map_val(self) -> Self::Value;
|
|
|
|
fn as_os_result<'a>(
|
|
self,
|
|
name: &'static str,
|
|
arg1: Option<&'a str>,
|
|
arg2: Option<&'a str>,
|
|
) -> OsResult<'a, Self::Value> {
|
|
if self.is_error() {
|
|
Err(OsError::last_os_error(name, arg1, arg2))
|
|
} else {
|
|
Ok(self.map_val())
|
|
}
|
|
}
|
|
|
|
fn check_os_err<'a>(
|
|
self,
|
|
name: &'static str,
|
|
arg1: Option<&'a str>,
|
|
arg2: Option<&'a str>,
|
|
) -> OsResult<'a, ()> {
|
|
if self.is_error() {
|
|
Err(OsError::last_os_error(name, arg1, arg2))
|
|
} else {
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
fn check_io_err(self) -> io::Result<()> {
|
|
if self.is_error() {
|
|
Err(io::Error::last_os_error())
|
|
} else {
|
|
Ok(())
|
|
}
|
|
}
|
|
}
|
|
|
|
macro_rules! impl_libc_return {
|
|
($($t:ty)*) => ($(
|
|
impl LibcReturn for $t {
|
|
type Value = Self;
|
|
|
|
#[inline(always)]
|
|
fn is_error(&self) -> bool {
|
|
*self < 0
|
|
}
|
|
|
|
#[inline(always)]
|
|
fn map_val(self) -> Self::Value {
|
|
self
|
|
}
|
|
}
|
|
)*)
|
|
}
|
|
|
|
impl_libc_return! { i8 i16 i32 i64 isize }
|
|
|
|
impl<T> LibcReturn for *mut T {
|
|
type Value = NonNull<T>;
|
|
|
|
#[inline(always)]
|
|
fn is_error(&self) -> bool {
|
|
self.is_null()
|
|
}
|
|
|
|
#[inline(always)]
|
|
fn map_val(self) -> NonNull<T> {
|
|
// SAFETY: pointer is null checked by is_error
|
|
unsafe { NonNull::new_unchecked(self.cast()) }
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
enum OwnableStr<'a> {
|
|
None,
|
|
Borrowed(&'a str),
|
|
Owned(Box<str>),
|
|
}
|
|
|
|
impl OwnableStr<'_> {
|
|
fn into_owned(self) -> OwnableStr<'static> {
|
|
match self {
|
|
OwnableStr::None => OwnableStr::None,
|
|
OwnableStr::Borrowed(s) => OwnableStr::Owned(Box::from(s)),
|
|
OwnableStr::Owned(s) => OwnableStr::Owned(s),
|
|
}
|
|
}
|
|
|
|
fn ok(&self) -> Option<&str> {
|
|
match self {
|
|
OwnableStr::None => None,
|
|
OwnableStr::Borrowed(s) => Some(*s),
|
|
OwnableStr::Owned(s) => Some(s),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> From<Option<&'a str>> for OwnableStr<'a> {
|
|
fn from(value: Option<&'a str>) -> Self {
|
|
value.map(OwnableStr::Borrowed).unwrap_or(OwnableStr::None)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub struct OsError<'a> {
|
|
code: i32,
|
|
name: &'static str,
|
|
arg1: OwnableStr<'a>,
|
|
arg2: OwnableStr<'a>,
|
|
}
|
|
|
|
impl OsError<'_> {
|
|
pub fn with_os_error<'a>(
|
|
code: i32,
|
|
name: &'static str,
|
|
arg1: Option<&'a str>,
|
|
arg2: Option<&'a str>,
|
|
) -> OsError<'a> {
|
|
OsError {
|
|
code,
|
|
name,
|
|
arg1: OwnableStr::from(arg1),
|
|
arg2: OwnableStr::from(arg2),
|
|
}
|
|
}
|
|
|
|
pub fn last_os_error<'a>(
|
|
name: &'static str,
|
|
arg1: Option<&'a str>,
|
|
arg2: Option<&'a str>,
|
|
) -> OsError<'a> {
|
|
Self::with_os_error(*errno(), name, arg1, arg2)
|
|
}
|
|
|
|
pub fn set_args<'a>(self, arg1: Option<&'a str>, arg2: Option<&'a str>) -> OsError<'a> {
|
|
Self::with_os_error(self.code, self.name, arg1, arg2)
|
|
}
|
|
|
|
pub fn into_owned(self) -> OsError<'static> {
|
|
OsError {
|
|
code: *errno(),
|
|
name: self.name,
|
|
arg1: self.arg1.into_owned(),
|
|
arg2: self.arg2.into_owned(),
|
|
}
|
|
}
|
|
|
|
fn as_io_error(&self) -> io::Error {
|
|
io::Error::from_raw_os_error(self.code)
|
|
}
|
|
}
|
|
|
|
impl Display for OsError<'_> {
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
let error = self.as_io_error();
|
|
if self.name.is_empty() {
|
|
write!(f, "{:#}", error)
|
|
} else {
|
|
match (self.arg1.ok(), self.arg2.ok()) {
|
|
(Some(arg1), Some(arg2)) => {
|
|
write!(f, "{} '{}' '{}': {:#}", self.name, arg1, arg2, error)
|
|
}
|
|
(Some(arg1), None) => {
|
|
write!(f, "{} '{}': {:#}", self.name, arg1, error)
|
|
}
|
|
_ => {
|
|
write!(f, "{}: {:#}", self.name, error)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::error::Error for OsError<'_> {}
|
|
|
|
pub type OsResult<'a, T> = Result<T, OsError<'a>>;
|
|
|
|
#[derive(Debug, Error)]
|
|
pub enum OsErrorStatic {
|
|
#[error(transparent)]
|
|
Os(OsError<'static>),
|
|
#[error(transparent)]
|
|
Io(#[from] io::Error),
|
|
}
|
|
|
|
// Convert non-static OsError to static
|
|
impl<'a> From<OsError<'a>> for OsErrorStatic {
|
|
fn from(value: OsError<'a>) -> Self {
|
|
OsErrorStatic::Os(value.into_owned())
|
|
}
|
|
}
|
|
|
|
pub type OsResultStatic<T> = Result<T, OsErrorStatic>;
|