2022-10-06 13:33:09 +02:00
|
|
|
//! Utilities for dealing with span tracking in the compiler and in
|
|
|
|
//! error reporting.
|
2022-09-16 20:51:03 +02:00
|
|
|
|
|
|
|
use codemap::{File, Span};
|
2022-10-06 13:33:09 +02:00
|
|
|
use rnix::ast;
|
2022-09-16 20:51:03 +02:00
|
|
|
use rowan::ast::AstNode;
|
2022-12-11 13:16:59 +01:00
|
|
|
|
2022-09-16 20:51:03 +02:00
|
|
|
/// Trait implemented by all types from which we can retrieve a span.
|
2022-10-06 13:33:09 +02:00
|
|
|
pub trait ToSpan {
|
2022-09-16 20:51:03 +02:00
|
|
|
fn span_for(&self, file: &File) -> Span;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ToSpan for Span {
|
|
|
|
fn span_for(&self, _: &File) -> Span {
|
|
|
|
*self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-06 13:35:18 +02:00
|
|
|
impl ToSpan for rnix::TextRange {
|
|
|
|
fn span_for(&self, file: &File) -> Span {
|
|
|
|
file.span
|
|
|
|
.subspan(u32::from(self.start()) as u64, u32::from(self.end()) as u64)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-16 20:51:03 +02:00
|
|
|
impl ToSpan for rnix::SyntaxToken {
|
|
|
|
fn span_for(&self, file: &File) -> Span {
|
2022-10-06 13:35:18 +02:00
|
|
|
self.text_range().span_for(file)
|
2022-09-16 20:51:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ToSpan for rnix::SyntaxNode {
|
|
|
|
fn span_for(&self, file: &File) -> Span {
|
2022-10-06 13:35:18 +02:00
|
|
|
self.text_range().span_for(file)
|
2022-09-16 20:51:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-05 05:39:51 +02:00
|
|
|
/// A placeholder [`ToSpan`] implementation covering the entire source file.
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
|
|
pub struct EntireFile;
|
|
|
|
|
|
|
|
impl ToSpan for EntireFile {
|
|
|
|
fn span_for(&self, file: &File) -> Span {
|
|
|
|
file.span
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A placeholder [`ToSpan`] implementation which falls back to the entire file if its wrapped value
|
|
|
|
/// is [`None`]
|
|
|
|
#[derive(Debug, Clone, Copy)]
|
|
|
|
pub struct OrEntireFile<T>(pub Option<T>);
|
|
|
|
|
|
|
|
impl<T> ToSpan for OrEntireFile<T>
|
|
|
|
where
|
|
|
|
T: ToSpan,
|
|
|
|
{
|
|
|
|
fn span_for(&self, file: &File) -> Span {
|
|
|
|
match &self.0 {
|
|
|
|
Some(t) => t.span_for(file),
|
|
|
|
None => EntireFile.span_for(file),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-16 20:51:03 +02:00
|
|
|
/// Generates a `ToSpan` implementation for a type implementing
|
|
|
|
/// `rowan::AstNode`. This is impossible to do as a blanket
|
|
|
|
/// implementation because `rustc` forbids these implementations for
|
|
|
|
/// traits from third-party crates due to a belief that semantic
|
|
|
|
/// versioning truly could work (it doesn't).
|
|
|
|
macro_rules! expr_to_span {
|
|
|
|
( $type:path ) => {
|
|
|
|
impl ToSpan for $type {
|
|
|
|
fn span_for(&self, file: &File) -> Span {
|
|
|
|
self.syntax().span_for(file)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
expr_to_span!(ast::Expr);
|
|
|
|
expr_to_span!(ast::Apply);
|
|
|
|
expr_to_span!(ast::Assert);
|
|
|
|
expr_to_span!(ast::Attr);
|
|
|
|
expr_to_span!(ast::AttrSet);
|
|
|
|
expr_to_span!(ast::Attrpath);
|
|
|
|
expr_to_span!(ast::AttrpathValue);
|
|
|
|
expr_to_span!(ast::BinOp);
|
|
|
|
expr_to_span!(ast::HasAttr);
|
|
|
|
expr_to_span!(ast::Ident);
|
|
|
|
expr_to_span!(ast::IdentParam);
|
|
|
|
expr_to_span!(ast::IfElse);
|
|
|
|
expr_to_span!(ast::Inherit);
|
|
|
|
expr_to_span!(ast::Interpol);
|
|
|
|
expr_to_span!(ast::Lambda);
|
|
|
|
expr_to_span!(ast::LegacyLet);
|
|
|
|
expr_to_span!(ast::LetIn);
|
|
|
|
expr_to_span!(ast::List);
|
|
|
|
expr_to_span!(ast::Literal);
|
|
|
|
expr_to_span!(ast::PatBind);
|
|
|
|
expr_to_span!(ast::Path);
|
|
|
|
expr_to_span!(ast::Pattern);
|
|
|
|
expr_to_span!(ast::Select);
|
|
|
|
expr_to_span!(ast::Str);
|
|
|
|
expr_to_span!(ast::UnaryOp);
|
|
|
|
expr_to_span!(ast::With);
|