#[cfg(not(feature = "parking_lot"))]
use std::sync::Mutex;
use std::{
borrow::Cow,
cmp, fmt,
hash::{Hash, Hasher},
ops::{Add, Sub},
path::PathBuf,
sync::atomic::AtomicU32,
};
#[cfg(feature = "parking_lot")]
use parking_lot::Mutex;
use serde::{Deserialize, Serialize};
use url::Url;
use self::hygiene::MarkData;
pub use self::hygiene::{Mark, SyntaxContext};
use crate::{cache::CacheCell, rustc_data_structures::stable_hasher::StableHasher, sync::Lrc};
mod analyze_source_file;
pub mod hygiene;
#[derive(Clone, Copy, Hash, PartialEq, Eq, Ord, PartialOrd, Serialize, Deserialize)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct Span {
#[serde(rename = "start")]
#[cfg_attr(feature = "__rkyv", rkyv(omit_bounds))]
pub lo: BytePos,
#[serde(rename = "end")]
#[cfg_attr(feature = "__rkyv", rkyv(omit_bounds))]
pub hi: BytePos,
}
impl std::fmt::Debug for Span {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}..{}", self.lo.0, self.hi.0,)
}
}
impl From<(BytePos, BytePos)> for Span {
#[inline]
fn from(sp: (BytePos, BytePos)) -> Self {
Span::new(sp.0, sp.1)
}
}
impl From<Span> for (BytePos, BytePos) {
#[inline]
fn from(sp: Span) -> Self {
(sp.lo, sp.hi)
}
}
#[cfg(feature = "arbitrary")]
#[cfg_attr(docsrs, doc(cfg(feature = "arbitrary")))]
impl<'a> arbitrary::Arbitrary<'a> for Span {
fn arbitrary(u: &mut arbitrary::Unstructured<'_>) -> arbitrary::Result<Self> {
let lo = u.arbitrary::<BytePos>()?;
let hi = u.arbitrary::<BytePos>()?;
Ok(Self::new(lo, hi))
}
}
pub const DUMMY_SP: Span = Span {
lo: BytePos::DUMMY,
hi: BytePos::DUMMY,
};
pub const PURE_SP: Span = Span {
lo: BytePos::PURE,
hi: BytePos::PURE,
};
pub const PLACEHOLDER_SP: Span = Span {
lo: BytePos::PLACEHOLDER,
hi: BytePos::PLACEHOLDER,
};
pub struct Globals {
hygiene_data: Mutex<hygiene::HygieneData>,
#[allow(unused)]
dummy_cnt: AtomicU32,
#[allow(unused)]
marks: Mutex<Vec<MarkData>>,
}
const DUMMY_RESERVE: u32 = u32::MAX - 2_u32.pow(16);
impl Default for Globals {
fn default() -> Self {
Self::new()
}
}
impl Globals {
pub fn new() -> Globals {
Globals {
hygiene_data: Mutex::new(hygiene::HygieneData::new()),
marks: Mutex::new(vec![MarkData {
parent: Mark::root(),
}]),
dummy_cnt: AtomicU32::new(DUMMY_RESERVE),
}
}
}
better_scoped_tls::scoped_tls!(
pub static GLOBALS: Globals
);
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(u32))]
#[derive(Debug, Eq, PartialEq, Clone, Ord, PartialOrd, Hash)]
pub enum FileName {
Real(
#[cfg_attr(
any(feature = "rkyv-impl"),
rkyv(with = crate::source_map::EncodePathBuf)
)]
PathBuf,
),
Macros(String),
QuoteExpansion,
Anon,
MacroExpansion,
ProcMacroSourceCode,
Url(#[cfg_attr(any(feature = "rkyv-impl"), rkyv(with = crate::source_map::EncodeUrl))] Url),
Internal(String),
Custom(String),
}
#[cfg(feature = "rkyv-impl")]
#[derive(Debug, Clone, Copy)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct EncodePathBuf;
#[cfg(feature = "rkyv-impl")]
impl rkyv::with::ArchiveWith<PathBuf> for EncodePathBuf {
type Archived = rkyv::string::ArchivedString;
type Resolver = rkyv::string::StringResolver;
#[inline]
fn resolve_with(field: &PathBuf, resolver: Self::Resolver, out: rkyv::Place<Self::Archived>) {
rkyv::string::ArchivedString::resolve_from_str(field.to_str().unwrap(), resolver, out);
}
}
#[cfg(feature = "rkyv-impl")]
impl<S> rkyv::with::SerializeWith<PathBuf, S> for EncodePathBuf
where
S: ?Sized + rancor::Fallible + rkyv::ser::Writer,
S::Error: rancor::Source,
{
#[inline]
fn serialize_with(field: &PathBuf, serializer: &mut S) -> Result<Self::Resolver, S::Error> {
let s = field.to_str().unwrap_or_default();
rkyv::string::ArchivedString::serialize_from_str(s, serializer)
}
}
#[cfg(feature = "rkyv-impl")]
impl<D> rkyv::with::DeserializeWith<rkyv::string::ArchivedString, PathBuf, D> for EncodePathBuf
where
D: ?Sized + rancor::Fallible,
{
#[inline]
fn deserialize_with(
field: &rkyv::string::ArchivedString,
_: &mut D,
) -> Result<PathBuf, D::Error> {
Ok(<PathBuf as std::str::FromStr>::from_str(field.as_str()).unwrap())
}
}
#[cfg(feature = "rkyv-impl")]
#[derive(Debug, Clone, Copy)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct EncodeUrl;
#[cfg(feature = "rkyv-impl")]
impl rkyv::with::ArchiveWith<Url> for EncodeUrl {
type Archived = rkyv::string::ArchivedString;
type Resolver = rkyv::string::StringResolver;
#[inline]
fn resolve_with(field: &Url, resolver: Self::Resolver, out: rkyv::Place<Self::Archived>) {
rkyv::string::ArchivedString::resolve_from_str(field.as_str(), resolver, out);
}
}
#[cfg(feature = "rkyv-impl")]
impl<S> rkyv::with::SerializeWith<Url, S> for EncodeUrl
where
S: ?Sized + rancor::Fallible + rkyv::ser::Writer,
S::Error: rancor::Source,
{
#[inline]
fn serialize_with(field: &Url, serializer: &mut S) -> Result<Self::Resolver, S::Error> {
let field = field.as_str();
rkyv::string::ArchivedString::serialize_from_str(field, serializer)
}
}
#[cfg(feature = "rkyv-impl")]
impl<D> rkyv::with::DeserializeWith<rkyv::Archived<String>, Url, D> for EncodeUrl
where
D: ?Sized + rancor::Fallible,
{
#[inline]
fn deserialize_with(field: &rkyv::string::ArchivedString, _: &mut D) -> Result<Url, D::Error> {
Ok(Url::parse(field.as_str()).unwrap())
}
}
impl std::fmt::Display for FileName {
fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match *self {
FileName::Real(ref path) => write!(fmt, "{}", path.display()),
FileName::Macros(ref name) => write!(fmt, "<{} macros>", name),
FileName::QuoteExpansion => write!(fmt, "<quote expansion>"),
FileName::MacroExpansion => write!(fmt, "<macro expansion>"),
FileName::Anon => write!(fmt, "<anon>"),
FileName::ProcMacroSourceCode => write!(fmt, "<proc-macro source code>"),
FileName::Url(ref u) => write!(fmt, "{}", u),
FileName::Custom(ref s) => {
write!(fmt, "{}", s)
}
FileName::Internal(ref s) => write!(fmt, "<{}>", s),
}
}
}
impl From<PathBuf> for FileName {
fn from(p: PathBuf) -> Self {
assert!(!p.to_string_lossy().ends_with('>'));
FileName::Real(p)
}
}
impl From<Url> for FileName {
fn from(url: Url) -> Self {
FileName::Url(url)
}
}
impl FileName {
pub fn is_real(&self) -> bool {
match *self {
FileName::Real(_) => true,
FileName::Macros(_)
| FileName::Anon
| FileName::MacroExpansion
| FileName::ProcMacroSourceCode
| FileName::Custom(_)
| FileName::QuoteExpansion
| FileName::Internal(_)
| FileName::Url(_) => false,
}
}
pub fn is_macros(&self) -> bool {
match *self {
FileName::Real(_)
| FileName::Anon
| FileName::MacroExpansion
| FileName::ProcMacroSourceCode
| FileName::Custom(_)
| FileName::QuoteExpansion
| FileName::Internal(_)
| FileName::Url(_) => false,
FileName::Macros(_) => true,
}
}
}
#[derive(Clone, Debug, Default, Hash, PartialEq, Eq)]
#[cfg_attr(
feature = "diagnostic-serde",
derive(serde::Serialize, serde::Deserialize)
)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct PrimarySpanLabel(pub Span, pub String);
#[derive(Clone, Debug, Default, Hash, PartialEq, Eq)]
#[cfg_attr(
feature = "diagnostic-serde",
derive(serde::Serialize, serde::Deserialize)
)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct MultiSpan {
primary_spans: Vec<Span>,
span_labels: Vec<PrimarySpanLabel>,
}
extern "C" {
fn __span_dummy_with_cmt_proxy() -> u32;
}
impl Span {
#[inline]
pub fn lo(self) -> BytePos {
self.lo
}
#[inline]
pub fn new(mut lo: BytePos, mut hi: BytePos) -> Self {
if lo > hi {
std::mem::swap(&mut lo, &mut hi);
}
Span { lo, hi }
}
#[inline]
pub fn with_lo(&self, lo: BytePos) -> Span {
Span::new(lo, self.hi)
}
#[inline]
pub fn hi(self) -> BytePos {
self.hi
}
#[inline]
pub fn with_hi(&self, hi: BytePos) -> Span {
Span::new(self.lo, hi)
}
#[inline]
pub fn is_dummy(self) -> bool {
self.lo.0 == 0 && self.hi.0 == 0 || self.lo.0 >= DUMMY_RESERVE
}
#[inline]
pub fn is_pure(self) -> bool {
self.lo.is_pure()
}
#[inline]
pub fn is_placeholder(self) -> bool {
self.lo.is_placeholder()
}
#[inline]
pub fn is_dummy_ignoring_cmt(self) -> bool {
self.lo.0 == 0 && self.hi.0 == 0
}
#[inline]
pub fn shrink_to_lo(self) -> Span {
self.with_hi(self.lo)
}
#[inline]
pub fn shrink_to_hi(self) -> Span {
self.with_lo(self.hi)
}
pub fn substitute_dummy(self, other: Span) -> Span {
if self.is_dummy() {
other
} else {
self
}
}
pub fn contains(self, other: Span) -> bool {
self.lo <= other.lo && other.hi <= self.hi
}
pub fn source_equal(self, other: Span) -> bool {
self.lo == other.lo && self.hi == other.hi
}
pub fn trim_start(self, other: Span) -> Option<Span> {
if self.hi > other.hi {
Some(self.with_lo(cmp::max(self.lo, other.hi)))
} else {
None
}
}
pub fn to(self, end: Span) -> Span {
let span_data = self;
let end_data = end;
Span::new(
cmp::min(span_data.lo, end_data.lo),
cmp::max(span_data.hi, end_data.hi),
)
}
pub fn between(self, end: Span) -> Span {
let span = self;
Span::new(span.hi, end.lo)
}
pub fn until(self, end: Span) -> Span {
let span = self;
Span::new(span.lo, end.lo)
}
pub fn from_inner_byte_pos(self, start: usize, end: usize) -> Span {
let span = self;
Span::new(
span.lo + BytePos::from_usize(start),
span.lo + BytePos::from_usize(end),
)
}
pub fn dummy_with_cmt() -> Self {
#[cfg(all(feature = "__plugin_mode", target_arch = "wasm32"))]
{
let lo = BytePos(unsafe { __span_dummy_with_cmt_proxy() });
return Span { lo, hi: lo };
}
#[cfg(not(all(any(feature = "__plugin_mode"), target_arch = "wasm32")))]
return GLOBALS.with(|globals| {
let lo = BytePos(
globals
.dummy_cnt
.fetch_add(1, std::sync::atomic::Ordering::SeqCst),
);
Span { lo, hi: lo }
});
}
}
#[derive(Clone, Debug)]
pub struct SpanLabel {
pub span: Span,
pub is_primary: bool,
pub label: Option<String>,
}
impl Default for Span {
fn default() -> Self {
DUMMY_SP
}
}
impl MultiSpan {
#[inline]
pub fn new() -> MultiSpan {
Self::default()
}
pub fn from_span(primary_span: Span) -> MultiSpan {
MultiSpan {
primary_spans: vec![primary_span],
span_labels: Vec::new(),
}
}
pub fn from_spans(vec: Vec<Span>) -> MultiSpan {
MultiSpan {
primary_spans: vec,
span_labels: Vec::new(),
}
}
pub fn push_span_label(&mut self, span: Span, label: String) {
self.span_labels.push(PrimarySpanLabel(span, label));
}
pub fn primary_span(&self) -> Option<Span> {
self.primary_spans.first().cloned()
}
pub fn primary_spans(&self) -> &[Span] {
&self.primary_spans
}
pub fn is_dummy(&self) -> bool {
let mut is_dummy = true;
for span in &self.primary_spans {
if !span.is_dummy() {
is_dummy = false;
}
}
is_dummy
}
pub fn replace(&mut self, before: Span, after: Span) -> bool {
let mut replacements_occurred = false;
for primary_span in &mut self.primary_spans {
if *primary_span == before {
*primary_span = after;
replacements_occurred = true;
}
}
for span_label in &mut self.span_labels {
if span_label.0 == before {
span_label.0 = after;
replacements_occurred = true;
}
}
replacements_occurred
}
pub fn span_labels(&self) -> Vec<SpanLabel> {
let is_primary = |span| self.primary_spans.contains(&span);
let mut span_labels = self
.span_labels
.iter()
.map(|&PrimarySpanLabel(span, ref label)| SpanLabel {
span,
is_primary: is_primary(span),
label: Some(label.clone()),
})
.collect::<Vec<_>>();
for &span in &self.primary_spans {
if !span_labels.iter().any(|sl| sl.span == span) {
span_labels.push(SpanLabel {
span,
is_primary: true,
label: None,
});
}
}
span_labels
}
}
impl From<Span> for MultiSpan {
fn from(span: Span) -> MultiSpan {
MultiSpan::from_span(span)
}
}
impl From<Vec<Span>> for MultiSpan {
fn from(spans: Vec<Span>) -> MultiSpan {
MultiSpan::from_spans(spans)
}
}
pub const NO_EXPANSION: SyntaxContext = SyntaxContext::empty();
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
#[derive(Copy, Clone, Eq, PartialEq, Debug)]
pub struct MultiByteChar {
pub pos: BytePos,
pub bytes: u8,
}
impl MultiByteChar {
pub fn byte_to_char_diff(&self) -> u8 {
if self.bytes == 4 {
2
} else {
self.bytes - 1
}
}
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(u32))]
#[derive(Copy, Clone, Eq, PartialEq, Debug)]
pub enum NonNarrowChar {
ZeroWidth(BytePos),
Wide(BytePos),
Tab(BytePos),
}
impl NonNarrowChar {
fn new(pos: BytePos, width: usize) -> Self {
match width {
0 => NonNarrowChar::ZeroWidth(pos),
2 => NonNarrowChar::Wide(pos),
4 => NonNarrowChar::Tab(pos),
_ => panic!("width {} given for non-narrow character", width),
}
}
pub fn pos(self) -> BytePos {
match self {
NonNarrowChar::ZeroWidth(p) | NonNarrowChar::Wide(p) | NonNarrowChar::Tab(p) => p,
}
}
pub fn width(self) -> usize {
match self {
NonNarrowChar::ZeroWidth(_) => 0,
NonNarrowChar::Wide(_) => 2,
NonNarrowChar::Tab(_) => 4,
}
}
}
impl Add<BytePos> for NonNarrowChar {
type Output = Self;
fn add(self, rhs: BytePos) -> Self {
match self {
NonNarrowChar::ZeroWidth(pos) => NonNarrowChar::ZeroWidth(pos + rhs),
NonNarrowChar::Wide(pos) => NonNarrowChar::Wide(pos + rhs),
NonNarrowChar::Tab(pos) => NonNarrowChar::Tab(pos + rhs),
}
}
}
impl Sub<BytePos> for NonNarrowChar {
type Output = Self;
fn sub(self, rhs: BytePos) -> Self {
match self {
NonNarrowChar::ZeroWidth(pos) => NonNarrowChar::ZeroWidth(pos - rhs),
NonNarrowChar::Wide(pos) => NonNarrowChar::Wide(pos - rhs),
NonNarrowChar::Tab(pos) => NonNarrowChar::Tab(pos - rhs),
}
}
}
#[doc(hidden)]
#[cfg(feature = "rkyv-impl")]
#[derive(Debug, Clone, Copy)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct EncodeArcString;
#[cfg(feature = "rkyv-impl")]
impl rkyv::with::ArchiveWith<Lrc<String>> for EncodeArcString {
type Archived = rkyv::Archived<String>;
type Resolver = rkyv::Resolver<String>;
fn resolve_with(
field: &Lrc<String>,
resolver: Self::Resolver,
out: rkyv::Place<Self::Archived>,
) {
let s = field.to_string();
rkyv::Archive::resolve(&s, resolver, out);
}
}
#[cfg(feature = "rkyv-impl")]
impl<S> rkyv::with::SerializeWith<Lrc<String>, S> for EncodeArcString
where
S: ?Sized + rancor::Fallible + rkyv::ser::Writer,
S::Error: rancor::Source,
{
fn serialize_with(field: &Lrc<String>, serializer: &mut S) -> Result<Self::Resolver, S::Error> {
rkyv::string::ArchivedString::serialize_from_str(field, serializer)
}
}
#[cfg(feature = "rkyv-impl")]
impl<D> rkyv::with::DeserializeWith<rkyv::Archived<String>, Lrc<String>, D> for EncodeArcString
where
D: ?Sized + rancor::Fallible,
{
fn deserialize_with(
field: &rkyv::Archived<String>,
deserializer: &mut D,
) -> Result<Lrc<String>, D::Error> {
use rkyv::Deserialize;
let s: String = field.deserialize(deserializer)?;
Ok(s.into())
}
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
#[derive(Clone)]
pub struct SourceFile {
pub name: Lrc<FileName>,
pub name_was_remapped: bool,
pub unmapped_path: Option<Lrc<FileName>>,
pub crate_of_origin: u32,
#[cfg_attr(any(feature = "rkyv-impl"), rkyv(with = EncodeArcString))]
pub src: Lrc<String>,
pub src_hash: u128,
pub start_pos: BytePos,
pub end_pos: BytePos,
pub name_hash: u128,
lazy: CacheCell<SourceFileAnalysis>,
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
#[derive(Clone)]
pub struct SourceFileAnalysis {
pub lines: Vec<BytePos>,
pub multibyte_chars: Vec<MultiByteChar>,
pub non_narrow_chars: Vec<NonNarrowChar>,
}
impl fmt::Debug for SourceFile {
fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(fmt, "SourceFile({})", self.name)
}
}
impl SourceFile {
pub fn new(
name: Lrc<FileName>,
name_was_remapped: bool,
unmapped_path: Lrc<FileName>,
mut src: String,
start_pos: BytePos,
) -> SourceFile {
remove_bom(&mut src);
Self::new_from(
name,
name_was_remapped,
unmapped_path,
Lrc::new(src),
start_pos,
)
}
pub fn new_from(
name: Lrc<FileName>,
name_was_remapped: bool,
unmapped_path: Lrc<FileName>,
src: Lrc<String>,
start_pos: BytePos,
) -> SourceFile {
debug_assert_ne!(
start_pos,
BytePos::DUMMY,
"BytePos::DUMMY is reserved and `SourceFile` should not use it"
);
let src_hash = {
let mut hasher: StableHasher = StableHasher::new();
hasher.write(src.as_bytes());
hasher.finish()
};
let name_hash = {
let mut hasher: StableHasher = StableHasher::new();
name.hash(&mut hasher);
hasher.finish()
};
let end_pos = start_pos.to_usize() + src.len();
SourceFile {
name,
name_was_remapped,
unmapped_path: Some(unmapped_path),
crate_of_origin: 0,
src,
src_hash,
start_pos,
end_pos: SmallPos::from_usize(end_pos),
name_hash,
lazy: CacheCell::new(),
}
}
pub fn line_begin_pos(&self, pos: BytePos) -> BytePos {
let line_index = self.lookup_line(pos).unwrap();
let analysis = self.analyze();
analysis.lines[line_index]
}
pub fn get_line(&self, line_number: usize) -> Option<Cow<'_, str>> {
fn get_until_newline(src: &str, begin: usize) -> &str {
let slice = &src[begin..];
match slice.find('\n') {
Some(e) => &slice[..e],
None => slice,
}
}
let begin = {
let analysis = self.analyze();
let line = analysis.lines.get(line_number)?;
let begin: BytePos = *line - self.start_pos;
begin.to_usize()
};
Some(Cow::from(get_until_newline(&self.src, begin)))
}
pub fn is_real_file(&self) -> bool {
self.name.is_real()
}
pub fn byte_length(&self) -> u32 {
self.end_pos.0 - self.start_pos.0
}
pub fn count_lines(&self) -> usize {
let analysis = self.analyze();
analysis.lines.len()
}
pub fn lookup_line(&self, pos: BytePos) -> Option<usize> {
let analysis = self.analyze();
if analysis.lines.is_empty() {
return None;
}
let line_index = lookup_line(&analysis.lines, pos);
assert!(line_index < analysis.lines.len() as isize);
if line_index >= 0 {
Some(line_index as usize)
} else {
None
}
}
pub fn line_bounds(&self, line_index: usize) -> (BytePos, BytePos) {
if self.start_pos == self.end_pos {
return (self.start_pos, self.end_pos);
}
let analysis = self.analyze();
assert!(line_index < analysis.lines.len());
if line_index == (analysis.lines.len() - 1) {
(analysis.lines[line_index], self.end_pos)
} else {
(analysis.lines[line_index], analysis.lines[line_index + 1])
}
}
#[inline]
pub fn contains(&self, byte_pos: BytePos) -> bool {
byte_pos >= self.start_pos && byte_pos <= self.end_pos
}
pub fn analyze(&self) -> &SourceFileAnalysis {
self.lazy.get_or_init(|| {
let (lines, multibyte_chars, non_narrow_chars) =
analyze_source_file::analyze_source_file(&self.src[..], self.start_pos);
SourceFileAnalysis {
lines,
multibyte_chars,
non_narrow_chars,
}
})
}
}
pub(super) fn remove_bom(src: &mut String) {
if src.starts_with('\u{feff}') {
src.drain(..3);
}
}
pub trait SmallPos {
fn from_usize(n: usize) -> Self;
fn to_usize(&self) -> usize;
fn from_u32(n: u32) -> Self;
fn to_u32(&self) -> u32;
}
#[derive(
Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord, Debug, Serialize, Deserialize, Default,
)]
#[serde(transparent)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct BytePos(#[cfg_attr(feature = "__rkyv", rkyv(omit_bounds))] pub u32);
impl BytePos {
pub const DUMMY: Self = BytePos(0);
const MIN_RESERVED: Self = BytePos(DUMMY_RESERVE);
pub const PLACEHOLDER: Self = BytePos(u32::MAX - 2);
pub const PURE: Self = BytePos(u32::MAX - 1);
pub const SYNTHESIZED: Self = BytePos(u32::MAX);
pub const fn is_reserved_for_comments(self) -> bool {
self.0 >= Self::MIN_RESERVED.0 && self.0 != u32::MAX
}
pub const fn is_dummy(self) -> bool {
self.0 == 0
}
pub const fn is_pure(self) -> bool {
self.0 == Self::PURE.0
}
pub const fn is_placeholder(self) -> bool {
self.0 == Self::PLACEHOLDER.0
}
pub const fn can_have_comment(self) -> bool {
self.0 != 0
}
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
#[derive(Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Debug)]
pub struct CharPos(pub usize);
impl SmallPos for BytePos {
#[inline(always)]
fn from_usize(n: usize) -> BytePos {
BytePos(n as u32)
}
#[inline(always)]
fn to_usize(&self) -> usize {
self.0 as usize
}
#[inline(always)]
fn from_u32(n: u32) -> BytePos {
BytePos(n)
}
#[inline(always)]
fn to_u32(&self) -> u32 {
self.0
}
}
impl Add for BytePos {
type Output = BytePos;
#[inline(always)]
fn add(self, rhs: BytePos) -> BytePos {
BytePos((self.to_usize() + rhs.to_usize()) as u32)
}
}
impl Sub for BytePos {
type Output = BytePos;
#[inline(always)]
fn sub(self, rhs: BytePos) -> BytePos {
BytePos((self.to_usize() - rhs.to_usize()) as u32)
}
}
impl SmallPos for CharPos {
#[inline(always)]
fn from_usize(n: usize) -> CharPos {
CharPos(n)
}
#[inline(always)]
fn to_usize(&self) -> usize {
self.0
}
#[inline(always)]
fn from_u32(n: u32) -> CharPos {
CharPos(n as usize)
}
#[inline(always)]
fn to_u32(&self) -> u32 {
self.0 as u32
}
}
impl Add for CharPos {
type Output = CharPos;
#[inline(always)]
fn add(self, rhs: CharPos) -> CharPos {
CharPos(self.to_usize() + rhs.to_usize())
}
}
impl Sub for CharPos {
type Output = CharPos;
#[inline(always)]
fn sub(self, rhs: CharPos) -> CharPos {
CharPos(self.to_usize() - rhs.to_usize())
}
}
#[derive(Debug, Clone)]
pub struct Loc {
pub file: Lrc<SourceFile>,
pub line: usize,
pub col: CharPos,
pub col_display: usize,
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize, Debug, Clone)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct PartialLoc {
pub source_file: Option<Lrc<SourceFile>>,
pub line: usize,
pub col: usize,
pub col_display: usize,
}
#[derive(Debug)]
pub struct LocWithOpt {
pub filename: Lrc<FileName>,
pub line: usize,
pub col: CharPos,
pub file: Option<Lrc<SourceFile>>,
}
#[derive(Debug)]
pub struct SourceFileAndLine {
pub sf: Lrc<SourceFile>,
pub line: usize,
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
#[derive(Debug)]
pub struct SourceFileAndBytePos {
pub sf: Lrc<SourceFile>,
pub pos: BytePos,
}
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct LineInfo {
pub line_index: usize,
pub start_col: CharPos,
pub end_col: CharPos,
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub struct LineCol {
pub line: u32,
pub col: u32,
}
pub struct FileLines {
pub file: Lrc<SourceFile>,
pub lines: Vec<LineInfo>,
}
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize, Debug, Clone)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct PartialFileLines {
pub file: Option<Lrc<SourceFile>>,
pub lines: Vec<LineInfo>,
}
pub type FileLinesResult = Result<FileLines, Box<SpanLinesError>>;
#[cfg(feature = "__plugin")]
pub type PartialFileLinesResult = Result<PartialFileLines, Box<SpanLinesError>>;
#[derive(Clone, PartialEq, Eq, Debug)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(u32))]
pub enum SpanLinesError {
IllFormedSpan(Span),
DistinctSources(DistinctSources),
}
#[derive(Clone, PartialEq, Eq, Debug)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(u32))]
pub enum SpanSnippetError {
DummyBytePos,
IllFormedSpan(Span),
DistinctSources(DistinctSources),
MalformedForSourcemap(MalformedSourceMapPositions),
SourceNotAvailable { filename: FileName },
LookupFailed(SourceMapLookupError),
}
#[derive(Clone, PartialEq, Eq, Debug)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(u32))]
pub enum SourceMapLookupError {
NoFileFor(BytePos),
}
#[derive(Clone, PartialEq, Eq, Debug)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct FilePos(pub Lrc<FileName>, pub BytePos);
#[derive(Clone, PartialEq, Eq, Debug)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct DistinctSources {
pub begin: FilePos,
pub end: FilePos,
}
#[derive(Clone, PartialEq, Eq, Debug)]
#[cfg_attr(
any(feature = "rkyv-impl"),
derive(rkyv::Archive, rkyv::Serialize, rkyv::Deserialize)
)]
#[cfg_attr(feature = "rkyv-impl", derive(bytecheck::CheckBytes))]
#[cfg_attr(feature = "rkyv-impl", repr(C))]
pub struct MalformedSourceMapPositions {
pub name: Lrc<FileName>,
pub source_len: usize,
pub begin_pos: BytePos,
pub end_pos: BytePos,
}
fn lookup_line(lines: &[BytePos], pos: BytePos) -> isize {
match lines.binary_search(&pos) {
Ok(line) => line as isize,
Err(line) => line as isize - 1,
}
}
impl From<SourceMapLookupError> for Box<SpanSnippetError> {
#[cold]
fn from(err: SourceMapLookupError) -> Self {
Box::new(SpanSnippetError::LookupFailed(err))
}
}
#[cfg(test)]
mod tests {
use super::{lookup_line, BytePos, Span};
#[test]
fn test_lookup_line() {
let lines = &[BytePos(3), BytePos(17), BytePos(28)];
assert_eq!(lookup_line(lines, BytePos(0)), -1);
assert_eq!(lookup_line(lines, BytePos(3)), 0);
assert_eq!(lookup_line(lines, BytePos(4)), 0);
assert_eq!(lookup_line(lines, BytePos(16)), 0);
assert_eq!(lookup_line(lines, BytePos(17)), 1);
assert_eq!(lookup_line(lines, BytePos(18)), 1);
assert_eq!(lookup_line(lines, BytePos(28)), 2);
assert_eq!(lookup_line(lines, BytePos(29)), 2);
}
#[test]
fn size_of_span() {
assert_eq!(std::mem::size_of::<Span>(), 8);
}
}