1use crate::{ExternType, Pages, progress::UserAbort};
3use std::io;
4use thiserror::Error;
5
6#[derive(Error, Debug)]
9pub enum SerializeError {
10 #[error(transparent)]
12 Io(#[from] io::Error),
13 #[error("{0}")]
15 Generic(String),
16}
17
18#[derive(Error, Debug)]
21pub enum DeserializeError {
22 #[error(transparent)]
24 Io(#[from] io::Error),
25 #[error("{0}")]
27 Generic(String),
28 #[error("incompatible binary: {0}")]
30 Incompatible(String),
31 #[error("corrupted binary: {0}")]
33 CorruptedBinary(String),
34 #[error(transparent)]
37 Compiler(#[from] CompileError),
38 #[error("invalid input bytes: expected {expected} bytes, got {got}")]
40 InvalidByteLength {
41 expected: usize,
43 got: usize,
45 },
46}
47
48#[derive(Error, Debug, Clone, PartialEq, Eq, Hash)]
50#[non_exhaustive]
51pub enum MemoryError {
52 #[error("Error when allocating memory: {0}")]
54 Region(String),
55 #[error("The memory could not grow: current size {} pages, requested increase: {} pages", current.0, attempted_delta.0)]
58 CouldNotGrow {
59 current: Pages,
61 attempted_delta: Pages,
63 },
64 #[error("The memory is invalid because {}", reason)]
66 InvalidMemory {
67 reason: String,
69 },
70 #[error("The minimum requested ({} pages) memory is greater than the maximum allowed memory ({} pages)", min_requested.0, max_allowed.0)]
72 MinimumMemoryTooLarge {
73 min_requested: Pages,
75 max_allowed: Pages,
77 },
78 #[error("The maximum requested memory ({} pages) is greater than the maximum allowed memory ({} pages)", max_requested.0, max_allowed.0)]
80 MaximumMemoryTooLarge {
81 max_requested: Pages,
83 max_allowed: Pages,
85 },
86 #[error("The memory is not shared")]
88 MemoryNotShared,
89 #[error("tried to call an unsupported memory operation: {message}")]
92 UnsupportedOperation {
93 message: String,
95 },
96 #[error("Atomic operation failed: {0}")]
98 AtomicOperationFailed(AtomicsError),
99 #[error("A user-defined error occurred: {0}")]
101 Generic(String),
102}
103
104#[derive(PartialEq, Eq, Debug, Error, Clone, Copy, Hash)]
107#[non_exhaustive]
108pub enum AtomicsError {
109 #[error("The memory does not support atomic operations")]
111 Unimplemented,
112 #[error("Too many waiters for address")]
114 TooManyWaiters,
115 #[error("Atomic operations are disabled for this memory")]
117 AtomicsDisabled,
118 #[error("The memory was already dropped")]
120 MemoryDropped,
121}
122
123#[derive(Error, Debug, Clone)]
128pub enum ImportError {
129 #[error("incompatible import type. Expected {0:?} but received {1:?}")]
132 IncompatibleType(ExternType, ExternType),
133
134 #[error("unknown import. Expected {0:?}")]
137 UnknownImport(ExternType),
138
139 #[error("memory error. {0}")]
141 MemoryError(String),
142}
143
144#[derive(Error, Debug)]
147pub enum PreInstantiationError {
148 #[error("module compiled with CPU feature that is missing from host")]
151 CpuFeature(String),
152}
153
154use crate::lib::std::string::String;
155
156#[derive(Debug)]
167#[cfg_attr(feature = "std", derive(Error))]
168pub enum CompileError {
169 #[cfg_attr(feature = "std", error("WebAssembly translation error: {0}"))]
171 Wasm(WasmError),
172
173 #[cfg_attr(feature = "std", error("Compilation error: {0}"))]
175 Codegen(String),
176
177 #[cfg_attr(feature = "std", error("Validation error: {0}"))]
179 Validate(String),
180
181 #[cfg_attr(feature = "std", error("Feature {0} is not yet supported"))]
183 UnsupportedFeature(String),
184
185 #[cfg_attr(
188 feature = "std",
189 error("The target {0} is not yet supported (see https://docs.wasmer.io/runtime/features)")
190 )]
191 UnsupportedTarget(String),
192
193 #[cfg_attr(feature = "std", error("Insufficient resources: {0}"))]
195 Resource(String),
196
197 #[cfg_attr(feature = "std", error("Middleware error: {0}"))]
199 MiddlewareError(String),
200
201 #[cfg_attr(feature = "std", error("Compilation aborted: {0}"))]
203 Aborted(UserAbort),
204}
205
206impl From<WasmError> for CompileError {
207 fn from(original: WasmError) -> Self {
208 Self::Wasm(original)
209 }
210}
211
212impl From<UserAbort> for CompileError {
213 fn from(abort: UserAbort) -> Self {
214 Self::Aborted(abort)
215 }
216}
217
218#[derive(Debug)]
220#[cfg_attr(feature = "std", derive(Error))]
221#[cfg_attr(feature = "std", error("Error in middleware {name}: {message}"))]
222pub struct MiddlewareError {
223 pub name: String,
225 pub message: String,
227}
228
229impl MiddlewareError {
230 pub fn new<A: Into<String>, B: Into<String>>(name: A, message: B) -> Self {
232 Self {
233 name: name.into(),
234 message: message.into(),
235 }
236 }
237}
238
239impl From<MiddlewareError> for CompileError {
240 fn from(error: MiddlewareError) -> Self {
241 WasmError::Middleware(error).into()
242 }
243}
244
245#[derive(Debug)]
250#[cfg_attr(feature = "std", derive(Error))]
251pub enum WasmError {
252 #[cfg_attr(
257 feature = "std",
258 error("Invalid input WebAssembly code at offset {offset}: {message}")
259 )]
260 InvalidWebAssembly {
261 message: String,
263 offset: usize,
265 },
266
267 #[cfg_attr(feature = "std", error("Unsupported feature: {0}"))]
271 Unsupported(String),
272
273 #[cfg_attr(feature = "std", error("Implementation limit exceeded"))]
275 ImplLimitExceeded,
276
277 #[cfg_attr(feature = "std", error("{0}"))]
279 Middleware(MiddlewareError),
280
281 #[cfg_attr(feature = "std", error("{0}"))]
283 Generic(String),
284}
285
286impl From<MiddlewareError> for WasmError {
287 fn from(original: MiddlewareError) -> Self {
288 Self::Middleware(original)
289 }
290}
291
292#[derive(Debug)]
295#[cfg_attr(feature = "std", derive(Error))]
296pub enum ParseCpuFeatureError {
297 #[cfg_attr(feature = "std", error("CpuFeature {0} not recognized"))]
299 Missing(String),
300}
301
302pub type WasmResult<T> = Result<T, WasmError>;
304
305#[cfg(test)]
306mod tests {
307 use super::*;
308
309 #[test]
310 fn middleware_error_can_be_created() {
311 let msg = String::from("Something went wrong");
312 let error = MiddlewareError::new("manipulator3000", msg);
313 assert_eq!(error.name, "manipulator3000");
314 assert_eq!(error.message, "Something went wrong");
315 }
316
317 #[test]
318 fn middleware_error_be_converted_to_wasm_error() {
319 let error = WasmError::from(MiddlewareError::new("manipulator3000", "foo"));
320 match error {
321 WasmError::Middleware(MiddlewareError { name, message }) => {
322 assert_eq!(name, "manipulator3000");
323 assert_eq!(message, "foo");
324 }
325 err => panic!("Unexpected error: {err:?}"),
326 }
327 }
328}