//! Result and error types representing the outcome of compiling a function. use regalloc2::checker::CheckerErrors; use crate::ir::pcc::PccError; use crate::{ir::Function, verifier::VerifierErrors}; use std::string::String; /// A compilation error. /// /// When Cranelift fails to compile a function, it will return one of these error codes. #[derive(Debug)] pub enum CodegenError { /// A list of IR verifier errors. /// /// This always represents a bug, either in the code that generated IR for Cranelift, or a bug /// in Cranelift itself. Verifier(VerifierErrors), /// An implementation limit was exceeded. /// /// Cranelift can compile very large and complicated functions, but the [implementation has /// limits][limits] that cause compilation to fail when they are exceeded. /// /// [limits]: https://github.com/bytecodealliance/wasmtime/blob/main/cranelift/docs/ir.md#implementation-limits ImplLimitExceeded, /// The code size for the function is too large. /// /// Different target ISAs may impose a limit on the size of a compiled function. If that limit /// is exceeded, compilation fails. CodeTooLarge, /// Something is not supported by the code generator. This might be an indication that a /// feature is used without explicitly enabling it, or that something is temporarily /// unsupported by a given target backend. Unsupported(String), /// A failure to map Cranelift register representation to a DWARF register representation. #[cfg(feature = "unwind")] RegisterMappingError(crate::isa::unwind::systemv::RegisterMappingError), /// Register allocator internal error discovered by the symbolic checker. Regalloc(CheckerErrors), /// Proof-carrying-code validation error. Pcc(PccError), } /// A convenient alias for a `Result` that uses `CodegenError` as the error type. pub type CodegenResult = Result; // This is manually implementing Error and Display instead of using thiserror to reduce the amount // of dependencies used by Cranelift. impl std::error::Error for CodegenError { fn source(&self) -> Option<&(dyn std::error::Error + 'static)> { match self { CodegenError::Verifier(source) => Some(source), CodegenError::ImplLimitExceeded { .. } | CodegenError::CodeTooLarge { .. } | CodegenError::Unsupported { .. } => None, #[cfg(feature = "unwind")] CodegenError::RegisterMappingError { .. } => None, CodegenError::Regalloc(..) => None, CodegenError::Pcc(..) => None, } } } impl std::fmt::Display for CodegenError { fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { match self { CodegenError::Verifier(_) => write!(f, "Verifier errors"), CodegenError::ImplLimitExceeded => write!(f, "Implementation limit exceeded"), CodegenError::CodeTooLarge => write!(f, "Code for function is too large"), CodegenError::Unsupported(feature) => write!(f, "Unsupported feature: {feature}"), #[cfg(feature = "unwind")] CodegenError::RegisterMappingError(_0) => write!(f, "Register mapping error"), CodegenError::Regalloc(errors) => write!(f, "Regalloc validation errors: {errors:?}"), // NOTE: if this is changed, please update the `is_pcc_error` function defined in // `wasmtime/crates/fuzzing/src/oracles.rs` CodegenError::Pcc(e) => write!(f, "Proof-carrying-code validation error: {e:?}"), } } } impl From for CodegenError { fn from(source: VerifierErrors) -> Self { CodegenError::Verifier { 0: source } } } /// Compilation error, with the accompanying function to help printing it. pub struct CompileError<'a> { /// Underlying `CodegenError` that triggered the error. pub inner: CodegenError, /// Function we tried to compile, for display purposes. pub func: &'a Function, } // By default, have `CompileError` be displayed as the internal error, and let consumers care if // they want to use the func field for adding details. impl<'a> core::fmt::Debug for CompileError<'a> { fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { self.inner.fmt(f) } } /// A convenient alias for a `Result` that uses `CompileError` as the error type. pub type CompileResult<'a, T> = Result>;