1use std::array::TryFromSliceError;
18use std::cell::{BorrowError, BorrowMutError};
19use std::convert::Infallible;
20use std::error::Error;
21use std::num::TryFromIntError;
22use std::string::FromUtf8Error;
23use std::sync::{MutexGuard, PoisonError};
24use std::time::SystemTimeError;
25
26#[cfg(target_os = "windows")]
27use crossbeam_channel::{RecvError, SendError};
28use flatbuffers::InvalidFlatbuffer;
29use hyperlight_common::flatbuffer_wrappers::function_types::{ParameterValue, ReturnValue};
30use hyperlight_common::flatbuffer_wrappers::guest_error::ErrorCode;
31use thiserror::Error;
32
33use crate::hypervisor::hyperlight_vm::HyperlightVmError;
34#[cfg(target_os = "windows")]
35use crate::hypervisor::wrappers::HandleWrapper;
36use crate::mem::memory_region::MemoryRegionFlags;
37use crate::mem::ptr::RawPtr;
38
39#[derive(Error, Debug)]
41pub enum HyperlightError {
42 #[error("Anyhow Error was returned: {0}")]
44 AnyhowError(#[from] anyhow::Error),
45 #[error("Offset: {0} out of bounds, Max is: {1}")]
47 BoundsCheckFailed(u64, usize),
48
49 #[error("Couldn't add offset to base address. Offset: {0}, Base Address: {1}")]
51 CheckedAddOverflow(u64, u64),
52
53 #[error("{0:?}")]
55 #[cfg(target_os = "windows")]
56 CrossBeamReceiveError(#[from] RecvError),
57
58 #[error("{0:?}")]
60 #[cfg(target_os = "windows")]
61 CrossBeamSendError(#[from] SendError<HandleWrapper>),
62
63 #[error("Error converting CString {0:?}")]
65 CStringConversionError(#[from] std::ffi::NulError),
66
67 #[error("{0}")]
69 Error(String),
70
71 #[error("Non-executable address {0:#x} tried to be executed")]
73 ExecutionAccessViolation(u64),
74
75 #[error("Execution was cancelled by the host.")]
77 ExecutionCanceledByHost(),
78
79 #[error("Failed to get a value from flat buffer parameter")]
81 FailedToGetValueFromParameter(),
82
83 #[error("Field Name {0} not found in decoded GuestLogData")]
85 FieldIsMissingInGuestLogData(String),
86
87 #[error("Guest aborted: {0} {1}")]
89 GuestAborted(u8, String),
90
91 #[error("Guest error occurred {0:?}: {1}")]
93 GuestError(ErrorCode, String),
94
95 #[error("Guest execution hung on the execution of a host function call")]
97 GuestExecutionHungOnHostFunctionCall(),
98
99 #[error("Guest call is already in progress")]
101 GuestFunctionCallAlreadyInProgress(),
102
103 #[error("Unsupported type: {0}")]
105 GuestInterfaceUnsupportedType(String),
106
107 #[error(
111 "Guest binary was built with hyperlight-guest-bin {guest_bin_version}, \
112 but the host is running hyperlight {host_version}"
113 )]
114 GuestBinVersionMismatch {
115 guest_bin_version: String,
117 host_version: String,
119 },
120
121 #[error("HostFunction {0} was not found")]
123 HostFunctionNotFound(String),
124
125 #[doc(hidden)]
130 #[error("Internal Hyperlight VM error: {0}")]
131 HyperlightVmError(#[from] HyperlightVmError),
132
133 #[error("Reading Writing or Seeking data failed {0:?}")]
135 IOError(#[from] std::io::Error),
136
137 #[error("Failed To Convert Size to usize")]
139 IntConversionFailure(#[from] TryFromIntError),
140
141 #[error("The flatbuffer is invalid")]
143 InvalidFlatBuffer(#[from] InvalidFlatbuffer),
144
145 #[error("Conversion of str data to json failed")]
147 JsonConversionFailure(#[from] serde_json::Error),
148
149 #[error("Unable to lock resource")]
151 LockAttemptFailed(String),
152
153 #[error("Memory Access Violation at address {0:#x} of type {1}, but memory is marked as {2}")]
155 MemoryAccessViolation(u64, MemoryRegionFlags, MemoryRegionFlags),
156
157 #[error("Memory Allocation Failed with OS Error {0:?}.")]
159 MemoryAllocationFailed(Option<i32>),
160
161 #[error("Memory Protection Failed with OS Error {0:?}.")]
163 MemoryProtectionFailed(Option<i32>),
164
165 #[error("Memory region size mismatch: host size {0:?}, guest size {1:?} region {2:?}")]
167 MemoryRegionSizeMismatch(usize, usize, String),
168
169 #[error("Memory requested {0} exceeds maximum size allowed {1}")]
171 MemoryRequestTooBig(usize, usize),
172
173 #[error("Memory requested {0} is less than the minimum size allowed {1}")]
176 MemoryRequestTooSmall(usize, usize),
177
178 #[error("Metric Not Found {0:?}.")]
180 MetricNotFound(&'static str),
181
182 #[error("mmap failed with os error {0:?}")]
184 MmapFailed(Option<i32>),
185
186 #[error("mprotect failed with os error {0:?}")]
188 MprotectFailed(Option<i32>),
189
190 #[error("No Hypervisor was found for Sandbox")]
192 NoHypervisorFound(),
193
194 #[error("Restore_state called with no valid snapshot")]
196 NoMemorySnapshot,
197
198 #[error("Failed To Convert Parameter Value {0:?} to {1:?}")]
200 ParameterValueConversionFailure(ParameterValue, &'static str),
201
202 #[error("Failure processing PE File {0:?}")]
204 PEFileProcessingFailure(#[from] goblin::error::Error),
205
206 #[error("The sandbox was poisoned")]
225 PoisonedSandbox,
226
227 #[error("Raw pointer ({0:?}) was less than the base address ({1})")]
229 RawPointerLessThanBaseAddress(RawPtr, u64),
230
231 #[error("RefCell borrow failed")]
233 RefCellBorrowFailed(#[from] BorrowError),
234
235 #[error("RefCell mut borrow failed")]
237 RefCellMutBorrowFailed(#[from] BorrowMutError),
238
239 #[error("Failed To Convert Return Value {0:?} to {1:?}")]
241 ReturnValueConversionFailure(ReturnValue, &'static str),
242
243 #[error("Snapshot was taken from a different sandbox")]
245 SnapshotSandboxMismatch,
246
247 #[error("SystemTimeError {0:?}")]
249 SystemTimeError(#[from] SystemTimeError),
250
251 #[error("TryFromSliceError {0:?}")]
253 TryFromSliceError(#[from] TryFromSliceError),
254
255 #[error("The number of arguments to the function is wrong: got {0:?} expected {1:?}")]
257 UnexpectedNoOfArguments(usize, usize),
258
259 #[error("The parameter value type is unexpected got {0:?} expected {1:?}")]
261 UnexpectedParameterValueType(ParameterValue, String),
262
263 #[error("The return value type is unexpected got {0:?} expected {1:?}")]
265 UnexpectedReturnValueType(ReturnValue, String),
266
267 #[error("String Conversion of UTF8 data to str failed")]
269 UTF8StringConversionFailure(#[from] FromUtf8Error),
270
271 #[error(
273 "The capacity of the vector is incorrect. Capacity: {0}, Length: {1}, FlatBuffer Size: {2}"
274 )]
275 VectorCapacityIncorrect(usize, usize, i32),
276
277 #[error("vmm sys Error {0:?}")]
279 #[cfg(target_os = "linux")]
280 VmmSysError(vmm_sys_util::errno::Error),
281
282 #[cfg(target_os = "windows")]
284 #[error("Windows API Error Result {0:?}")]
285 WindowsAPIError(#[from] windows_result::Error),
286}
287
288impl From<Infallible> for HyperlightError {
289 fn from(_: Infallible) -> Self {
290 "Impossible as this is an infallible error".into()
291 }
292}
293
294impl From<&str> for HyperlightError {
295 fn from(s: &str) -> Self {
296 HyperlightError::Error(s.to_string())
297 }
298}
299
300impl<T> From<PoisonError<MutexGuard<'_, T>>> for HyperlightError {
301 fn from(e: PoisonError<MutexGuard<'_, T>>) -> Self {
305 let source = match e.source() {
306 Some(s) => s.to_string(),
307 None => String::from(""),
308 };
309 HyperlightError::LockAttemptFailed(source)
310 }
311}
312
313impl HyperlightError {
314 pub(crate) fn is_poison_error(&self) -> bool {
324 match self {
327 HyperlightError::GuestAborted(_, _)
330 | HyperlightError::ExecutionCanceledByHost()
331 | HyperlightError::PoisonedSandbox
332 | HyperlightError::ExecutionAccessViolation(_)
333 | HyperlightError::MemoryAccessViolation(_, _, _)
334 | HyperlightError::MemoryRegionSizeMismatch(_, _, _)
335 | HyperlightError::HyperlightVmError(HyperlightVmError::Restore(_)) => true,
338
339 HyperlightError::HyperlightVmError(HyperlightVmError::UpdateRegion(_))
343 | HyperlightError::HyperlightVmError(HyperlightVmError::AccessPageTable(_)) => true,
344
345 HyperlightError::HyperlightVmError(HyperlightVmError::DispatchGuestCall(e)) => {
347 e.is_poison_error()
348 }
349
350 HyperlightError::AnyhowError(_)
352 | HyperlightError::BoundsCheckFailed(_, _)
353 | HyperlightError::CheckedAddOverflow(_, _)
354 | HyperlightError::CStringConversionError(_)
355 | HyperlightError::Error(_)
356 | HyperlightError::FailedToGetValueFromParameter()
357 | HyperlightError::FieldIsMissingInGuestLogData(_)
358 | HyperlightError::GuestBinVersionMismatch { .. }
359 | HyperlightError::GuestError(_, _)
360 | HyperlightError::GuestExecutionHungOnHostFunctionCall()
361 | HyperlightError::GuestFunctionCallAlreadyInProgress()
362 | HyperlightError::GuestInterfaceUnsupportedType(_)
363 | HyperlightError::HostFunctionNotFound(_)
364 | HyperlightError::HyperlightVmError(HyperlightVmError::Create(_))
365 | HyperlightError::HyperlightVmError(HyperlightVmError::Initialize(_))
366 | HyperlightError::HyperlightVmError(HyperlightVmError::MapRegion(_))
367 | HyperlightError::HyperlightVmError(HyperlightVmError::UnmapRegion(_))
368 | HyperlightError::IOError(_)
369 | HyperlightError::IntConversionFailure(_)
370 | HyperlightError::InvalidFlatBuffer(_)
371 | HyperlightError::JsonConversionFailure(_)
372 | HyperlightError::LockAttemptFailed(_)
373 | HyperlightError::MemoryAllocationFailed(_)
374 | HyperlightError::MemoryProtectionFailed(_)
375 | HyperlightError::MemoryRequestTooBig(_, _)
376 | HyperlightError::MemoryRequestTooSmall(_, _)
377 | HyperlightError::MetricNotFound(_)
378 | HyperlightError::MmapFailed(_)
379 | HyperlightError::MprotectFailed(_)
380 | HyperlightError::NoHypervisorFound()
381 | HyperlightError::NoMemorySnapshot
382 | HyperlightError::ParameterValueConversionFailure(_, _)
383 | HyperlightError::PEFileProcessingFailure(_)
384 | HyperlightError::RawPointerLessThanBaseAddress(_, _)
385 | HyperlightError::RefCellBorrowFailed(_)
386 | HyperlightError::RefCellMutBorrowFailed(_)
387 | HyperlightError::ReturnValueConversionFailure(_, _)
388 | HyperlightError::SnapshotSandboxMismatch
389 | HyperlightError::SystemTimeError(_)
390 | HyperlightError::TryFromSliceError(_)
391 | HyperlightError::UnexpectedNoOfArguments(_, _)
392 | HyperlightError::UnexpectedParameterValueType(_, _)
393 | HyperlightError::UnexpectedReturnValueType(_, _)
394 | HyperlightError::UTF8StringConversionFailure(_)
395 | HyperlightError::VectorCapacityIncorrect(_, _, _) => false,
396
397 #[cfg(target_os = "windows")]
398 HyperlightError::CrossBeamReceiveError(_) => false,
399 #[cfg(target_os = "windows")]
400 HyperlightError::CrossBeamSendError(_) => false,
401 #[cfg(target_os = "windows")]
402 HyperlightError::WindowsAPIError(_) => false,
403 #[cfg(target_os = "linux")]
404 HyperlightError::VmmSysError(_) => false,
405 }
406 }
407}
408
409#[macro_export]
411macro_rules! new_error {
412 ($msg:literal $(,)?) => {{
413 let __args = std::format_args!($msg);
414 let __err_msg = match __args.as_str() {
415 Some(msg) => String::from(msg),
416 None => std::format!($msg),
417 };
418 $crate::HyperlightError::Error(__err_msg)
419 }};
420 ($fmtstr:expr, $($arg:tt)*) => {{
421 let __err_msg = std::format!($fmtstr, $($arg)*);
422 $crate::error::HyperlightError::Error(__err_msg)
423 }};
424}
425
426#[cfg(test)]
427mod tests {
428 use super::*;
429 use crate::hypervisor::hyperlight_vm::{
430 DispatchGuestCallError, HandleIoError, HyperlightVmError, RunVmError,
431 };
432 use crate::sandbox::outb::HandleOutbError;
433
434 #[test]
436 fn test_promote_execution_cancelled_by_host() {
437 let err = DispatchGuestCallError::Run(RunVmError::ExecutionCancelledByHost);
438 let (promoted, should_poison) = err.promote();
439
440 assert!(
441 should_poison,
442 "ExecutionCancelledByHost should poison the sandbox"
443 );
444 assert!(
445 matches!(promoted, HyperlightError::ExecutionCanceledByHost()),
446 "Expected HyperlightError::ExecutionCanceledByHost, got {:?}",
447 promoted
448 );
449 }
450
451 #[test]
453 fn test_promote_guest_aborted() {
454 let err = DispatchGuestCallError::Run(RunVmError::HandleIo(HandleIoError::Outb(
455 HandleOutbError::GuestAborted {
456 code: 42,
457 message: "test abort".to_string(),
458 },
459 )));
460 let (promoted, should_poison) = err.promote();
461
462 assert!(should_poison, "GuestAborted should poison the sandbox");
463 match promoted {
464 HyperlightError::GuestAborted(code, msg) => {
465 assert_eq!(code, 42);
466 assert_eq!(msg, "test abort");
467 }
468 _ => panic!("Expected HyperlightError::GuestAborted, got {:?}", promoted),
469 }
470 }
471
472 #[test]
474 fn test_promote_memory_access_violation() {
475 let err = DispatchGuestCallError::Run(RunVmError::MemoryAccessViolation {
476 addr: 0xDEADBEEF,
477 access_type: MemoryRegionFlags::WRITE,
478 region_flags: MemoryRegionFlags::READ,
479 });
480 let (promoted, should_poison) = err.promote();
481
482 assert!(
483 should_poison,
484 "MemoryAccessViolation should poison the sandbox"
485 );
486 match promoted {
487 HyperlightError::MemoryAccessViolation(addr, access_type, region_flags) => {
488 assert_eq!(addr, 0xDEADBEEF);
489 assert_eq!(access_type, MemoryRegionFlags::WRITE);
490 assert_eq!(region_flags, MemoryRegionFlags::READ);
491 }
492 _ => panic!(
493 "Expected HyperlightError::MemoryAccessViolation, got {:?}",
494 promoted
495 ),
496 }
497 }
498
499 #[test]
501 fn test_promote_other_run_errors_wrapped() {
502 let err = DispatchGuestCallError::Run(RunVmError::MmioReadUnmapped(0x1000));
503 let (promoted, should_poison) = err.promote();
504
505 assert!(should_poison, "Run errors should poison the sandbox");
506 assert!(
507 matches!(
508 promoted,
509 HyperlightError::HyperlightVmError(HyperlightVmError::DispatchGuestCall(_))
510 ),
511 "Expected HyperlightError::HyperlightVmError, got {:?}",
512 promoted
513 );
514 }
515}