1use rustc_data_structures::fx::FxIndexSet;
2use rustc_serialize::int_overflow::DebugStrictAdd;
5
6use crate::def_id::{DefIndex, LocalDefId};
7use crate::hygiene::SyntaxContext;
8use crate::{BytePos, SPAN_TRACK, SpanData};
9
10#[derive(Clone, Copy, Eq, PartialEq, Hash)]
80#[rustc_pass_by_value]
81pub struct Span {
82 lo_or_index: u32,
83 len_with_tag_or_marker: u16,
84 ctxt_or_parent_or_marker: u16,
85}
86
87#[derive(Clone, Copy)]
89struct InlineCtxt {
90 lo: u32,
91 len: u16,
92 ctxt: u16,
93}
94
95#[derive(Clone, Copy)]
96struct InlineParent {
97 lo: u32,
98 len_with_tag: u16,
99 parent: u16,
100}
101
102#[derive(Clone, Copy)]
103struct PartiallyInterned {
104 index: u32,
105 ctxt: u16,
106}
107
108#[derive(Clone, Copy)]
109struct Interned {
110 index: u32,
111}
112
113impl InlineCtxt {
114 #[inline]
115 fn data(self) -> SpanData {
116 let len = self.len as u32;
117 debug_assert!(len <= MAX_LEN);
118 SpanData {
119 lo: BytePos(self.lo),
120 hi: BytePos(self.lo.debug_strict_add(len)),
121 ctxt: SyntaxContext::from_u16(self.ctxt),
122 parent: None,
123 }
124 }
125 #[inline]
126 fn span(lo: u32, len: u16, ctxt: u16) -> Span {
127 Span { lo_or_index: lo, len_with_tag_or_marker: len, ctxt_or_parent_or_marker: ctxt }
128 }
129 #[inline]
130 fn from_span(span: Span) -> InlineCtxt {
131 let (lo, len, ctxt) =
132 (span.lo_or_index, span.len_with_tag_or_marker, span.ctxt_or_parent_or_marker);
133 InlineCtxt { lo, len, ctxt }
134 }
135}
136
137impl InlineParent {
138 #[inline]
139 fn data(self) -> SpanData {
140 let len = (self.len_with_tag & !PARENT_TAG) as u32;
141 debug_assert!(len <= MAX_LEN);
142 SpanData {
143 lo: BytePos(self.lo),
144 hi: BytePos(self.lo.debug_strict_add(len)),
145 ctxt: SyntaxContext::root(),
146 parent: Some(LocalDefId { local_def_index: DefIndex::from_u16(self.parent) }),
147 }
148 }
149 #[inline]
150 fn span(lo: u32, len: u16, parent: u16) -> Span {
151 let (lo_or_index, len_with_tag_or_marker, ctxt_or_parent_or_marker) =
152 (lo, PARENT_TAG | len, parent);
153 Span { lo_or_index, len_with_tag_or_marker, ctxt_or_parent_or_marker }
154 }
155 #[inline]
156 fn from_span(span: Span) -> InlineParent {
157 let (lo, len_with_tag, parent) =
158 (span.lo_or_index, span.len_with_tag_or_marker, span.ctxt_or_parent_or_marker);
159 InlineParent { lo, len_with_tag, parent }
160 }
161}
162
163impl PartiallyInterned {
164 #[inline]
165 fn data(self) -> SpanData {
166 SpanData {
167 ctxt: SyntaxContext::from_u16(self.ctxt),
168 ..with_span_interner(|interner| interner.spans[self.index as usize])
169 }
170 }
171 #[inline]
172 fn span(index: u32, ctxt: u16) -> Span {
173 let (lo_or_index, len_with_tag_or_marker, ctxt_or_parent_or_marker) =
174 (index, BASE_LEN_INTERNED_MARKER, ctxt);
175 Span { lo_or_index, len_with_tag_or_marker, ctxt_or_parent_or_marker }
176 }
177 #[inline]
178 fn from_span(span: Span) -> PartiallyInterned {
179 PartiallyInterned { index: span.lo_or_index, ctxt: span.ctxt_or_parent_or_marker }
180 }
181}
182
183impl Interned {
184 #[inline]
185 fn data(self) -> SpanData {
186 with_span_interner(|interner| interner.spans[self.index as usize])
187 }
188 #[inline]
189 fn span(index: u32) -> Span {
190 let (lo_or_index, len_with_tag_or_marker, ctxt_or_parent_or_marker) =
191 (index, BASE_LEN_INTERNED_MARKER, CTXT_INTERNED_MARKER);
192 Span { lo_or_index, len_with_tag_or_marker, ctxt_or_parent_or_marker }
193 }
194 #[inline]
195 fn from_span(span: Span) -> Interned {
196 Interned { index: span.lo_or_index }
197 }
198}
199
200macro_rules! match_span_kind {
204 (
205 $span:expr,
206 InlineCtxt($span1:ident) => $arm1:expr,
207 InlineParent($span2:ident) => $arm2:expr,
208 PartiallyInterned($span3:ident) => $arm3:expr,
209 Interned($span4:ident) => $arm4:expr,
210 ) => {
211 if $span.len_with_tag_or_marker != BASE_LEN_INTERNED_MARKER {
212 if $span.len_with_tag_or_marker & PARENT_TAG == 0 {
213 let $span1 = InlineCtxt::from_span($span);
215 $arm1
216 } else {
217 let $span2 = InlineParent::from_span($span);
219 $arm2
220 }
221 } else if $span.ctxt_or_parent_or_marker != CTXT_INTERNED_MARKER {
222 let $span3 = PartiallyInterned::from_span($span);
224 $arm3
225 } else {
226 let $span4 = Interned::from_span($span);
228 $arm4
229 }
230 };
231}
232
233const MAX_LEN: u32 = 0b0111_1111_1111_1110;
236const MAX_CTXT: u32 = 0b0111_1111_1111_1110;
237const PARENT_TAG: u16 = 0b1000_0000_0000_0000;
238const BASE_LEN_INTERNED_MARKER: u16 = 0b1111_1111_1111_1111;
239const CTXT_INTERNED_MARKER: u16 = 0b1111_1111_1111_1111;
240
241pub const DUMMY_SP: Span =
243 Span { lo_or_index: 0, len_with_tag_or_marker: 0, ctxt_or_parent_or_marker: 0 };
244
245impl Span {
246 #[inline]
247 pub fn new(
248 mut lo: BytePos,
249 mut hi: BytePos,
250 ctxt: SyntaxContext,
251 parent: Option<LocalDefId>,
252 ) -> Self {
253 if lo > hi {
254 std::mem::swap(&mut lo, &mut hi);
255 }
256
257 let (len, ctxt32) = (hi.0 - lo.0, ctxt.as_u32());
259 if len <= MAX_LEN && ctxt32 <= MAX_CTXT {
260 match parent {
261 None => return InlineCtxt::span(lo.0, len as u16, ctxt32 as u16),
262 Some(parent) => {
263 let parent32 = parent.local_def_index.as_u32();
264 if ctxt32 == 0 && parent32 <= MAX_CTXT {
265 return InlineParent::span(lo.0, len as u16, parent32 as u16);
266 }
267 }
268 }
269 }
270
271 let index = |ctxt| {
273 with_span_interner(|interner| interner.intern(&SpanData { lo, hi, ctxt, parent }))
274 };
275 if ctxt32 <= MAX_CTXT {
276 PartiallyInterned::span(index(SyntaxContext::from_u32(u32::MAX)), ctxt32 as u16)
278 } else {
279 Interned::span(index(ctxt))
280 }
281 }
282
283 #[inline]
284 pub fn data(self) -> SpanData {
285 let data = self.data_untracked();
286 if let Some(parent) = data.parent {
287 (*SPAN_TRACK)(parent);
288 }
289 data
290 }
291
292 #[inline]
295 pub fn data_untracked(self) -> SpanData {
296 match_span_kind! {
297 self,
298 InlineCtxt(span) => span.data(),
299 InlineParent(span) => span.data(),
300 PartiallyInterned(span) => span.data(),
301 Interned(span) => span.data(),
302 }
303 }
304
305 #[inline]
307 pub fn from_expansion(self) -> bool {
308 let ctxt = match_span_kind! {
309 self,
310 InlineCtxt(span) => SyntaxContext::from_u16(span.ctxt),
318 InlineParent(_span) => SyntaxContext::root(),
319 PartiallyInterned(span) => SyntaxContext::from_u16(span.ctxt),
320 Interned(_span) => SyntaxContext::from_u16(CTXT_INTERNED_MARKER),
321 };
322 !ctxt.is_root()
323 }
324
325 #[inline]
327 pub fn is_dummy(self) -> bool {
328 if self.len_with_tag_or_marker != BASE_LEN_INTERNED_MARKER {
329 let lo = self.lo_or_index;
331 let len = (self.len_with_tag_or_marker & !PARENT_TAG) as u32;
332 debug_assert!(len <= MAX_LEN);
333 lo == 0 && len == 0
334 } else {
335 let index = self.lo_or_index;
337 let data = with_span_interner(|interner| interner.spans[index as usize]);
338 data.lo == BytePos(0) && data.hi == BytePos(0)
339 }
340 }
341
342 #[inline]
343 pub fn map_ctxt(self, map: impl FnOnce(SyntaxContext) -> SyntaxContext) -> Span {
344 let data = match_span_kind! {
345 self,
346 InlineCtxt(span) => {
347 let new_ctxt = map(SyntaxContext::from_u16(span.ctxt));
350 let new_ctxt32 = new_ctxt.as_u32();
351 return if new_ctxt32 <= MAX_CTXT {
352 InlineCtxt::span(span.lo, span.len, new_ctxt32 as u16)
354 } else {
355 span.data().with_ctxt(new_ctxt)
356 };
357 },
358 InlineParent(span) => span.data(),
359 PartiallyInterned(span) => span.data(),
360 Interned(span) => span.data(),
361 };
362
363 data.with_ctxt(map(data.ctxt))
364 }
365
366 #[inline]
369 fn inline_ctxt(self) -> Result<SyntaxContext, usize> {
370 match_span_kind! {
371 self,
372 InlineCtxt(span) => Ok(SyntaxContext::from_u16(span.ctxt)),
373 InlineParent(_span) => Ok(SyntaxContext::root()),
374 PartiallyInterned(span) => Ok(SyntaxContext::from_u16(span.ctxt)),
375 Interned(span) => Err(span.index as usize),
376 }
377 }
378
379 #[cfg_attr(not(test), rustc_diagnostic_item = "SpanCtxt")]
382 #[inline]
383 pub fn ctxt(self) -> SyntaxContext {
384 self.inline_ctxt()
385 .unwrap_or_else(|index| with_span_interner(|interner| interner.spans[index].ctxt))
386 }
387
388 #[inline]
389 pub fn eq_ctxt(self, other: Span) -> bool {
390 match (self.inline_ctxt(), other.inline_ctxt()) {
391 (Ok(ctxt1), Ok(ctxt2)) => ctxt1 == ctxt2,
392 (Ok(_), Err(_)) | (Err(_), Ok(_)) => false,
396 (Err(index1), Err(index2)) => with_span_interner(|interner| {
397 interner.spans[index1].ctxt == interner.spans[index2].ctxt
398 }),
399 }
400 }
401
402 #[inline]
403 pub fn with_parent(self, parent: Option<LocalDefId>) -> Span {
404 let data = match_span_kind! {
405 self,
406 InlineCtxt(span) => {
407 match parent {
411 None => return self,
412 Some(parent) => {
413 let parent32 = parent.local_def_index.as_u32();
414 if span.ctxt == 0 && parent32 <= MAX_CTXT {
415 return InlineParent::span(span.lo, span.len, parent32 as u16);
416 }
417 }
418 }
419 span.data()
420 },
421 InlineParent(span) => span.data(),
422 PartiallyInterned(span) => span.data(),
423 Interned(span) => span.data(),
424 };
425
426 if let Some(old_parent) = data.parent {
427 (*SPAN_TRACK)(old_parent);
428 }
429 data.with_parent(parent)
430 }
431
432 #[inline]
433 pub fn parent(self) -> Option<LocalDefId> {
434 let interned_parent =
435 |index: u32| with_span_interner(|interner| interner.spans[index as usize].parent);
436 match_span_kind! {
437 self,
438 InlineCtxt(_span) => None,
439 InlineParent(span) => Some(LocalDefId { local_def_index: DefIndex::from_u16(span.parent) }),
440 PartiallyInterned(span) => interned_parent(span.index),
441 Interned(span) => interned_parent(span.index),
442 }
443 }
444}
445
446#[derive(Default)]
447pub(crate) struct SpanInterner {
448 spans: FxIndexSet<SpanData>,
449}
450
451impl SpanInterner {
452 fn intern(&mut self, span_data: &SpanData) -> u32 {
453 let (index, _) = self.spans.insert_full(*span_data);
454 index as u32
455 }
456}
457
458#[inline]
460fn with_span_interner<T, F: FnOnce(&mut SpanInterner) -> T>(f: F) -> T {
461 crate::with_session_globals(|session_globals| f(&mut session_globals.span_interner.lock()))
462}