1use std::ops::{Bound, Range};
2
3use ast::token::IdentIsRaw;
4use rustc_ast as ast;
5use rustc_ast::token;
6use rustc_ast::tokenstream::{self, DelimSpacing, Spacing, TokenStream};
7use rustc_ast::util::literal::escape_byte_str_symbol;
8use rustc_ast_pretty::pprust;
9use rustc_data_structures::fx::FxHashMap;
10use rustc_errors::{Diag, ErrorGuaranteed, MultiSpan, PResult};
11use rustc_parse::lexer::nfc_normalize;
12use rustc_parse::parser::Parser;
13use rustc_parse::{exp, new_parser_from_source_str, source_str_to_stream, unwrap_or_emit_fatal};
14use rustc_proc_macro::bridge::{
15 DelimSpan, Diagnostic, ExpnGlobals, Group, Ident, LitKind, Literal, Punct, TokenTree, server,
16};
17use rustc_proc_macro::{Delimiter, Level};
18use rustc_session::parse::ParseSess;
19use rustc_span::def_id::CrateNum;
20use rustc_span::{BytePos, FileName, Pos, Span, Symbol, sym};
21use smallvec::{SmallVec, smallvec};
22
23use crate::base::ExtCtxt;
24
25trait FromInternal<T> {
26 fn from_internal(x: T) -> Self;
27}
28
29trait ToInternal<T> {
30 fn to_internal(self) -> T;
31}
32
33impl FromInternal<token::Delimiter> for Delimiter {
34 fn from_internal(delim: token::Delimiter) -> Delimiter {
35 match delim {
36 token::Delimiter::Parenthesis => Delimiter::Parenthesis,
37 token::Delimiter::Brace => Delimiter::Brace,
38 token::Delimiter::Bracket => Delimiter::Bracket,
39 token::Delimiter::Invisible(_) => Delimiter::None,
40 }
41 }
42}
43
44impl ToInternal<token::Delimiter> for Delimiter {
45 fn to_internal(self) -> token::Delimiter {
46 match self {
47 Delimiter::Parenthesis => token::Delimiter::Parenthesis,
48 Delimiter::Brace => token::Delimiter::Brace,
49 Delimiter::Bracket => token::Delimiter::Bracket,
50 Delimiter::None => token::Delimiter::Invisible(token::InvisibleOrigin::ProcMacro),
51 }
52 }
53}
54
55impl FromInternal<token::LitKind> for LitKind {
56 fn from_internal(kind: token::LitKind) -> Self {
57 match kind {
58 token::Byte => LitKind::Byte,
59 token::Char => LitKind::Char,
60 token::Integer => LitKind::Integer,
61 token::Float => LitKind::Float,
62 token::Str => LitKind::Str,
63 token::StrRaw(n) => LitKind::StrRaw(n),
64 token::ByteStr => LitKind::ByteStr,
65 token::ByteStrRaw(n) => LitKind::ByteStrRaw(n),
66 token::CStr => LitKind::CStr,
67 token::CStrRaw(n) => LitKind::CStrRaw(n),
68 token::Err(_guar) => {
69 LitKind::ErrWithGuar
73 }
74 token::Bool => unreachable!(),
75 }
76 }
77}
78
79impl ToInternal<token::LitKind> for LitKind {
80 fn to_internal(self) -> token::LitKind {
81 match self {
82 LitKind::Byte => token::Byte,
83 LitKind::Char => token::Char,
84 LitKind::Integer => token::Integer,
85 LitKind::Float => token::Float,
86 LitKind::Str => token::Str,
87 LitKind::StrRaw(n) => token::StrRaw(n),
88 LitKind::ByteStr => token::ByteStr,
89 LitKind::ByteStrRaw(n) => token::ByteStrRaw(n),
90 LitKind::CStr => token::CStr,
91 LitKind::CStrRaw(n) => token::CStrRaw(n),
92 LitKind::ErrWithGuar => {
93 #[allow(deprecated)]
99 let guar = ErrorGuaranteed::unchecked_error_guaranteed();
100 token::Err(guar)
101 }
102 }
103 }
104}
105
106impl FromInternal<(TokenStream, &mut Rustc<'_, '_>)> for Vec<TokenTree<TokenStream, Span, Symbol>> {
107 fn from_internal((stream, rustc): (TokenStream, &mut Rustc<'_, '_>)) -> Self {
108 use rustc_ast::token::*;
109
110 let mut trees = Vec::with_capacity(stream.len().next_power_of_two());
113 let mut iter = stream.iter();
114
115 while let Some(tree) = iter.next() {
116 let (Token { kind, span }, joint) = match tree.clone() {
117 tokenstream::TokenTree::Delimited(span, _, mut delim, mut stream) => {
118 if let Delimiter::Invisible(InvisibleOrigin::MetaVar(kind)) = delim {
127 crate::base::stream_pretty_printing_compatibility_hack(
128 kind,
129 &stream,
130 rustc.psess(),
131 );
132 }
133
134 while let Delimiter::Invisible(InvisibleOrigin::MetaVar(_)) = delim {
139 if stream.len() == 1
140 && let tree = stream.iter().next().unwrap()
141 && let tokenstream::TokenTree::Delimited(_, _, delim2, stream2) = tree
142 && let Delimiter::Invisible(InvisibleOrigin::MetaVar(_)) = delim2
143 {
144 delim = *delim2;
145 stream = stream2.clone();
146 } else {
147 break;
148 }
149 }
150
151 trees.push(TokenTree::Group(Group {
152 delimiter: rustc_proc_macro::Delimiter::from_internal(delim),
153 stream: Some(stream),
154 span: DelimSpan {
155 open: span.open,
156 close: span.close,
157 entire: span.entire(),
158 },
159 }));
160 continue;
161 }
162 tokenstream::TokenTree::Token(token, spacing) => {
163 let joint = match spacing {
173 Spacing::Alone | Spacing::JointHidden => false,
174 Spacing::Joint => true,
175 };
176 (token, joint)
177 }
178 };
179
180 let mut op = |s: &str| {
184 assert!(s.is_ascii());
185 trees.extend(s.bytes().enumerate().map(|(i, ch)| {
186 let is_final = i == s.len() - 1;
187 let span = if (span.hi() - span.lo()).to_usize() == s.len() {
193 let lo = span.lo() + BytePos::from_usize(i);
194 let hi = lo + BytePos::from_usize(1);
195 span.with_lo(lo).with_hi(hi)
196 } else {
197 span
198 };
199 let joint = if is_final { joint } else { true };
200 TokenTree::Punct(Punct { ch, joint, span })
201 }));
202 };
203
204 match kind {
205 Eq => op("="),
206 Lt => op("<"),
207 Le => op("<="),
208 EqEq => op("=="),
209 Ne => op("!="),
210 Ge => op(">="),
211 Gt => op(">"),
212 AndAnd => op("&&"),
213 OrOr => op("||"),
214 Bang => op("!"),
215 Tilde => op("~"),
216 Plus => op("+"),
217 Minus => op("-"),
218 Star => op("*"),
219 Slash => op("/"),
220 Percent => op("%"),
221 Caret => op("^"),
222 And => op("&"),
223 Or => op("|"),
224 Shl => op("<<"),
225 Shr => op(">>"),
226 PlusEq => op("+="),
227 MinusEq => op("-="),
228 StarEq => op("*="),
229 SlashEq => op("/="),
230 PercentEq => op("%="),
231 CaretEq => op("^="),
232 AndEq => op("&="),
233 OrEq => op("|="),
234 ShlEq => op("<<="),
235 ShrEq => op(">>="),
236 At => op("@"),
237 Dot => op("."),
238 DotDot => op(".."),
239 DotDotDot => op("..."),
240 DotDotEq => op("..="),
241 Comma => op(","),
242 Semi => op(";"),
243 Colon => op(":"),
244 PathSep => op("::"),
245 RArrow => op("->"),
246 LArrow => op("<-"),
247 FatArrow => op("=>"),
248 Pound => op("#"),
249 Dollar => op("$"),
250 Question => op("?"),
251 SingleQuote => op("'"),
252
253 Ident(sym, is_raw) => trees.push(TokenTree::Ident(Ident {
254 sym,
255 is_raw: matches!(is_raw, IdentIsRaw::Yes),
256 span,
257 })),
258 NtIdent(ident, is_raw) => trees.push(TokenTree::Ident(Ident {
259 sym: ident.name,
260 is_raw: matches!(is_raw, IdentIsRaw::Yes),
261 span: ident.span,
262 })),
263
264 Lifetime(name, is_raw) => {
265 let ident = rustc_span::Ident::new(name, span).without_first_quote();
266 trees.extend([
267 TokenTree::Punct(Punct { ch: b'\'', joint: true, span }),
268 TokenTree::Ident(Ident {
269 sym: ident.name,
270 is_raw: matches!(is_raw, IdentIsRaw::Yes),
271 span,
272 }),
273 ]);
274 }
275 NtLifetime(ident, is_raw) => {
276 let stream =
277 TokenStream::token_alone(token::Lifetime(ident.name, is_raw), ident.span);
278 trees.push(TokenTree::Group(Group {
279 delimiter: rustc_proc_macro::Delimiter::None,
280 stream: Some(stream),
281 span: DelimSpan::from_single(span),
282 }))
283 }
284
285 Literal(token::Lit { kind, symbol, suffix }) => {
286 trees.push(TokenTree::Literal(self::Literal {
287 kind: FromInternal::from_internal(kind),
288 symbol,
289 suffix,
290 span,
291 }));
292 }
293 DocComment(_, attr_style, data) => {
294 let mut escaped = String::new();
295 for ch in data.as_str().chars() {
296 escaped.extend(ch.escape_debug());
297 }
298 let stream = [
299 Ident(sym::doc, IdentIsRaw::No),
300 Eq,
301 TokenKind::lit(token::Str, Symbol::intern(&escaped), None),
302 ]
303 .into_iter()
304 .map(|kind| tokenstream::TokenTree::token_alone(kind, span))
305 .collect();
306 trees.push(TokenTree::Punct(Punct { ch: b'#', joint: false, span }));
307 if attr_style == ast::AttrStyle::Inner {
308 trees.push(TokenTree::Punct(Punct { ch: b'!', joint: false, span }));
309 }
310 trees.push(TokenTree::Group(Group {
311 delimiter: rustc_proc_macro::Delimiter::Bracket,
312 stream: Some(stream),
313 span: DelimSpan::from_single(span),
314 }));
315 }
316
317 OpenParen | CloseParen | OpenBrace | CloseBrace | OpenBracket | CloseBracket
318 | OpenInvisible(_) | CloseInvisible(_) | Eof => unreachable!(),
319 }
320 }
321 trees
322 }
323}
324
325impl ToInternal<SmallVec<[tokenstream::TokenTree; 2]>>
327 for (TokenTree<TokenStream, Span, Symbol>, &mut Rustc<'_, '_>)
328{
329 fn to_internal(self) -> SmallVec<[tokenstream::TokenTree; 2]> {
330 use rustc_ast::token::*;
331
332 let (tree, rustc) = self;
338 match tree {
339 TokenTree::Punct(Punct { ch, joint, span }) => {
340 let kind = match ch {
341 b'=' => Eq,
342 b'<' => Lt,
343 b'>' => Gt,
344 b'!' => Bang,
345 b'~' => Tilde,
346 b'+' => Plus,
347 b'-' => Minus,
348 b'*' => Star,
349 b'/' => Slash,
350 b'%' => Percent,
351 b'^' => Caret,
352 b'&' => And,
353 b'|' => Or,
354 b'@' => At,
355 b'.' => Dot,
356 b',' => Comma,
357 b';' => Semi,
358 b':' => Colon,
359 b'#' => Pound,
360 b'$' => Dollar,
361 b'?' => Question,
362 b'\'' => SingleQuote,
363 _ => unreachable!(),
364 };
365 smallvec![if joint {
371 tokenstream::TokenTree::token_joint(kind, span)
372 } else {
373 tokenstream::TokenTree::token_alone(kind, span)
374 }]
375 }
376 TokenTree::Group(Group { delimiter, stream, span: DelimSpan { open, close, .. } }) => {
377 smallvec![tokenstream::TokenTree::Delimited(
378 tokenstream::DelimSpan { open, close },
379 DelimSpacing::new(Spacing::Alone, Spacing::Alone),
380 delimiter.to_internal(),
381 stream.unwrap_or_default(),
382 )]
383 }
384 TokenTree::Ident(self::Ident { sym, is_raw, span }) => {
385 rustc.psess().symbol_gallery.insert(sym, span);
386 smallvec![tokenstream::TokenTree::token_alone(Ident(sym, is_raw.into()), span)]
387 }
388 TokenTree::Literal(self::Literal {
389 kind: self::LitKind::Integer,
390 symbol,
391 suffix,
392 span,
393 }) if let Some(symbol) = symbol.as_str().strip_prefix('-') => {
394 let symbol = Symbol::intern(symbol);
395 let integer = TokenKind::lit(token::Integer, symbol, suffix);
396 let a = tokenstream::TokenTree::token_joint_hidden(Minus, span);
397 let b = tokenstream::TokenTree::token_alone(integer, span);
398 smallvec![a, b]
399 }
400 TokenTree::Literal(self::Literal {
401 kind: self::LitKind::Float,
402 symbol,
403 suffix,
404 span,
405 }) if let Some(symbol) = symbol.as_str().strip_prefix('-') => {
406 let symbol = Symbol::intern(symbol);
407 let float = TokenKind::lit(token::Float, symbol, suffix);
408 let a = tokenstream::TokenTree::token_joint_hidden(Minus, span);
409 let b = tokenstream::TokenTree::token_alone(float, span);
410 smallvec![a, b]
411 }
412 TokenTree::Literal(self::Literal { kind, symbol, suffix, span }) => {
413 smallvec![tokenstream::TokenTree::token_alone(
414 TokenKind::lit(kind.to_internal(), symbol, suffix),
415 span,
416 )]
417 }
418 }
419 }
420}
421
422impl ToInternal<rustc_errors::Level> for Level {
423 fn to_internal(self) -> rustc_errors::Level {
424 match self {
425 Level::Error => rustc_errors::Level::Error,
426 Level::Warning => rustc_errors::Level::Warning,
427 Level::Note => rustc_errors::Level::Note,
428 Level::Help => rustc_errors::Level::Help,
429 _ => unreachable!("unknown proc_macro::Level variant: {:?}", self),
430 }
431 }
432}
433
434pub(crate) struct FreeFunctions;
435
436pub(crate) struct Rustc<'a, 'b> {
437 ecx: &'a mut ExtCtxt<'b>,
438 def_site: Span,
439 call_site: Span,
440 mixed_site: Span,
441 krate: CrateNum,
442 rebased_spans: FxHashMap<usize, Span>,
443}
444
445impl<'a, 'b> Rustc<'a, 'b> {
446 pub(crate) fn new(ecx: &'a mut ExtCtxt<'b>) -> Self {
447 let expn_data = ecx.current_expansion.id.expn_data();
448 Rustc {
449 def_site: ecx.with_def_site_ctxt(expn_data.def_site),
450 call_site: ecx.with_call_site_ctxt(expn_data.call_site),
451 mixed_site: ecx.with_mixed_site_ctxt(expn_data.call_site),
452 krate: expn_data.macro_def_id.unwrap().krate,
453 rebased_spans: FxHashMap::default(),
454 ecx,
455 }
456 }
457
458 fn psess(&self) -> &ParseSess {
459 self.ecx.psess()
460 }
461}
462
463impl server::Types for Rustc<'_, '_> {
464 type FreeFunctions = FreeFunctions;
465 type TokenStream = TokenStream;
466 type Span = Span;
467 type Symbol = Symbol;
468}
469
470impl server::FreeFunctions for Rustc<'_, '_> {
471 fn injected_env_var(&mut self, var: &str) -> Option<String> {
472 self.ecx.sess.opts.logical_env.get(var).cloned()
473 }
474
475 fn track_env_var(&mut self, var: &str, value: Option<&str>) {
476 self.psess()
477 .env_depinfo
478 .borrow_mut()
479 .insert((Symbol::intern(var), value.map(Symbol::intern)));
480 }
481
482 fn track_path(&mut self, path: &str) {
483 self.psess().file_depinfo.borrow_mut().insert(Symbol::intern(path));
484 }
485
486 fn literal_from_str(&mut self, s: &str) -> Result<Literal<Self::Span, Self::Symbol>, ()> {
487 let name = FileName::proc_macro_source_code(s);
488 let mut parser =
489 unwrap_or_emit_fatal(new_parser_from_source_str(self.psess(), name, s.to_owned()));
490
491 let first_span = parser.token.span.data();
492 let minus_present = parser.eat(exp!(Minus));
493
494 let lit_span = parser.token.span.data();
495 let token::Literal(mut lit) = parser.token.kind else {
496 return Err(());
497 };
498
499 if (lit_span.hi.0 - first_span.lo.0) as usize != s.len() {
502 return Err(());
503 }
504
505 if minus_present {
506 if first_span.hi.0 != lit_span.lo.0 {
509 return Err(());
510 }
511
512 match lit.kind {
514 token::LitKind::Bool
515 | token::LitKind::Byte
516 | token::LitKind::Char
517 | token::LitKind::Str
518 | token::LitKind::StrRaw(_)
519 | token::LitKind::ByteStr
520 | token::LitKind::ByteStrRaw(_)
521 | token::LitKind::CStr
522 | token::LitKind::CStrRaw(_)
523 | token::LitKind::Err(_) => return Err(()),
524 token::LitKind::Integer | token::LitKind::Float => {}
525 }
526
527 let symbol = Symbol::intern(&s[..1 + lit.symbol.as_str().len()]);
529 lit = token::Lit::new(lit.kind, symbol, lit.suffix);
530 }
531 let token::Lit { kind, symbol, suffix } = lit;
532 Ok(Literal {
533 kind: FromInternal::from_internal(kind),
534 symbol,
535 suffix,
536 span: self.call_site,
537 })
538 }
539
540 fn emit_diagnostic(&mut self, diagnostic: Diagnostic<Self::Span>) {
541 let message = rustc_errors::DiagMessage::from(diagnostic.message);
542 let mut diag: Diag<'_, ()> =
543 Diag::new(self.psess().dcx(), diagnostic.level.to_internal(), message);
544 diag.span(MultiSpan::from_spans(diagnostic.spans));
545 for child in diagnostic.children {
546 #[allow(rustc::untranslatable_diagnostic)]
549 diag.sub(child.level.to_internal(), child.message, MultiSpan::from_spans(child.spans));
550 }
551 diag.emit();
552 }
553}
554
555impl server::TokenStream for Rustc<'_, '_> {
556 fn is_empty(&mut self, stream: &Self::TokenStream) -> bool {
557 stream.is_empty()
558 }
559
560 fn from_str(&mut self, src: &str) -> Self::TokenStream {
561 unwrap_or_emit_fatal(source_str_to_stream(
562 self.psess(),
563 FileName::proc_macro_source_code(src),
564 src.to_string(),
565 Some(self.call_site),
566 ))
567 }
568
569 fn to_string(&mut self, stream: &Self::TokenStream) -> String {
570 pprust::tts_to_string(stream)
571 }
572
573 fn expand_expr(&mut self, stream: &Self::TokenStream) -> Result<Self::TokenStream, ()> {
574 let expr: PResult<'_, _> = try {
576 let mut p = Parser::new(self.psess(), stream.clone(), Some("proc_macro expand expr"));
577 let expr = p.parse_expr()?;
578 if p.token != token::Eof {
579 p.unexpected()?;
580 }
581 expr
582 };
583 let expr = expr.map_err(|err| {
584 err.emit();
585 })?;
586
587 let expr = self
589 .ecx
590 .expander()
591 .fully_expand_fragment(crate::expand::AstFragment::Expr(expr))
592 .make_expr();
593
594 match &expr.kind {
599 ast::ExprKind::Lit(token_lit) if token_lit.kind == token::Bool => {
600 Ok(tokenstream::TokenStream::token_alone(
601 token::Ident(token_lit.symbol, IdentIsRaw::No),
602 expr.span,
603 ))
604 }
605 ast::ExprKind::Lit(token_lit) => {
606 Ok(tokenstream::TokenStream::token_alone(token::Literal(*token_lit), expr.span))
607 }
608 ast::ExprKind::IncludedBytes(byte_sym) => {
609 let lit = token::Lit::new(
610 token::ByteStr,
611 escape_byte_str_symbol(byte_sym.as_byte_str()),
612 None,
613 );
614 Ok(tokenstream::TokenStream::token_alone(token::TokenKind::Literal(lit), expr.span))
615 }
616 ast::ExprKind::Unary(ast::UnOp::Neg, e) => match &e.kind {
617 ast::ExprKind::Lit(token_lit) => match token_lit {
618 token::Lit { kind: token::Integer | token::Float, .. } => {
619 Ok(Self::TokenStream::from_iter([
620 tokenstream::TokenTree::token_joint_hidden(token::Minus, e.span),
623 tokenstream::TokenTree::token_alone(token::Literal(*token_lit), e.span),
624 ]))
625 }
626 _ => Err(()),
627 },
628 _ => Err(()),
629 },
630 _ => Err(()),
631 }
632 }
633
634 fn from_token_tree(
635 &mut self,
636 tree: TokenTree<Self::TokenStream, Self::Span, Self::Symbol>,
637 ) -> Self::TokenStream {
638 Self::TokenStream::new((tree, &mut *self).to_internal().into_iter().collect::<Vec<_>>())
639 }
640
641 fn concat_trees(
642 &mut self,
643 base: Option<Self::TokenStream>,
644 trees: Vec<TokenTree<Self::TokenStream, Self::Span, Self::Symbol>>,
645 ) -> Self::TokenStream {
646 let mut stream = base.unwrap_or_default();
647 for tree in trees {
648 for tt in (tree, &mut *self).to_internal() {
649 stream.push_tree(tt);
650 }
651 }
652 stream
653 }
654
655 fn concat_streams(
656 &mut self,
657 base: Option<Self::TokenStream>,
658 streams: Vec<Self::TokenStream>,
659 ) -> Self::TokenStream {
660 let mut stream = base.unwrap_or_default();
661 for s in streams {
662 stream.push_stream(s);
663 }
664 stream
665 }
666
667 fn into_trees(
668 &mut self,
669 stream: Self::TokenStream,
670 ) -> Vec<TokenTree<Self::TokenStream, Self::Span, Self::Symbol>> {
671 FromInternal::from_internal((stream, self))
672 }
673}
674
675impl server::Span for Rustc<'_, '_> {
676 fn debug(&mut self, span: Self::Span) -> String {
677 if self.ecx.ecfg.span_debug {
678 format!("{span:?}")
679 } else {
680 format!("{:?} bytes({}..{})", span.ctxt(), span.lo().0, span.hi().0)
681 }
682 }
683
684 fn file(&mut self, span: Self::Span) -> String {
685 self.psess()
686 .source_map()
687 .lookup_char_pos(span.lo())
688 .file
689 .name
690 .prefer_remapped_unconditionally()
691 .to_string()
692 }
693
694 fn local_file(&mut self, span: Self::Span) -> Option<String> {
695 self.psess()
696 .source_map()
697 .lookup_char_pos(span.lo())
698 .file
699 .name
700 .clone()
701 .into_local_path()
702 .map(|p| {
703 p.to_str()
704 .expect("non-UTF8 file path in `proc_macro::SourceFile::path`")
705 .to_string()
706 })
707 }
708
709 fn parent(&mut self, span: Self::Span) -> Option<Self::Span> {
710 span.parent_callsite()
711 }
712
713 fn source(&mut self, span: Self::Span) -> Self::Span {
714 span.source_callsite()
715 }
716
717 fn byte_range(&mut self, span: Self::Span) -> Range<usize> {
718 let source_map = self.psess().source_map();
719
720 let relative_start_pos = source_map.lookup_byte_offset(span.lo()).pos;
721 let relative_end_pos = source_map.lookup_byte_offset(span.hi()).pos;
722
723 Range { start: relative_start_pos.0 as usize, end: relative_end_pos.0 as usize }
724 }
725 fn start(&mut self, span: Self::Span) -> Self::Span {
726 span.shrink_to_lo()
727 }
728
729 fn end(&mut self, span: Self::Span) -> Self::Span {
730 span.shrink_to_hi()
731 }
732
733 fn line(&mut self, span: Self::Span) -> usize {
734 let loc = self.psess().source_map().lookup_char_pos(span.lo());
735 loc.line
736 }
737
738 fn column(&mut self, span: Self::Span) -> usize {
739 let loc = self.psess().source_map().lookup_char_pos(span.lo());
740 loc.col.to_usize() + 1
741 }
742
743 fn join(&mut self, first: Self::Span, second: Self::Span) -> Option<Self::Span> {
744 let self_loc = self.psess().source_map().lookup_char_pos(first.lo());
745 let other_loc = self.psess().source_map().lookup_char_pos(second.lo());
746
747 if self_loc.file.name != other_loc.file.name {
748 return None;
749 }
750
751 Some(first.to(second))
752 }
753
754 fn subspan(
755 &mut self,
756 span: Self::Span,
757 start: Bound<usize>,
758 end: Bound<usize>,
759 ) -> Option<Self::Span> {
760 let length = span.hi().to_usize() - span.lo().to_usize();
761
762 let start = match start {
763 Bound::Included(lo) => lo,
764 Bound::Excluded(lo) => lo.checked_add(1)?,
765 Bound::Unbounded => 0,
766 };
767
768 let end = match end {
769 Bound::Included(hi) => hi.checked_add(1)?,
770 Bound::Excluded(hi) => hi,
771 Bound::Unbounded => length,
772 };
773
774 if start > u32::MAX as usize
776 || end > u32::MAX as usize
777 || (u32::MAX - start as u32) < span.lo().to_u32()
778 || (u32::MAX - end as u32) < span.lo().to_u32()
779 || start >= end
780 || end > length
781 {
782 return None;
783 }
784
785 let new_lo = span.lo() + BytePos::from_usize(start);
786 let new_hi = span.lo() + BytePos::from_usize(end);
787 Some(span.with_lo(new_lo).with_hi(new_hi))
788 }
789
790 fn resolved_at(&mut self, span: Self::Span, at: Self::Span) -> Self::Span {
791 span.with_ctxt(at.ctxt())
792 }
793
794 fn source_text(&mut self, span: Self::Span) -> Option<String> {
795 self.psess().source_map().span_to_snippet(span).ok()
796 }
797
798 fn save_span(&mut self, span: Self::Span) -> usize {
823 self.psess().save_proc_macro_span(span)
824 }
825
826 fn recover_proc_macro_span(&mut self, id: usize) -> Self::Span {
827 let (resolver, krate, def_site) = (&*self.ecx.resolver, self.krate, self.def_site);
828 *self.rebased_spans.entry(id).or_insert_with(|| {
829 resolver.get_proc_macro_quoted_span(krate, id).with_ctxt(def_site.ctxt())
832 })
833 }
834}
835
836impl server::Symbol for Rustc<'_, '_> {
837 fn normalize_and_validate_ident(&mut self, string: &str) -> Result<Self::Symbol, ()> {
838 let sym = nfc_normalize(string);
839 if rustc_lexer::is_ident(sym.as_str()) { Ok(sym) } else { Err(()) }
840 }
841}
842
843impl server::Server for Rustc<'_, '_> {
844 fn globals(&mut self) -> ExpnGlobals<Self::Span> {
845 ExpnGlobals {
846 def_site: self.def_site,
847 call_site: self.call_site,
848 mixed_site: self.mixed_site,
849 }
850 }
851
852 fn intern_symbol(string: &str) -> Self::Symbol {
853 Symbol::intern(string)
854 }
855
856 fn with_symbol_string(symbol: &Self::Symbol, f: impl FnOnce(&str)) {
857 f(symbol.as_str())
858 }
859}