1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46
use crate::{error::CompileError, warning::CompileWarning};
use core::cell::RefCell;
/// A handler with which you can emit diagnostics.
#[derive(Default)]
pub struct Handler {
/// The inner handler.
/// This construction is used to avoid `&mut` all over the compiler.
inner: RefCell<HandlerInner>,
}
/// Contains the actual data for `Handler`.
/// Modelled this way to afford an API using interior mutability.
#[derive(Default)]
struct HandlerInner {
/// The sink through which errors will be emitted.
errors: Vec<CompileError>,
/// The sink through which warnings will be emitted.
warnings: Vec<CompileWarning>,
}
impl Handler {
/// Emit the error `err`.
pub fn emit_err(&self, err: CompileError) -> ErrorEmitted {
self.inner.borrow_mut().errors.push(err);
ErrorEmitted { _priv: () }
}
/// Emit the warning `warn`.
pub fn emit_warn(&self, warn: CompileWarning) {
self.inner.borrow_mut().warnings.push(warn);
}
/// Extract all the errors from this handler.
pub fn consume(self) -> (Vec<CompileError>, Vec<CompileWarning>) {
let inner = self.inner.into_inner();
(inner.errors, inner.warnings)
}
}
/// Proof that an error was emitted through a `Handler`.
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct ErrorEmitted {
_priv: (),
}