1//! Span maps for real files and macro expansions.
23use span::{EditionedFileId, HirFileId, HirFileIdRepr, MacroFileId, Span, SyntaxContext};
4use stdx::TupleExt;
5use syntax::{AstNode, TextRange, ast};
6use triomphe::Arc;
78pub use span::RealSpanMap;
910use crate::{attrs::collect_attrs, db::ExpandDatabase};
1112pub type ExpansionSpanMap = span::SpanMap<SyntaxContext>;
1314/// Spanmap for a macro file or a real file
15#[derive(Clone, Debug, PartialEq, Eq)]
16pub enum SpanMap {
17/// Spanmap for a macro file
18ExpansionSpanMap(Arc<ExpansionSpanMap>),
19/// Spanmap for a real file
20RealSpanMap(Arc<RealSpanMap>),
21}
2223#[derive(Copy, Clone)]
24pub enum SpanMapRef<'a> {
25/// Spanmap for a macro file
26ExpansionSpanMap(&'a ExpansionSpanMap),
27/// Spanmap for a real file
28RealSpanMap(&'a RealSpanMap),
29}
3031impl syntax_bridge::SpanMapper<Span> for SpanMap {
32fn span_for(&self, range: TextRange) -> Span {
33self.span_for_range(range)
34 }
35}
3637impl syntax_bridge::SpanMapper<Span> for SpanMapRef<'_> {
38fn span_for(&self, range: TextRange) -> Span {
39self.span_for_range(range)
40 }
41}
4243impl SpanMap {
44pub fn span_for_range(&self, range: TextRange) -> Span {
45match self {
46// FIXME: Is it correct for us to only take the span at the start? This feels somewhat
47 // wrong. The context will be right, but the range could be considered wrong. See
48 // https://github.com/rust-lang/rust/issues/23480, we probably want to fetch the span at
49 // the start and end, then merge them like rustc does in `Span::to
50Self::ExpansionSpanMap(span_map) => span_map.span_at(range.start()),
51Self::RealSpanMap(span_map) => span_map.span_for_range(range),
52 }
53 }
5455pub fn as_ref(&self) -> SpanMapRef<'_> {
56match self {
57Self::ExpansionSpanMap(span_map) => SpanMapRef::ExpansionSpanMap(span_map),
58Self::RealSpanMap(span_map) => SpanMapRef::RealSpanMap(span_map),
59 }
60 }
6162#[inline]
63pub(crate) fn new(db: &dyn ExpandDatabase, file_id: HirFileId) -> SpanMap {
64match file_id.repr() {
65 HirFileIdRepr::FileId(file_id) => SpanMap::RealSpanMap(db.real_span_map(file_id)),
66 HirFileIdRepr::MacroFile(m) => {
67 SpanMap::ExpansionSpanMap(db.parse_macro_expansion(m).value.1)
68 }
69 }
70 }
71}
7273impl SpanMapRef<'_> {
74pub fn span_for_range(self, range: TextRange) -> Span {
75match self {
76Self::ExpansionSpanMap(span_map) => span_map.span_at(range.start()),
77Self::RealSpanMap(span_map) => span_map.span_for_range(range),
78 }
79 }
80}
8182pub(crate) fn real_span_map(
83 db: &dyn ExpandDatabase,
84 editioned_file_id: EditionedFileId,
85) -> Arc<RealSpanMap> {
86use syntax::ast::HasModuleItem;
87let mut pairs = vec![(syntax::TextSize::new(0), span::ROOT_ERASED_FILE_AST_ID)];
88let ast_id_map = db.ast_id_map(editioned_file_id.into());
8990let file_id = base_db::EditionedFileId::new(db, editioned_file_id);
9192let tree = db.parse(file_id).tree();
93// This is an incrementality layer. Basically we can't use absolute ranges for our spans as that
94 // would mean we'd invalidate everything whenever we type. So instead we make the text ranges
95 // relative to some AstIds reducing the risk of invalidation as typing somewhere no longer
96 // affects all following spans in the file.
97 // There is some stuff to bear in mind here though, for one, the more "anchors" we create, the
98 // easier it gets to invalidate things again as spans are as stable as their anchor's ID.
99 // The other problem is proc-macros. Proc-macros have a `Span::join` api that allows them
100 // to join two spans that come from the same file. rust-analyzer's proc-macro server
101 // can only join two spans if they belong to the same anchor though, as the spans are relative
102 // to that anchor. To do cross anchor joining we'd need to access to the ast id map to resolve
103 // them again, something we might get access to in the future. But even then, proc-macros doing
104 // this kind of joining makes them as stable as the AstIdMap (which is basically changing on
105 // every input of the file)…
106107let item_to_entry =
108 |item: ast::Item| (item.syntax().text_range().start(), ast_id_map.ast_id(&item).erase());
109// Top level items make for great anchors as they are the most stable and a decent boundary
110pairs.extend(tree.items().map(item_to_entry));
111// Unfortunately, assoc items are very common in Rust, so descend into those as well and make
112 // them anchors too, but only if they have no attributes attached, as those might be proc-macros
113 // and using different anchors inside of them will prevent spans from being joinable.
114tree.items().for_each(|item| match &item {
115 ast::Item::ExternBlock(it)
116if !collect_attrs(it).map(TupleExt::tail).any(|it| it.is_left()) =>
117 {
118if let Some(extern_item_list) = it.extern_item_list() {
119 pairs.extend(
120 extern_item_list.extern_items().map(ast::Item::from).map(item_to_entry),
121 );
122 }
123 }
124 ast::Item::Impl(it) if !collect_attrs(it).map(TupleExt::tail).any(|it| it.is_left()) => {
125if let Some(assoc_item_list) = it.assoc_item_list() {
126 pairs.extend(assoc_item_list.assoc_items().map(ast::Item::from).map(item_to_entry));
127 }
128 }
129 ast::Item::Module(it) if !collect_attrs(it).map(TupleExt::tail).any(|it| it.is_left()) => {
130if let Some(item_list) = it.item_list() {
131 pairs.extend(item_list.items().map(item_to_entry));
132 }
133 }
134 ast::Item::Trait(it) if !collect_attrs(it).map(TupleExt::tail).any(|it| it.is_left()) => {
135if let Some(assoc_item_list) = it.assoc_item_list() {
136 pairs.extend(assoc_item_list.assoc_items().map(ast::Item::from).map(item_to_entry));
137 }
138 }
139_ => (),
140 });
141142 Arc::new(RealSpanMap::from_file(
143 editioned_file_id,
144 pairs.into_boxed_slice(),
145 tree.syntax().text_range().end(),
146 ))
147}
148149pub(crate) fn expansion_span_map(
150 db: &dyn ExpandDatabase,
151 file_id: MacroFileId,
152) -> Arc<ExpansionSpanMap> {
153 db.parse_macro_expansion(file_id).value.1
154}