1//! A support library for macro authors when defining new macros.
2//!
3//! This library, provided by the standard distribution, provides the types
4//! consumed in the interfaces of procedurally defined macro definitions such as
5//! function-like macros `#[proc_macro]`, macro attributes `#[proc_macro_attribute]` and
6//! custom derive attributes `#[proc_macro_derive]`.
7//!
8//! See [the book] for more.
9//!
10//! [the book]: ../book/ch19-06-macros.html#procedural-macros-for-generating-code-from-attributes
1112#![stable(feature = "proc_macro_lib", since = "1.15.0")]
13#![deny(missing_docs)]
14#![doc(
15 html_playground_url = "https://play.rust-lang.org/",
16 issue_tracker_base_url = "https://github.com/rust-lang/rust/issues/",
17 test(no_crate_inject, attr(deny(warnings))),
18 test(attr(allow(dead_code, deprecated, unused_variables, unused_mut)))
19)]
20#![doc(rust_logo)]
21#![feature(rustdoc_internals)]
22#![feature(staged_api)]
23#![feature(allow_internal_unstable)]
24#![feature(decl_macro)]
25#![feature(negative_impls)]
26#![feature(panic_can_unwind)]
27#![feature(restricted_std)]
28#![feature(rustc_attrs)]
29#![feature(extend_one)]
30#![feature(mem_conjure_zst)]
31#![feature(f16)]
32#![recursion_limit = "256"]
33#![allow(internal_features)]
34#![deny(ffi_unwind_calls)]
35#![allow(rustc::internal)] // Can't use FxHashMap when compiled as part of the standard library
36#![warn(rustdoc::unescaped_backticks)]
37#![warn(unreachable_pub)]
38#![deny(unsafe_op_in_unsafe_fn)]
3940#[unstable(feature = "proc_macro_internals", issue = "27812")]
41#[doc(hidden)]
42pub mod bridge;
4344mod diagnostic;
45mod escape;
46mod to_tokens;
4748use core::convert::From;
49use core::ops::BitOr;
50use std::borrow::Cow;
51use std::ffi::CStr;
52use std::ops::{Range, RangeBounds};
53use std::path::PathBuf;
54use std::str::FromStr;
55use std::{error, fmt};
5657#[unstable(feature = "proc_macro_diagnostic", issue = "54140")]
58pub use diagnostic::{Diagnostic, Level, MultiSpan};
59use rustc_literal_escaper::{
60MixedUnit, unescape_byte, unescape_byte_str, unescape_c_str, unescape_char, unescape_str,
61};
62#[unstable(feature = "proc_macro_totokens", issue = "130977")]
63pub use to_tokens::ToTokens;
6465use crate::bridge::client::Methodsas BridgeMethods;
66use crate::escape::{EscapeOptions, escape_bytes};
6768/// Mostly relating to malformed escape sequences, but also a few other problems.
69#[unstable(feature = "proc_macro_value", issue = "136652")]
70#[derive(#[automatically_derived]
#[unstable(feature = "proc_macro_value", issue = "136652")]
impl ::core::fmt::Debug for EscapeError {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
EscapeError::ZeroChars => "ZeroChars",
EscapeError::MoreThanOneChar => "MoreThanOneChar",
EscapeError::LoneSlash => "LoneSlash",
EscapeError::InvalidEscape => "InvalidEscape",
EscapeError::BareCarriageReturn => "BareCarriageReturn",
EscapeError::BareCarriageReturnInRawString =>
"BareCarriageReturnInRawString",
EscapeError::EscapeOnlyChar => "EscapeOnlyChar",
EscapeError::TooShortHexEscape => "TooShortHexEscape",
EscapeError::InvalidCharInHexEscape =>
"InvalidCharInHexEscape",
EscapeError::OutOfRangeHexEscape => "OutOfRangeHexEscape",
EscapeError::NoBraceInUnicodeEscape =>
"NoBraceInUnicodeEscape",
EscapeError::InvalidCharInUnicodeEscape =>
"InvalidCharInUnicodeEscape",
EscapeError::EmptyUnicodeEscape => "EmptyUnicodeEscape",
EscapeError::UnclosedUnicodeEscape => "UnclosedUnicodeEscape",
EscapeError::LeadingUnderscoreUnicodeEscape =>
"LeadingUnderscoreUnicodeEscape",
EscapeError::OverlongUnicodeEscape => "OverlongUnicodeEscape",
EscapeError::LoneSurrogateUnicodeEscape =>
"LoneSurrogateUnicodeEscape",
EscapeError::OutOfRangeUnicodeEscape =>
"OutOfRangeUnicodeEscape",
EscapeError::UnicodeEscapeInByte => "UnicodeEscapeInByte",
EscapeError::NonAsciiCharInByte => "NonAsciiCharInByte",
EscapeError::NulInCStr => "NulInCStr",
EscapeError::UnskippedWhitespaceWarning =>
"UnskippedWhitespaceWarning",
EscapeError::MultipleSkippedLinesWarning =>
"MultipleSkippedLinesWarning",
})
}
}Debug, #[automatically_derived]
#[unstable(feature = "proc_macro_value", issue = "136652")]
impl ::core::cmp::PartialEq for EscapeError {
#[inline]
fn eq(&self, other: &EscapeError) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
#[unstable(feature = "proc_macro_value", issue = "136652")]
impl ::core::cmp::Eq for EscapeError {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
71#[non_exhaustive]
72pub enum EscapeError {
73/// Expected 1 char, but 0 were found.
74ZeroChars,
75/// Expected 1 char, but more than 1 were found.
76MoreThanOneChar,
7778/// Escaped '\' character without continuation.
79LoneSlash,
80/// Invalid escape character (e.g. '\z').
81InvalidEscape,
82/// Raw '\r' encountered.
83BareCarriageReturn,
84/// Raw '\r' encountered in raw string.
85BareCarriageReturnInRawString,
86/// Unescaped character that was expected to be escaped (e.g. raw '\t').
87EscapeOnlyChar,
8889/// Numeric character escape is too short (e.g. '\x1').
90TooShortHexEscape,
91/// Invalid character in numeric escape (e.g. '\xz')
92InvalidCharInHexEscape,
93/// Character code in numeric escape is non-ascii (e.g. '\xFF').
94OutOfRangeHexEscape,
9596/// '\u' not followed by '{'.
97NoBraceInUnicodeEscape,
98/// Non-hexadecimal value in '\u{..}'.
99InvalidCharInUnicodeEscape,
100/// '\u{}'
101EmptyUnicodeEscape,
102/// No closing brace in '\u{..}', e.g. '\u{12'.
103UnclosedUnicodeEscape,
104/// '\u{_12}'
105LeadingUnderscoreUnicodeEscape,
106/// More than 6 characters in '\u{..}', e.g. '\u{10FFFF_FF}'
107OverlongUnicodeEscape,
108/// Invalid in-bound unicode character code, e.g. '\u{DFFF}'.
109LoneSurrogateUnicodeEscape,
110/// Out of bounds unicode character code, e.g. '\u{FFFFFF}'.
111OutOfRangeUnicodeEscape,
112113/// Unicode escape code in byte literal.
114UnicodeEscapeInByte,
115/// Non-ascii character in byte literal, byte string literal, or raw byte string literal.
116NonAsciiCharInByte,
117118/// `\0` in a C string literal.
119NulInCStr,
120121/// After a line ending with '\', the next line contains whitespace
122 /// characters that are not skipped.
123UnskippedWhitespaceWarning,
124125/// After a line ending with '\', multiple lines are skipped.
126MultipleSkippedLinesWarning,
127}
128129#[unstable(feature = "proc_macro_value", issue = "136652")]
130#[doc(hidden)]
131impl From<rustc_literal_escaper::EscapeError> for EscapeError {
132fn from(value: rustc_literal_escaper::EscapeError) -> Self {
133use rustc_literal_escaper::EscapeErroras EE;
134135match value {
136 EE::ZeroChars => Self::ZeroChars,
137 EE::MoreThanOneChar => Self::MoreThanOneChar,
138 EE::LoneSlash => Self::LoneSlash,
139 EE::InvalidEscape => Self::InvalidEscape,
140 EE::BareCarriageReturn => Self::BareCarriageReturn,
141 EE::BareCarriageReturnInRawString => Self::BareCarriageReturnInRawString,
142 EE::EscapeOnlyChar => Self::EscapeOnlyChar,
143 EE::TooShortHexEscape => Self::TooShortHexEscape,
144 EE::InvalidCharInHexEscape => Self::InvalidCharInHexEscape,
145 EE::OutOfRangeHexEscape => Self::OutOfRangeHexEscape,
146 EE::NoBraceInUnicodeEscape => Self::NoBraceInUnicodeEscape,
147 EE::InvalidCharInUnicodeEscape => Self::InvalidCharInUnicodeEscape,
148 EE::EmptyUnicodeEscape => Self::EmptyUnicodeEscape,
149 EE::UnclosedUnicodeEscape => Self::UnclosedUnicodeEscape,
150 EE::LeadingUnderscoreUnicodeEscape => Self::LeadingUnderscoreUnicodeEscape,
151 EE::OverlongUnicodeEscape => Self::OverlongUnicodeEscape,
152 EE::LoneSurrogateUnicodeEscape => Self::LoneSurrogateUnicodeEscape,
153 EE::OutOfRangeUnicodeEscape => Self::OutOfRangeUnicodeEscape,
154 EE::UnicodeEscapeInByte => Self::UnicodeEscapeInByte,
155 EE::NonAsciiCharInByte => Self::NonAsciiCharInByte,
156 EE::NulInCStr => Self::NulInCStr,
157 EE::UnskippedWhitespaceWarning => Self::UnskippedWhitespaceWarning,
158 EE::MultipleSkippedLinesWarning => Self::MultipleSkippedLinesWarning,
159 }
160 }
161}
162163#[unstable(feature = "proc_macro_value", issue = "136652")]
164impl error::Errorfor EscapeError {}
165166#[unstable(feature = "proc_macro_value", issue = "136652")]
167impl fmt::Displayfor EscapeError {
168fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
169f.write_str(match self {
170Self::ZeroChars => "zero chars",
171Self::MoreThanOneChar => "more than one char",
172Self::LoneSlash => "lone slash",
173Self::InvalidEscape => "invalid escape",
174Self::BareCarriageReturn => "bare carriage return",
175Self::BareCarriageReturnInRawString => "bare carriage return in raw string",
176Self::EscapeOnlyChar => "escape only char",
177Self::TooShortHexEscape => "too short hex escape",
178Self::InvalidCharInHexEscape => "invalid char in hex escape",
179Self::OutOfRangeHexEscape => "out of range hex escape",
180Self::NoBraceInUnicodeEscape => "no brace in unicode escape",
181Self::InvalidCharInUnicodeEscape => "invalid char in unicode escape",
182Self::EmptyUnicodeEscape => "empty unicode escape",
183Self::UnclosedUnicodeEscape => "unclosed unicode escape",
184Self::LeadingUnderscoreUnicodeEscape => "leading underscore unicode escape",
185Self::OverlongUnicodeEscape => "overlong unicode escape",
186Self::LoneSurrogateUnicodeEscape => "lone surrogate unicode escape",
187Self::OutOfRangeUnicodeEscape => "out of range unicode escape",
188Self::UnicodeEscapeInByte => "unicode escape in byte",
189Self::NonAsciiCharInByte => "non ascii char in byte",
190Self::NulInCStr => "nul in CStr",
191Self::UnskippedWhitespaceWarning => "unskipped whitespace warning",
192Self::MultipleSkippedLinesWarning => "multiple skipped lines warning",
193 })
194 }
195}
196197/// Errors returned when trying to retrieve a literal unescaped value.
198#[unstable(feature = "proc_macro_value", issue = "136652")]
199#[derive(#[automatically_derived]
#[unstable(feature = "proc_macro_value", issue = "136652")]
impl ::core::fmt::Debug for ConversionErrorKind {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
ConversionErrorKind::FailedToUnescape(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"FailedToUnescape", &__self_0),
ConversionErrorKind::InvalidLiteralKind =>
::core::fmt::Formatter::write_str(f, "InvalidLiteralKind"),
}
}
}Debug, #[automatically_derived]
#[unstable(feature = "proc_macro_value", issue = "136652")]
impl ::core::cmp::PartialEq for ConversionErrorKind {
#[inline]
fn eq(&self, other: &ConversionErrorKind) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(ConversionErrorKind::FailedToUnescape(__self_0),
ConversionErrorKind::FailedToUnescape(__arg1_0)) =>
__self_0 == __arg1_0,
_ => true,
}
}
}PartialEq, #[automatically_derived]
#[unstable(feature = "proc_macro_value", issue = "136652")]
impl ::core::cmp::Eq for ConversionErrorKind {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<EscapeError>;
}
}Eq)]
200pub enum ConversionErrorKind {
201/// The literal failed to be escaped, take a look at [`EscapeError`] for more information.
202FailedToUnescape(EscapeError),
203/// Trying to convert a literal with the wrong type.
204InvalidLiteralKind,
205}
206207/// Determines whether proc_macro has been made accessible to the currently
208/// running program.
209///
210/// The proc_macro crate is only intended for use inside the implementation of
211/// procedural macros. All the functions in this crate panic if invoked from
212/// outside of a procedural macro, such as from a build script or unit test or
213/// ordinary Rust binary.
214///
215/// With consideration for Rust libraries that are designed to support both
216/// macro and non-macro use cases, `proc_macro::is_available()` provides a
217/// non-panicking way to detect whether the infrastructure required to use the
218/// API of proc_macro is presently available. Returns true if invoked from
219/// inside of a procedural macro, false if invoked from any other binary.
220#[stable(feature = "proc_macro_is_available", since = "1.57.0")]
221pub fn is_available() -> bool {
222 bridge::client::is_available()
223}
224225/// The main type provided by this crate, representing an abstract stream of
226/// tokens, or, more specifically, a sequence of token trees.
227/// The type provides interfaces for iterating over those token trees and, conversely,
228/// collecting a number of token trees into one stream.
229///
230/// This is both the input and output of `#[proc_macro]`, `#[proc_macro_attribute]`
231/// and `#[proc_macro_derive]` definitions.
232#[cfg_attr(feature = "rustc-dep-of-std", rustc_diagnostic_item = "TokenStream")]
233#[stable(feature = "proc_macro_lib", since = "1.15.0")]
234#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib", since = "1.15.0")]
impl ::core::clone::Clone for TokenStream {
#[inline]
fn clone(&self) -> TokenStream {
TokenStream(::core::clone::Clone::clone(&self.0))
}
}Clone)]
235pub struct TokenStream(Option<bridge::client::TokenStream>);
236237#[stable(feature = "proc_macro_lib", since = "1.15.0")]
238impl !Sendfor TokenStream {}
239#[stable(feature = "proc_macro_lib", since = "1.15.0")]
240impl !Syncfor TokenStream {}
241242/// Error returned from `TokenStream::from_str`.
243///
244/// The contained error message is explicitly not guaranteed to be stable in any way,
245/// and may change between Rust versions or across compilations.
246#[stable(feature = "proc_macro_lib", since = "1.15.0")]
247#[non_exhaustive]
248#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib", since = "1.15.0")]
impl ::core::fmt::Debug for LexError {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_tuple_field1_finish(f, "LexError",
&&self.0)
}
}Debug)]
249pub struct LexError(String);
250251#[stable(feature = "proc_macro_lexerror_impls", since = "1.44.0")]
252impl fmt::Displayfor LexError {
253fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
254f.write_str(&self.0)
255 }
256}
257258#[stable(feature = "proc_macro_lexerror_impls", since = "1.44.0")]
259impl error::Errorfor LexError {}
260261#[stable(feature = "proc_macro_lib", since = "1.15.0")]
262impl !Sendfor LexError {}
263#[stable(feature = "proc_macro_lib", since = "1.15.0")]
264impl !Syncfor LexError {}
265266/// Error returned from `TokenStream::expand_expr`.
267#[unstable(feature = "proc_macro_expand", issue = "90765")]
268#[non_exhaustive]
269#[derive(#[automatically_derived]
#[unstable(feature = "proc_macro_expand", issue = "90765")]
impl ::core::fmt::Debug for ExpandError {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f, "ExpandError")
}
}Debug)]
270pub struct ExpandError;
271272#[unstable(feature = "proc_macro_expand", issue = "90765")]
273impl fmt::Displayfor ExpandError {
274fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
275f.write_str("macro expansion failed")
276 }
277}
278279#[unstable(feature = "proc_macro_expand", issue = "90765")]
280impl error::Errorfor ExpandError {}
281282#[unstable(feature = "proc_macro_expand", issue = "90765")]
283impl !Sendfor ExpandError {}
284285#[unstable(feature = "proc_macro_expand", issue = "90765")]
286impl !Syncfor ExpandError {}
287288impl TokenStream {
289/// Returns an empty `TokenStream` containing no token trees.
290#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
291pub fn new() -> TokenStream {
292TokenStream(None)
293 }
294295/// Checks if this `TokenStream` is empty.
296#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
297pub fn is_empty(&self) -> bool {
298self.0.as_ref().map(|h| BridgeMethods::ts_is_empty(h)).unwrap_or(true)
299 }
300301/// Parses this `TokenStream` as an expression and attempts to expand any
302 /// macros within it. Returns the expanded `TokenStream`.
303 ///
304 /// Currently only expressions expanding to literals will succeed, although
305 /// this may be relaxed in the future.
306 ///
307 /// NOTE: In error conditions, `expand_expr` may leave macros unexpanded,
308 /// report an error, failing compilation, and/or return an `Err(..)`. The
309 /// specific behavior for any error condition, and what conditions are
310 /// considered errors, is unspecified and may change in the future.
311#[unstable(feature = "proc_macro_expand", issue = "90765")]
312pub fn expand_expr(&self) -> Result<TokenStream, ExpandError> {
313let stream = self.0.as_ref().ok_or(ExpandError)?;
314match BridgeMethods::ts_expand_expr(stream) {
315Ok(stream) => Ok(TokenStream(Some(stream))),
316Err(_) => Err(ExpandError),
317 }
318 }
319}
320321/// Attempts to break the string into tokens and parse those tokens into a token stream.
322/// May fail for a number of reasons, for example, if the string contains unbalanced delimiters
323/// or characters not existing in the language.
324/// All tokens in the parsed stream get `Span::call_site()` spans.
325///
326/// NOTE: some errors may cause panics instead of returning `LexError`. We reserve the right to
327/// change these errors into `LexError`s later.
328#[stable(feature = "proc_macro_lib", since = "1.15.0")]
329impl FromStrfor TokenStream {
330type Err = LexError;
331332fn from_str(src: &str) -> Result<TokenStream, LexError> {
333Ok(TokenStream(Some(BridgeMethods::ts_from_str(src).map_err(LexError)?)))
334 }
335}
336337/// Prints the token stream as a string that is supposed to be losslessly convertible back
338/// into the same token stream (modulo spans), except for possibly `TokenTree::Group`s
339/// with `Delimiter::None` delimiters and negative numeric literals.
340///
341/// Note: the exact form of the output is subject to change, e.g. there might
342/// be changes in the whitespace used between tokens. Therefore, you should
343/// *not* do any kind of simple substring matching on the output string (as
344/// produced by `to_string`) to implement a proc macro, because that matching
345/// might stop working if such changes happen. Instead, you should work at the
346/// `TokenTree` level, e.g. matching against `TokenTree::Ident`,
347/// `TokenTree::Punct`, or `TokenTree::Literal`.
348#[stable(feature = "proc_macro_lib", since = "1.15.0")]
349impl fmt::Displayfor TokenStream {
350fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
351match &self.0 {
352Some(ts) => f.write_fmt(format_args!("{0}", BridgeMethods::ts_to_string(ts)))write!(f, "{}", BridgeMethods::ts_to_string(ts)),
353None => Ok(()),
354 }
355 }
356}
357358/// Prints tokens in a form convenient for debugging.
359#[stable(feature = "proc_macro_lib", since = "1.15.0")]
360impl fmt::Debugfor TokenStream {
361fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
362 f.write_str("TokenStream ")?;
363f.debug_list().entries(self.clone()).finish()
364 }
365}
366367#[stable(feature = "proc_macro_token_stream_default", since = "1.45.0")]
368impl Defaultfor TokenStream {
369fn default() -> Self {
370TokenStream::new()
371 }
372}
373374#[unstable(feature = "proc_macro_quote", issue = "54722")]
375pub use quote::{HasIterator, RepInterp, ThereIsNoIteratorInRepetition, ext, quote, quote_span};
376377fn tree_to_bridge_tree(
378 tree: TokenTree,
379) -> bridge::TokenTree<bridge::client::TokenStream, bridge::client::Span, bridge::client::Symbol> {
380match tree {
381 TokenTree::Group(tt) => bridge::TokenTree::Group(tt.0),
382 TokenTree::Punct(tt) => bridge::TokenTree::Punct(tt.0),
383 TokenTree::Ident(tt) => bridge::TokenTree::Ident(tt.0),
384 TokenTree::Literal(tt) => bridge::TokenTree::Literal(tt.0),
385 }
386}
387388/// Creates a token stream containing a single token tree.
389#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
390impl From<TokenTree> for TokenStream {
391fn from(tree: TokenTree) -> TokenStream {
392TokenStream(Some(BridgeMethods::ts_from_token_tree(tree_to_bridge_tree(tree))))
393 }
394}
395396/// Non-generic helper for implementing `FromIterator<TokenTree>` and
397/// `Extend<TokenTree>` with less monomorphization in calling crates.
398struct ConcatTreesHelper {
399 trees: Vec<
400 bridge::TokenTree<
401 bridge::client::TokenStream,
402 bridge::client::Span,
403 bridge::client::Symbol,
404 >,
405 >,
406}
407408impl ConcatTreesHelper {
409fn new(capacity: usize) -> Self {
410ConcatTreesHelper { trees: Vec::with_capacity(capacity) }
411 }
412413fn push(&mut self, tree: TokenTree) {
414self.trees.push(tree_to_bridge_tree(tree));
415 }
416417fn build(self) -> TokenStream {
418if self.trees.is_empty() {
419TokenStream(None)
420 } else {
421TokenStream(Some(BridgeMethods::ts_concat_trees(None, self.trees)))
422 }
423 }
424425fn append_to(self, stream: &mut TokenStream) {
426if self.trees.is_empty() {
427return;
428 }
429stream.0 = Some(BridgeMethods::ts_concat_trees(stream.0.take(), self.trees))
430 }
431}
432433/// Non-generic helper for implementing `FromIterator<TokenStream>` and
434/// `Extend<TokenStream>` with less monomorphization in calling crates.
435struct ConcatStreamsHelper {
436 streams: Vec<bridge::client::TokenStream>,
437}
438439impl ConcatStreamsHelper {
440fn new(capacity: usize) -> Self {
441ConcatStreamsHelper { streams: Vec::with_capacity(capacity) }
442 }
443444fn push(&mut self, stream: TokenStream) {
445if let Some(stream) = stream.0 {
446self.streams.push(stream);
447 }
448 }
449450fn build(mut self) -> TokenStream {
451if self.streams.len() <= 1 {
452TokenStream(self.streams.pop())
453 } else {
454TokenStream(Some(BridgeMethods::ts_concat_streams(None, self.streams)))
455 }
456 }
457458fn append_to(mut self, stream: &mut TokenStream) {
459if self.streams.is_empty() {
460return;
461 }
462let base = stream.0.take();
463if base.is_none() && self.streams.len() == 1 {
464stream.0 = self.streams.pop();
465 } else {
466stream.0 = Some(BridgeMethods::ts_concat_streams(base, self.streams));
467 }
468 }
469}
470471/// Collects a number of token trees into a single stream.
472#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
473impl FromIterator<TokenTree> for TokenStream {
474fn from_iter<I: IntoIterator<Item = TokenTree>>(trees: I) -> Self {
475let iter = trees.into_iter();
476let mut builder = ConcatTreesHelper::new(iter.size_hint().0);
477iter.for_each(|tree| builder.push(tree));
478builder.build()
479 }
480}
481482/// A "flattening" operation on token streams, collects token trees
483/// from multiple token streams into a single stream.
484#[stable(feature = "proc_macro_lib", since = "1.15.0")]
485impl FromIterator<TokenStream> for TokenStream {
486fn from_iter<I: IntoIterator<Item = TokenStream>>(streams: I) -> Self {
487let iter = streams.into_iter();
488let mut builder = ConcatStreamsHelper::new(iter.size_hint().0);
489iter.for_each(|stream| builder.push(stream));
490builder.build()
491 }
492}
493494#[stable(feature = "token_stream_extend", since = "1.30.0")]
495impl Extend<TokenTree> for TokenStream {
496fn extend<I: IntoIterator<Item = TokenTree>>(&mut self, trees: I) {
497let iter = trees.into_iter();
498let mut builder = ConcatTreesHelper::new(iter.size_hint().0);
499iter.for_each(|tree| builder.push(tree));
500builder.append_to(self);
501 }
502}
503504#[stable(feature = "token_stream_extend", since = "1.30.0")]
505impl Extend<TokenStream> for TokenStream {
506fn extend<I: IntoIterator<Item = TokenStream>>(&mut self, streams: I) {
507let iter = streams.into_iter();
508let mut builder = ConcatStreamsHelper::new(iter.size_hint().0);
509iter.for_each(|stream| builder.push(stream));
510builder.append_to(self);
511 }
512}
513514macro_rules!extend_items {
515 ($($item:ident)*) => {
516 $(
517#[stable(feature = "token_stream_extend_ts_items", since = "1.92.0")]
518impl Extend<$item> for TokenStream {
519fn extend<T: IntoIterator<Item = $item>>(&mut self, iter: T) {
520self.extend(iter.into_iter().map(TokenTree::$item));
521 }
522 }
523 )*
524 };
525}
526527#[stable(feature = "token_stream_extend_ts_items", since = "1.92.0")]
impl Extend<Ident> for TokenStream {
fn extend<T: IntoIterator<Item = Ident>>(&mut self, iter: T) {
self.extend(iter.into_iter().map(TokenTree::Ident));
}
}extend_items!(Group Literal Punct Ident);
528529/// Public implementation details for the `TokenStream` type, such as iterators.
530#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
531pub mod token_stream {
532use crate::{BridgeMethods, Group, Ident, Literal, Punct, TokenStream, TokenTree, bridge};
533534/// An iterator over `TokenStream`'s `TokenTree`s.
535 /// The iteration is "shallow", e.g., the iterator doesn't recurse into delimited groups,
536 /// and returns whole groups as token trees.
537#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for IntoIter {
#[inline]
fn clone(&self) -> IntoIter {
IntoIter(::core::clone::Clone::clone(&self.0))
}
}Clone)]
538 #[stable(feature = "proc_macro_lib2", since = "1.29.0")]
539pub struct IntoIter(
540 std::vec::IntoIter<
541 bridge::TokenTree<
542 bridge::client::TokenStream,
543 bridge::client::Span,
544 bridge::client::Symbol,
545 >,
546 >,
547 );
548549#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
550impl Iteratorfor IntoIter {
551type Item = TokenTree;
552553fn next(&mut self) -> Option<TokenTree> {
554self.0.next().map(|tree| match tree {
555 bridge::TokenTree::Group(tt) => TokenTree::Group(Group(tt)),
556 bridge::TokenTree::Punct(tt) => TokenTree::Punct(Punct(tt)),
557 bridge::TokenTree::Ident(tt) => TokenTree::Ident(Ident(tt)),
558 bridge::TokenTree::Literal(tt) => TokenTree::Literal(Literal(tt)),
559 })
560 }
561562fn size_hint(&self) -> (usize, Option<usize>) {
563self.0.size_hint()
564 }
565566fn count(self) -> usize {
567self.0.count()
568 }
569 }
570571#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
572impl IntoIteratorfor TokenStream {
573type Item = TokenTree;
574type IntoIter = IntoIter;
575576fn into_iter(self) -> IntoIter {
577IntoIter(
578self.0.map(|v| BridgeMethods::ts_into_trees(v)).unwrap_or_default().into_iter(),
579 )
580 }
581 }
582}
583584/// `quote!(..)` accepts arbitrary tokens and expands into a `TokenStream` describing the input.
585/// For example, `quote!(a + b)` will produce an expression, that, when evaluated, constructs
586/// the `TokenStream` `[Ident("a"), Punct('+', Alone), Ident("b")]`.
587///
588/// Unquoting is done with `$`, and works by taking the single next ident as the unquoted term.
589/// To quote `$` itself, use `$$`.
590#[unstable(feature = "proc_macro_quote", issue = "54722")]
591#[allow_internal_unstable(proc_macro_def_site, proc_macro_internals, proc_macro_totokens)]
592#[rustc_builtin_macro]
593pub macro quote($($t:tt)*) {
594/* compiler built-in */
595}
596597#[unstable(feature = "proc_macro_internals", issue = "27812")]
598#[doc(hidden)]
599mod quote;
600601/// A region of source code, along with macro expansion information.
602#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
603#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::marker::Copy for Span { }Copy, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Span {
#[inline]
fn clone(&self) -> Span {
let _: ::core::clone::AssertParamIsClone<bridge::client::Span>;
*self
}
}Clone)]
604pub struct Span(bridge::client::Span);
605606#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
607impl !Sendfor Span {}
608#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
609impl !Syncfor Span {}
610611macro_rules!diagnostic_method {
612 ($name:ident, $level:expr) => {
613/// Creates a new `Diagnostic` with the given `message` at the span
614 /// `self`.
615#[unstable(feature = "proc_macro_diagnostic", issue = "54140")]
616pub fn $name<T: Into<String>>(self, message: T) -> Diagnostic {
617 Diagnostic::spanned(self, $level, message)
618 }
619 };
620}
621622impl Span {
623/// A span that resolves at the macro definition site.
624#[unstable(feature = "proc_macro_def_site", issue = "54724")]
625pub fn def_site() -> Span {
626Span(bridge::client::Span::def_site())
627 }
628629/// The span of the invocation of the current procedural macro.
630 /// Identifiers created with this span will be resolved as if they were written
631 /// directly at the macro call location (call-site hygiene) and other code
632 /// at the macro call site will be able to refer to them as well.
633#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
634pub fn call_site() -> Span {
635Span(bridge::client::Span::call_site())
636 }
637638/// A span that represents `macro_rules` hygiene, and sometimes resolves at the macro
639 /// definition site (local variables, labels, `$crate`) and sometimes at the macro
640 /// call site (everything else).
641 /// The span location is taken from the call-site.
642#[stable(feature = "proc_macro_mixed_site", since = "1.45.0")]
643pub fn mixed_site() -> Span {
644Span(bridge::client::Span::mixed_site())
645 }
646647/// The `Span` for the tokens in the previous macro expansion from which
648 /// `self` was generated from, if any.
649#[unstable(feature = "proc_macro_span", issue = "54725")]
650pub fn parent(&self) -> Option<Span> {
651BridgeMethods::span_parent(self.0).map(Span)
652 }
653654/// The span for the origin source code that `self` was generated from. If
655 /// this `Span` wasn't generated from other macro expansions then the return
656 /// value is the same as `*self`.
657#[unstable(feature = "proc_macro_span", issue = "54725")]
658pub fn source(&self) -> Span {
659Span(BridgeMethods::span_source(self.0))
660 }
661662/// Returns the span's byte position range in the source file.
663#[unstable(feature = "proc_macro_span", issue = "54725")]
664pub fn byte_range(&self) -> Range<usize> {
665BridgeMethods::span_byte_range(self.0)
666 }
667668/// Creates an empty span pointing to directly before this span.
669#[stable(feature = "proc_macro_span_location", since = "1.88.0")]
670pub fn start(&self) -> Span {
671Span(BridgeMethods::span_start(self.0))
672 }
673674/// Creates an empty span pointing to directly after this span.
675#[stable(feature = "proc_macro_span_location", since = "1.88.0")]
676pub fn end(&self) -> Span {
677Span(BridgeMethods::span_end(self.0))
678 }
679680/// The one-indexed line of the source file where the span starts.
681 ///
682 /// To obtain the line of the span's end, use `span.end().line()`.
683#[stable(feature = "proc_macro_span_location", since = "1.88.0")]
684pub fn line(&self) -> usize {
685BridgeMethods::span_line(self.0)
686 }
687688/// The one-indexed column of the source file where the span starts.
689 ///
690 /// To obtain the column of the span's end, use `span.end().column()`.
691#[stable(feature = "proc_macro_span_location", since = "1.88.0")]
692pub fn column(&self) -> usize {
693BridgeMethods::span_column(self.0)
694 }
695696/// The path to the source file in which this span occurs, for display purposes.
697 ///
698 /// This might not correspond to a valid file system path.
699 /// It might be remapped (e.g. `"/src/lib.rs"`) or an artificial path (e.g. `"<command line>"`).
700#[stable(feature = "proc_macro_span_file", since = "1.88.0")]
701pub fn file(&self) -> String {
702BridgeMethods::span_file(self.0)
703 }
704705/// The path to the source file in which this span occurs on the local file system.
706 ///
707 /// This is the actual path on disk. It is unaffected by path remapping.
708 ///
709 /// This path should not be embedded in the output of the macro; prefer `file()` instead.
710#[stable(feature = "proc_macro_span_file", since = "1.88.0")]
711pub fn local_file(&self) -> Option<PathBuf> {
712BridgeMethods::span_local_file(self.0).map(PathBuf::from)
713 }
714715/// Creates a new span encompassing `self` and `other`.
716 ///
717 /// Returns `None` if `self` and `other` are from different files.
718#[unstable(feature = "proc_macro_span", issue = "54725")]
719pub fn join(&self, other: Span) -> Option<Span> {
720BridgeMethods::span_join(self.0, other.0).map(Span)
721 }
722723/// Creates a new span with the same line/column information as `self` but
724 /// that resolves symbols as though it were at `other`.
725#[stable(feature = "proc_macro_span_resolved_at", since = "1.45.0")]
726pub fn resolved_at(&self, other: Span) -> Span {
727Span(BridgeMethods::span_resolved_at(self.0, other.0))
728 }
729730/// Creates a new span with the same name resolution behavior as `self` but
731 /// with the line/column information of `other`.
732#[stable(feature = "proc_macro_span_located_at", since = "1.45.0")]
733pub fn located_at(&self, other: Span) -> Span {
734other.resolved_at(*self)
735 }
736737/// Compares two spans to see if they're equal.
738#[unstable(feature = "proc_macro_span", issue = "54725")]
739pub fn eq(&self, other: &Span) -> bool {
740self.0 == other.0
741}
742743/// Returns the source text behind a span. This preserves the original source
744 /// code, including spaces and comments. It only returns a result if the span
745 /// corresponds to real source code.
746 ///
747 /// Note: The observable result of a macro should only rely on the tokens and
748 /// not on this source text. The result of this function is a best effort to
749 /// be used for diagnostics only.
750#[stable(feature = "proc_macro_source_text", since = "1.66.0")]
751pub fn source_text(&self) -> Option<String> {
752BridgeMethods::span_source_text(self.0)
753 }
754755// Used by the implementation of `Span::quote`
756#[doc(hidden)]
757 #[unstable(feature = "proc_macro_internals", issue = "27812")]
758pub fn save_span(&self) -> usize {
759BridgeMethods::span_save_span(self.0)
760 }
761762// Used by the implementation of `Span::quote`
763#[doc(hidden)]
764 #[unstable(feature = "proc_macro_internals", issue = "27812")]
765pub fn recover_proc_macro_span(id: usize) -> Span {
766Span(BridgeMethods::span_recover_proc_macro_span(id))
767 }
768769/// Creates a new `Diagnostic` with the given `message` at the span
/// `self`.
#[unstable(feature = "proc_macro_diagnostic", issue = "54140")]
pub fn error<T: Into<String>>(self, message: T) -> Diagnostic {
Diagnostic::spanned(self, Level::Error, message)
}diagnostic_method!(error, Level::Error);
770/// Creates a new `Diagnostic` with the given `message` at the span
/// `self`.
#[unstable(feature = "proc_macro_diagnostic", issue = "54140")]
pub fn warning<T: Into<String>>(self, message: T) -> Diagnostic {
Diagnostic::spanned(self, Level::Warning, message)
}diagnostic_method!(warning, Level::Warning);
771/// Creates a new `Diagnostic` with the given `message` at the span
/// `self`.
#[unstable(feature = "proc_macro_diagnostic", issue = "54140")]
pub fn note<T: Into<String>>(self, message: T) -> Diagnostic {
Diagnostic::spanned(self, Level::Note, message)
}diagnostic_method!(note, Level::Note);
772/// Creates a new `Diagnostic` with the given `message` at the span
/// `self`.
#[unstable(feature = "proc_macro_diagnostic", issue = "54140")]
pub fn help<T: Into<String>>(self, message: T) -> Diagnostic {
Diagnostic::spanned(self, Level::Help, message)
}diagnostic_method!(help, Level::Help);
773}
774775/// Prints a span in a form convenient for debugging.
776#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
777impl fmt::Debugfor Span {
778fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
779self.0.fmt(f)
780 }
781}
782783/// A single token or a delimited sequence of token trees (e.g., `[1, (), ..]`).
784#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
785#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for TokenTree {
#[inline]
fn clone(&self) -> TokenTree {
match self {
TokenTree::Group(__self_0) =>
TokenTree::Group(::core::clone::Clone::clone(__self_0)),
TokenTree::Ident(__self_0) =>
TokenTree::Ident(::core::clone::Clone::clone(__self_0)),
TokenTree::Punct(__self_0) =>
TokenTree::Punct(::core::clone::Clone::clone(__self_0)),
TokenTree::Literal(__self_0) =>
TokenTree::Literal(::core::clone::Clone::clone(__self_0)),
}
}
}Clone)]
786pub enum TokenTree {
787/// A token stream surrounded by bracket delimiters.
788#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
789Group(#[stable(feature = "proc_macro_lib2", since = "1.29.0")] Group),
790/// An identifier.
791#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
792Ident(#[stable(feature = "proc_macro_lib2", since = "1.29.0")] Ident),
793/// A single punctuation character (`+`, `,`, `$`, etc.).
794#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
795Punct(#[stable(feature = "proc_macro_lib2", since = "1.29.0")] Punct),
796/// A literal character (`'a'`), string (`"hello"`), number (`2.3`), etc.
797#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
798Literal(#[stable(feature = "proc_macro_lib2", since = "1.29.0")] Literal),
799}
800801#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
802impl !Sendfor TokenTree {}
803#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
804impl !Syncfor TokenTree {}
805806impl TokenTree {
807/// Returns the span of this tree, delegating to the `span` method of
808 /// the contained token or a delimited stream.
809#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
810pub fn span(&self) -> Span {
811match *self {
812 TokenTree::Group(ref t) => t.span(),
813 TokenTree::Ident(ref t) => t.span(),
814 TokenTree::Punct(ref t) => t.span(),
815 TokenTree::Literal(ref t) => t.span(),
816 }
817 }
818819/// Configures the span for *only this token*.
820 ///
821 /// Note that if this token is a `Group` then this method will not configure
822 /// the span of each of the internal tokens, this will simply delegate to
823 /// the `set_span` method of each variant.
824#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
825pub fn set_span(&mut self, span: Span) {
826match *self {
827 TokenTree::Group(ref mut t) => t.set_span(span),
828 TokenTree::Ident(ref mut t) => t.set_span(span),
829 TokenTree::Punct(ref mut t) => t.set_span(span),
830 TokenTree::Literal(ref mut t) => t.set_span(span),
831 }
832 }
833}
834835/// Prints token tree in a form convenient for debugging.
836#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
837impl fmt::Debugfor TokenTree {
838fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
839// Each of these has the name in the struct type in the derived debug,
840 // so don't bother with an extra layer of indirection
841match *self {
842 TokenTree::Group(ref tt) => tt.fmt(f),
843 TokenTree::Ident(ref tt) => tt.fmt(f),
844 TokenTree::Punct(ref tt) => tt.fmt(f),
845 TokenTree::Literal(ref tt) => tt.fmt(f),
846 }
847 }
848}
849850#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
851impl From<Group> for TokenTree {
852fn from(g: Group) -> TokenTree {
853 TokenTree::Group(g)
854 }
855}
856857#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
858impl From<Ident> for TokenTree {
859fn from(g: Ident) -> TokenTree {
860 TokenTree::Ident(g)
861 }
862}
863864#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
865impl From<Punct> for TokenTree {
866fn from(g: Punct) -> TokenTree {
867 TokenTree::Punct(g)
868 }
869}
870871#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
872impl From<Literal> for TokenTree {
873fn from(g: Literal) -> TokenTree {
874 TokenTree::Literal(g)
875 }
876}
877878/// Prints the token tree as a string that is supposed to be losslessly convertible back
879/// into the same token tree (modulo spans), except for possibly `TokenTree::Group`s
880/// with `Delimiter::None` delimiters and negative numeric literals.
881///
882/// Note: the exact form of the output is subject to change, e.g. there might
883/// be changes in the whitespace used between tokens. Therefore, you should
884/// *not* do any kind of simple substring matching on the output string (as
885/// produced by `to_string`) to implement a proc macro, because that matching
886/// might stop working if such changes happen. Instead, you should work at the
887/// `TokenTree` level, e.g. matching against `TokenTree::Ident`,
888/// `TokenTree::Punct`, or `TokenTree::Literal`.
889#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
890impl fmt::Displayfor TokenTree {
891fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
892match self {
893 TokenTree::Group(t) => f.write_fmt(format_args!("{0}", t))write!(f, "{t}"),
894 TokenTree::Ident(t) => f.write_fmt(format_args!("{0}", t))write!(f, "{t}"),
895 TokenTree::Punct(t) => f.write_fmt(format_args!("{0}", t))write!(f, "{t}"),
896 TokenTree::Literal(t) => f.write_fmt(format_args!("{0}", t))write!(f, "{t}"),
897 }
898 }
899}
900901/// A delimited token stream.
902///
903/// A `Group` internally contains a `TokenStream` which is surrounded by `Delimiter`s.
904#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Group {
#[inline]
fn clone(&self) -> Group { Group(::core::clone::Clone::clone(&self.0)) }
}Clone)]
905#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
906pub struct Group(bridge::Group<bridge::client::TokenStream, bridge::client::Span>);
907908#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
909impl !Sendfor Group {}
910#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
911impl !Syncfor Group {}
912913/// Describes how a sequence of token trees is delimited.
914#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::marker::Copy for Delimiter { }Copy, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Delimiter {
#[inline]
fn clone(&self) -> Delimiter { *self }
}Clone, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::fmt::Debug for Delimiter {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
Delimiter::Parenthesis => "Parenthesis",
Delimiter::Brace => "Brace",
Delimiter::Bracket => "Bracket",
Delimiter::None => "None",
})
}
}Debug, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::cmp::PartialEq for Delimiter {
#[inline]
fn eq(&self, other: &Delimiter) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::cmp::Eq for Delimiter {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
915#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
916pub enum Delimiter {
917/// `( ... )`
918#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
919Parenthesis,
920/// `{ ... }`
921#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
922Brace,
923/// `[ ... ]`
924#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
925Bracket,
926/// `∅ ... ∅`
927 /// An invisible delimiter, that may, for example, appear around tokens coming from a
928 /// "macro variable" `$var`. It is important to preserve operator priorities in cases like
929 /// `$var * 3` where `$var` is `1 + 2`.
930 /// Invisible delimiters might not survive roundtrip of a token stream through a string.
931 ///
932 /// <div class="warning">
933 ///
934 /// Note: rustc currently can ignore the grouping of tokens delimited by `None` in the output
935 /// of a proc_macro. Only `None`-delimited groups created by a macro_rules macro in the input
936 /// of a proc_macro macro are preserved, and only in very specific circumstances.
937 /// Any `None`-delimited groups (re)created by a proc_macro will therefore not preserve
938 /// operator priorities as indicated above. The other `Delimiter` variants should be used
939 /// instead in this context. This is a rustc bug. For details, see
940 /// [rust-lang/rust#67062](https://github.com/rust-lang/rust/issues/67062).
941 ///
942 /// </div>
943#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
944None,
945}
946947impl Group {
948/// Creates a new `Group` with the given delimiter and token stream.
949 ///
950 /// This constructor will set the span for this group to
951 /// `Span::call_site()`. To change the span you can use the `set_span`
952 /// method below.
953#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
954pub fn new(delimiter: Delimiter, stream: TokenStream) -> Group {
955Group(bridge::Group {
956delimiter,
957 stream: stream.0,
958 span: bridge::DelimSpan::from_single(Span::call_site().0),
959 })
960 }
961962/// Returns the delimiter of this `Group`
963#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
964pub fn delimiter(&self) -> Delimiter {
965self.0.delimiter
966 }
967968/// Returns the `TokenStream` of tokens that are delimited in this `Group`.
969 ///
970 /// Note that the returned token stream does not include the delimiter
971 /// returned above.
972#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
973pub fn stream(&self) -> TokenStream {
974TokenStream(self.0.stream.clone())
975 }
976977/// Returns the span for the delimiters of this token stream, spanning the
978 /// entire `Group`.
979 ///
980 /// ```text
981 /// pub fn span(&self) -> Span {
982 /// ^^^^^^^
983 /// ```
984#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
985pub fn span(&self) -> Span {
986Span(self.0.span.entire)
987 }
988989/// Returns the span pointing to the opening delimiter of this group.
990 ///
991 /// ```text
992 /// pub fn span_open(&self) -> Span {
993 /// ^
994 /// ```
995#[stable(feature = "proc_macro_group_span", since = "1.55.0")]
996pub fn span_open(&self) -> Span {
997Span(self.0.span.open)
998 }
9991000/// Returns the span pointing to the closing delimiter of this group.
1001 ///
1002 /// ```text
1003 /// pub fn span_close(&self) -> Span {
1004 /// ^
1005 /// ```
1006#[stable(feature = "proc_macro_group_span", since = "1.55.0")]
1007pub fn span_close(&self) -> Span {
1008Span(self.0.span.close)
1009 }
10101011/// Configures the span for this `Group`'s delimiters, but not its internal
1012 /// tokens.
1013 ///
1014 /// This method will **not** set the span of all the internal tokens spanned
1015 /// by this group, but rather it will only set the span of the delimiter
1016 /// tokens at the level of the `Group`.
1017#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1018pub fn set_span(&mut self, span: Span) {
1019self.0.span = bridge::DelimSpan::from_single(span.0);
1020 }
1021}
10221023/// Prints the group as a string that should be losslessly convertible back
1024/// into the same group (modulo spans), except for possibly `TokenTree::Group`s
1025/// with `Delimiter::None` delimiters.
1026#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1027impl fmt::Displayfor Group {
1028fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1029f.write_fmt(format_args!("{0}",
TokenStream::from(TokenTree::from(self.clone()))))write!(f, "{}", TokenStream::from(TokenTree::from(self.clone())))1030 }
1031}
10321033#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1034impl fmt::Debugfor Group {
1035fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1036f.debug_struct("Group")
1037 .field("delimiter", &self.delimiter())
1038 .field("stream", &self.stream())
1039 .field("span", &self.span())
1040 .finish()
1041 }
1042}
10431044/// A `Punct` is a single punctuation character such as `+`, `-` or `#`.
1045///
1046/// Multi-character operators like `+=` are represented as two instances of `Punct` with different
1047/// forms of `Spacing` returned.
1048#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1049#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Punct {
#[inline]
fn clone(&self) -> Punct { Punct(::core::clone::Clone::clone(&self.0)) }
}Clone)]
1050pub struct Punct(bridge::Punct<bridge::client::Span>);
10511052#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1053impl !Sendfor Punct {}
1054#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1055impl !Syncfor Punct {}
10561057/// Indicates whether a `Punct` token can join with the following token
1058/// to form a multi-character operator.
1059#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::marker::Copy for Spacing { }Copy, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Spacing {
#[inline]
fn clone(&self) -> Spacing { *self }
}Clone, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::fmt::Debug for Spacing {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
Spacing::Joint => "Joint",
Spacing::Alone => "Alone",
})
}
}Debug, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::cmp::PartialEq for Spacing {
#[inline]
fn eq(&self, other: &Spacing) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::cmp::Eq for Spacing {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
1060#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1061pub enum Spacing {
1062/// A `Punct` token can join with the following token to form a multi-character operator.
1063 ///
1064 /// In token streams constructed using proc macro interfaces, `Joint` punctuation tokens can be
1065 /// followed by any other tokens. However, in token streams parsed from source code, the
1066 /// compiler will only set spacing to `Joint` in the following cases.
1067 /// - When a `Punct` is immediately followed by another `Punct` without a whitespace. E.g. `+`
1068 /// is `Joint` in `+=` and `++`.
1069 /// - When a single quote `'` is immediately followed by an identifier without a whitespace.
1070 /// E.g. `'` is `Joint` in `'lifetime`.
1071 ///
1072 /// This list may be extended in the future to enable more token combinations.
1073#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1074Joint,
1075/// A `Punct` token cannot join with the following token to form a multi-character operator.
1076 ///
1077 /// `Alone` punctuation tokens can be followed by any other tokens. In token streams parsed
1078 /// from source code, the compiler will set spacing to `Alone` in all cases not covered by the
1079 /// conditions for `Joint` above. E.g. `+` is `Alone` in `+ =`, `+ident` and `+()`. In
1080 /// particular, tokens not followed by anything will be marked as `Alone`.
1081#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1082Alone,
1083}
10841085impl Punct {
1086/// Creates a new `Punct` from the given character and spacing.
1087 /// The `ch` argument must be a valid punctuation character permitted by the language,
1088 /// otherwise the function will panic.
1089 ///
1090 /// The returned `Punct` will have the default span of `Span::call_site()`
1091 /// which can be further configured with the `set_span` method below.
1092#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1093pub fn new(ch: char, spacing: Spacing) -> Punct {
1094const LEGAL_CHARS: &[char] = &[
1095'=', '<', '>', '!', '~', '+', '-', '*', '/', '%', '^', '&', '|', '@', '.', ',', ';',
1096':', '#', '$', '?', '\'',
1097 ];
1098if !LEGAL_CHARS.contains(&ch) {
1099{
::core::panicking::panic_fmt(format_args!("unsupported character `{0:?}`",
ch));
};panic!("unsupported character `{:?}`", ch);
1100 }
1101Punct(bridge::Punct {
1102 ch: chas u8,
1103 joint: spacing == Spacing::Joint,
1104 span: Span::call_site().0,
1105 })
1106 }
11071108/// Returns the value of this punctuation character as `char`.
1109#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1110pub fn as_char(&self) -> char {
1111self.0.ch as char1112 }
11131114/// Returns the spacing of this punctuation character, indicating whether it can be potentially
1115 /// combined into a multi-character operator with the following token (`Joint`), or whether the
1116 /// operator has definitely ended (`Alone`).
1117#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1118pub fn spacing(&self) -> Spacing {
1119if self.0.joint { Spacing::Joint } else { Spacing::Alone }
1120 }
11211122/// Returns the span for this punctuation character.
1123#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1124pub fn span(&self) -> Span {
1125Span(self.0.span)
1126 }
11271128/// Configure the span for this punctuation character.
1129#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1130pub fn set_span(&mut self, span: Span) {
1131self.0.span = span.0;
1132 }
1133}
11341135/// Prints the punctuation character as a string that should be losslessly convertible
1136/// back into the same character.
1137#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1138impl fmt::Displayfor Punct {
1139fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1140f.write_fmt(format_args!("{0}", self.as_char()))write!(f, "{}", self.as_char())1141 }
1142}
11431144#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1145impl fmt::Debugfor Punct {
1146fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1147f.debug_struct("Punct")
1148 .field("ch", &self.as_char())
1149 .field("spacing", &self.spacing())
1150 .field("span", &self.span())
1151 .finish()
1152 }
1153}
11541155#[stable(feature = "proc_macro_punct_eq", since = "1.50.0")]
1156impl PartialEq<char> for Punct {
1157fn eq(&self, rhs: &char) -> bool {
1158self.as_char() == *rhs1159 }
1160}
11611162#[stable(feature = "proc_macro_punct_eq_flipped", since = "1.52.0")]
1163impl PartialEq<Punct> for char {
1164fn eq(&self, rhs: &Punct) -> bool {
1165*self == rhs.as_char()
1166 }
1167}
11681169/// An identifier (`ident`).
1170#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Ident {
#[inline]
fn clone(&self) -> Ident { Ident(::core::clone::Clone::clone(&self.0)) }
}Clone)]
1171#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1172pub struct Ident(bridge::Ident<bridge::client::Span, bridge::client::Symbol>);
11731174impl Ident {
1175/// Creates a new `Ident` with the given `string` as well as the specified
1176 /// `span`.
1177 /// The `string` argument must be a valid identifier permitted by the
1178 /// language (including keywords, e.g. `self` or `fn`). Otherwise, the function will panic.
1179 ///
1180 /// The constructed identifier will be NFC-normalized. See the [Reference] for more info.
1181 ///
1182 /// Note that `span`, currently in rustc, configures the hygiene information
1183 /// for this identifier.
1184 ///
1185 /// As of this time `Span::call_site()` explicitly opts-in to "call-site" hygiene
1186 /// meaning that identifiers created with this span will be resolved as if they were written
1187 /// directly at the location of the macro call, and other code at the macro call site will be
1188 /// able to refer to them as well.
1189 ///
1190 /// Later spans like `Span::def_site()` will allow to opt-in to "definition-site" hygiene
1191 /// meaning that identifiers created with this span will be resolved at the location of the
1192 /// macro definition and other code at the macro call site will not be able to refer to them.
1193 ///
1194 /// Due to the current importance of hygiene this constructor, unlike other
1195 /// tokens, requires a `Span` to be specified at construction.
1196 ///
1197 /// [Reference]: https://doc.rust-lang.org/nightly/reference/identifiers.html#r-ident.normalization
1198#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1199pub fn new(string: &str, span: Span) -> Ident {
1200Ident(bridge::Ident {
1201 sym: bridge::client::Symbol::new_ident(string, false),
1202 is_raw: false,
1203 span: span.0,
1204 })
1205 }
12061207/// Same as `Ident::new`, but creates a raw identifier (`r#ident`).
1208 /// The `string` argument be a valid identifier permitted by the language
1209 /// (including keywords, e.g. `fn`). Keywords which are usable in path segments
1210 /// (e.g. `self`, `super`) are not supported, and will cause a panic.
1211#[stable(feature = "proc_macro_raw_ident", since = "1.47.0")]
1212pub fn new_raw(string: &str, span: Span) -> Ident {
1213Ident(bridge::Ident {
1214 sym: bridge::client::Symbol::new_ident(string, true),
1215 is_raw: true,
1216 span: span.0,
1217 })
1218 }
12191220/// Returns the span of this `Ident`, encompassing the entire string returned
1221 /// by [`to_string`](ToString::to_string).
1222#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1223pub fn span(&self) -> Span {
1224Span(self.0.span)
1225 }
12261227/// Configures the span of this `Ident`, possibly changing its hygiene context.
1228#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1229pub fn set_span(&mut self, span: Span) {
1230self.0.span = span.0;
1231 }
1232}
12331234/// Prints the identifier as a string that should be losslessly convertible back
1235/// into the same identifier.
1236#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1237impl fmt::Displayfor Ident {
1238fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1239if self.0.is_raw {
1240 f.write_str("r#")?;
1241 }
1242 fmt::Display::fmt(&self.0.sym, f)
1243 }
1244}
12451246#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1247impl fmt::Debugfor Ident {
1248fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1249f.debug_struct("Ident")
1250 .field("ident", &self.to_string())
1251 .field("span", &self.span())
1252 .finish()
1253 }
1254}
12551256/// A literal string (`"hello"`), byte string (`b"hello"`), C string (`c"hello"`),
1257/// character (`'a'`), byte character (`b'a'`), an integer or floating point number
1258/// with or without a suffix (`1`, `1u8`, `2.3`, `2.3f32`).
1259/// Boolean literals like `true` and `false` do not belong here, they are `Ident`s.
1260#[derive(#[automatically_derived]
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
impl ::core::clone::Clone for Literal {
#[inline]
fn clone(&self) -> Literal {
Literal(::core::clone::Clone::clone(&self.0))
}
}Clone)]
1261#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1262pub struct Literal(bridge::Literal<bridge::client::Span, bridge::client::Symbol>);
12631264macro_rules!suffixed_int_literals {
1265 ($($name:ident => $kind:ident,)*) => ($(
1266/// Creates a new suffixed integer literal with the specified value.
1267 ///
1268 /// This function will create an integer like `1u32` where the integer
1269 /// value specified is the first part of the token and the integral is
1270 /// also suffixed at the end.
1271 /// Literals created from negative numbers might not survive round-trips through
1272 /// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
1273 ///
1274 /// Literals created through this method have the `Span::call_site()`
1275 /// span by default, which can be configured with the `set_span` method
1276 /// below.
1277#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1278pub fn $name(n: $kind) -> Literal {
1279 Literal(bridge::Literal {
1280 kind: bridge::LitKind::Integer,
1281 symbol: bridge::client::Symbol::new(&n.to_string()),
1282 suffix: Some(bridge::client::Symbol::new(stringify!($kind))),
1283 span: Span::call_site().0,
1284 })
1285 }
1286 )*)
1287}
12881289macro_rules!unsuffixed_int_literals {
1290 ($($name:ident => $kind:ident,)*) => ($(
1291/// Creates a new unsuffixed integer literal with the specified value.
1292 ///
1293 /// This function will create an integer like `1` where the integer
1294 /// value specified is the first part of the token. No suffix is
1295 /// specified on this token, meaning that invocations like
1296 /// `Literal::i8_unsuffixed(1)` are equivalent to
1297 /// `Literal::u32_unsuffixed(1)`.
1298 /// Literals created from negative numbers might not survive rountrips through
1299 /// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
1300 ///
1301 /// Literals created through this method have the `Span::call_site()`
1302 /// span by default, which can be configured with the `set_span` method
1303 /// below.
1304#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1305pub fn $name(n: $kind) -> Literal {
1306 Literal(bridge::Literal {
1307 kind: bridge::LitKind::Integer,
1308 symbol: bridge::client::Symbol::new(&n.to_string()),
1309 suffix: None,
1310 span: Span::call_site().0,
1311 })
1312 }
1313 )*)
1314}
13151316macro_rules!integer_values {
1317 ($($nb:ident => $fn_name:ident,)+) => {
1318 $(
1319#[doc = concat!(
1320"Returns the unescaped `",
1321stringify!($nb),
1322"` value if the literal is a `",
1323stringify!($nb),
1324"` or if it's an \"unmarked\" integer which doesn't overflow.")]
1325 #[unstable(feature = "proc_macro_value", issue = "136652")]
1326pub fn $fn_name(&self) -> Result<$nb, ConversionErrorKind> {
1327if self.0.kind != bridge::LitKind::Integer {
1328return Err(ConversionErrorKind::InvalidLiteralKind);
1329 }
1330self.with_symbol_and_suffix(|symbol, suffix| {
1331match suffix {
1332stringify!($nb) | "" => {
1333let symbol = strip_underscores(symbol);
1334let (number, base) = parse_number(&symbol);
1335$nb::from_str_radix(&number, base as u32).map_err(|_| ConversionErrorKind::InvalidLiteralKind)
1336 }
1337_ => Err(ConversionErrorKind::InvalidLiteralKind),
1338 }
1339 })
1340 }
1341 )+
1342 }
1343}
13441345macro_rules!float_values {
1346 ($($nb:ident => $fn_name:ident,)+) => {
1347 $(
1348#[doc = concat!(
1349"Returns the unescaped `",
1350stringify!($nb),
1351"` value if the literal is a `",
1352stringify!($nb),
1353"` or if it's an \"unmarked\" float which doesn't overflow.")]
1354 #[unstable(feature = "proc_macro_value", issue = "136652")]
1355pub fn $fn_name(&self) -> Result<$nb, ConversionErrorKind> {
1356if self.0.kind != bridge::LitKind::Float {
1357return Err(ConversionErrorKind::InvalidLiteralKind);
1358 }
1359self.with_symbol_and_suffix(|symbol, suffix| {
1360match suffix {
1361stringify!($nb) | "" => {
1362let number = strip_underscores(symbol);
1363$nb::from_str(&number).map_err(|_| ConversionErrorKind::InvalidLiteralKind)
1364 }
1365_ => Err(ConversionErrorKind::InvalidLiteralKind),
1366 }
1367 })
1368 }
1369 )+
1370 }
1371}
13721373impl Literal {
1374fn new(kind: bridge::LitKind, value: &str, suffix: Option<&str>) -> Self {
1375Literal(bridge::Literal {
1376kind,
1377 symbol: bridge::client::Symbol::new(value),
1378 suffix: suffix.map(bridge::client::Symbol::new),
1379 span: Span::call_site().0,
1380 })
1381 }
13821383/// Creates a new suffixed integer literal with the specified value.
///
/// This function will create an integer like `1u32` where the integer
/// value specified is the first part of the token and the integral is
/// also suffixed at the end.
/// Literals created from negative numbers might not survive round-trips through
/// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
///
/// Literals created through this method have the `Span::call_site()`
/// span by default, which can be configured with the `set_span` method
/// below.
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
pub fn isize_suffixed(n: isize) -> Literal {
Literal(bridge::Literal {
kind: bridge::LitKind::Integer,
symbol: bridge::client::Symbol::new(&n.to_string()),
suffix: Some(bridge::client::Symbol::new("isize")),
span: Span::call_site().0,
})
}suffixed_int_literals! {
1384 u8_suffixed => u8,
1385 u16_suffixed => u16,
1386 u32_suffixed => u32,
1387 u64_suffixed => u64,
1388 u128_suffixed => u128,
1389 usize_suffixed => usize,
1390 i8_suffixed => i8,
1391 i16_suffixed => i16,
1392 i32_suffixed => i32,
1393 i64_suffixed => i64,
1394 i128_suffixed => i128,
1395 isize_suffixed => isize,
1396 }13971398/// Creates a new unsuffixed integer literal with the specified value.
///
/// This function will create an integer like `1` where the integer
/// value specified is the first part of the token. No suffix is
/// specified on this token, meaning that invocations like
/// `Literal::i8_unsuffixed(1)` are equivalent to
/// `Literal::u32_unsuffixed(1)`.
/// Literals created from negative numbers might not survive rountrips through
/// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
///
/// Literals created through this method have the `Span::call_site()`
/// span by default, which can be configured with the `set_span` method
/// below.
#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
pub fn isize_unsuffixed(n: isize) -> Literal {
Literal(bridge::Literal {
kind: bridge::LitKind::Integer,
symbol: bridge::client::Symbol::new(&n.to_string()),
suffix: None,
span: Span::call_site().0,
})
}unsuffixed_int_literals! {
1399 u8_unsuffixed => u8,
1400 u16_unsuffixed => u16,
1401 u32_unsuffixed => u32,
1402 u64_unsuffixed => u64,
1403 u128_unsuffixed => u128,
1404 usize_unsuffixed => usize,
1405 i8_unsuffixed => i8,
1406 i16_unsuffixed => i16,
1407 i32_unsuffixed => i32,
1408 i64_unsuffixed => i64,
1409 i128_unsuffixed => i128,
1410 isize_unsuffixed => isize,
1411 }14121413/// Creates a new unsuffixed floating-point literal.
1414 ///
1415 /// This constructor is similar to those like `Literal::i8_unsuffixed` where
1416 /// the float's value is emitted directly into the token but no suffix is
1417 /// used, so it may be inferred to be a `f64` later in the compiler.
1418 /// Literals created from negative numbers might not survive rountrips through
1419 /// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
1420 ///
1421 /// # Panics
1422 ///
1423 /// This function requires that the specified float is finite, for
1424 /// example if it is infinity or NaN this function will panic.
1425#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1426pub fn f32_unsuffixed(n: f32) -> Literal {
1427if !n.is_finite() {
1428{
::core::panicking::panic_fmt(format_args!("Invalid float literal {0}",
n));
};panic!("Invalid float literal {n}");
1429 }
1430let mut repr = n.to_string();
1431if !repr.contains('.') {
1432repr.push_str(".0");
1433 }
1434Literal::new(bridge::LitKind::Float, &repr, None)
1435 }
14361437/// Creates a new suffixed floating-point literal.
1438 ///
1439 /// This constructor will create a literal like `1.0f32` where the value
1440 /// specified is the preceding part of the token and `f32` is the suffix of
1441 /// the token. This token will always be inferred to be an `f32` in the
1442 /// compiler.
1443 /// Literals created from negative numbers might not survive rountrips through
1444 /// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
1445 ///
1446 /// # Panics
1447 ///
1448 /// This function requires that the specified float is finite, for
1449 /// example if it is infinity or NaN this function will panic.
1450#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1451pub fn f32_suffixed(n: f32) -> Literal {
1452if !n.is_finite() {
1453{
::core::panicking::panic_fmt(format_args!("Invalid float literal {0}",
n));
};panic!("Invalid float literal {n}");
1454 }
1455Literal::new(bridge::LitKind::Float, &n.to_string(), Some("f32"))
1456 }
14571458/// Creates a new unsuffixed floating-point literal.
1459 ///
1460 /// This constructor is similar to those like `Literal::i8_unsuffixed` where
1461 /// the float's value is emitted directly into the token but no suffix is
1462 /// used, so it may be inferred to be a `f64` later in the compiler.
1463 /// Literals created from negative numbers might not survive rountrips through
1464 /// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
1465 ///
1466 /// # Panics
1467 ///
1468 /// This function requires that the specified float is finite, for
1469 /// example if it is infinity or NaN this function will panic.
1470#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1471pub fn f64_unsuffixed(n: f64) -> Literal {
1472if !n.is_finite() {
1473{
::core::panicking::panic_fmt(format_args!("Invalid float literal {0}",
n));
};panic!("Invalid float literal {n}");
1474 }
1475let mut repr = n.to_string();
1476if !repr.contains('.') {
1477repr.push_str(".0");
1478 }
1479Literal::new(bridge::LitKind::Float, &repr, None)
1480 }
14811482/// Creates a new suffixed floating-point literal.
1483 ///
1484 /// This constructor will create a literal like `1.0f64` where the value
1485 /// specified is the preceding part of the token and `f64` is the suffix of
1486 /// the token. This token will always be inferred to be an `f64` in the
1487 /// compiler.
1488 /// Literals created from negative numbers might not survive rountrips through
1489 /// `TokenStream` or strings and may be broken into two tokens (`-` and positive literal).
1490 ///
1491 /// # Panics
1492 ///
1493 /// This function requires that the specified float is finite, for
1494 /// example if it is infinity or NaN this function will panic.
1495#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1496pub fn f64_suffixed(n: f64) -> Literal {
1497if !n.is_finite() {
1498{
::core::panicking::panic_fmt(format_args!("Invalid float literal {0}",
n));
};panic!("Invalid float literal {n}");
1499 }
1500Literal::new(bridge::LitKind::Float, &n.to_string(), Some("f64"))
1501 }
15021503/// String literal.
1504#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1505pub fn string(string: &str) -> Literal {
1506let escape = EscapeOptions {
1507 escape_single_quote: false,
1508 escape_double_quote: true,
1509 escape_nonascii: false,
1510 };
1511let repr = escape_bytes(string.as_bytes(), escape);
1512Literal::new(bridge::LitKind::Str, &repr, None)
1513 }
15141515/// Character literal.
1516#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1517pub fn character(ch: char) -> Literal {
1518let escape = EscapeOptions {
1519 escape_single_quote: true,
1520 escape_double_quote: false,
1521 escape_nonascii: false,
1522 };
1523let repr = escape_bytes(ch.encode_utf8(&mut [0u8; 4]).as_bytes(), escape);
1524Literal::new(bridge::LitKind::Char, &repr, None)
1525 }
15261527/// Byte character literal.
1528#[stable(feature = "proc_macro_byte_character", since = "1.79.0")]
1529pub fn byte_character(byte: u8) -> Literal {
1530let escape = EscapeOptions {
1531 escape_single_quote: true,
1532 escape_double_quote: false,
1533 escape_nonascii: true,
1534 };
1535let repr = escape_bytes(&[byte], escape);
1536Literal::new(bridge::LitKind::Byte, &repr, None)
1537 }
15381539/// Byte string literal.
1540#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1541pub fn byte_string(bytes: &[u8]) -> Literal {
1542let escape = EscapeOptions {
1543 escape_single_quote: false,
1544 escape_double_quote: true,
1545 escape_nonascii: true,
1546 };
1547let repr = escape_bytes(bytes, escape);
1548Literal::new(bridge::LitKind::ByteStr, &repr, None)
1549 }
15501551/// C string literal.
1552#[stable(feature = "proc_macro_c_str_literals", since = "1.79.0")]
1553pub fn c_string(string: &CStr) -> Literal {
1554let escape = EscapeOptions {
1555 escape_single_quote: false,
1556 escape_double_quote: true,
1557 escape_nonascii: false,
1558 };
1559let repr = escape_bytes(string.to_bytes(), escape);
1560Literal::new(bridge::LitKind::CStr, &repr, None)
1561 }
15621563/// Returns the span encompassing this literal.
1564#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1565pub fn span(&self) -> Span {
1566Span(self.0.span)
1567 }
15681569/// Configures the span associated for this literal.
1570#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1571pub fn set_span(&mut self, span: Span) {
1572self.0.span = span.0;
1573 }
15741575/// Returns a `Span` that is a subset of `self.span()` containing only the
1576 /// source bytes in range `range`. Returns `None` if the would-be trimmed
1577 /// span is outside the bounds of `self`.
1578// FIXME(SergioBenitez): check that the byte range starts and ends at a
1579 // UTF-8 boundary of the source. otherwise, it's likely that a panic will
1580 // occur elsewhere when the source text is printed.
1581 // FIXME(SergioBenitez): there is no way for the user to know what
1582 // `self.span()` actually maps to, so this method can currently only be
1583 // called blindly. For example, `to_string()` for the character 'c' returns
1584 // "'\u{63}'"; there is no way for the user to know whether the source text
1585 // was 'c' or whether it was '\u{63}'.
1586#[unstable(feature = "proc_macro_span", issue = "54725")]
1587pub fn subspan<R: RangeBounds<usize>>(&self, range: R) -> Option<Span> {
1588BridgeMethods::span_subspan(
1589self.0.span,
1590range.start_bound().cloned(),
1591range.end_bound().cloned(),
1592 )
1593 .map(Span)
1594 }
15951596fn with_symbol_and_suffix<R>(&self, f: impl FnOnce(&str, &str) -> R) -> R {
1597self.0.symbol.with(|symbol| match self.0.suffix {
1598Some(suffix) => suffix.with(|suffix| f(symbol, suffix)),
1599None => f(symbol, ""),
1600 })
1601 }
16021603/// Invokes the callback with a `&[&str]` consisting of each part of the
1604 /// literal's representation. This is done to allow the `ToString` and
1605 /// `Display` implementations to borrow references to symbol values, and
1606 /// both be optimized to reduce overhead.
1607fn with_stringify_parts<R>(&self, f: impl FnOnce(&[&str]) -> R) -> R {
1608/// Returns a string containing exactly `num` '#' characters.
1609 /// Uses a 256-character source string literal which is always safe to
1610 /// index with a `u8` index.
1611fn get_hashes_str(num: u8) -> &'static str {
1612const HASHES: &str = "\
1613 ################################################################\
1614 ################################################################\
1615 ################################################################\
1616 ################################################################\
1617 ";
1618const _: () = if !(HASHES.len() == 256) {
::core::panicking::panic("assertion failed: HASHES.len() == 256")
}assert!(HASHES.len() == 256);
1619&HASHES[..num as usize]
1620 }
16211622self.with_symbol_and_suffix(|symbol, suffix| match self.0.kind {
1623 bridge::LitKind::Byte => f(&["b'", symbol, "'", suffix]),
1624 bridge::LitKind::Char => f(&["'", symbol, "'", suffix]),
1625 bridge::LitKind::Str => f(&["\"", symbol, "\"", suffix]),
1626 bridge::LitKind::StrRaw(n) => {
1627let hashes = get_hashes_str(n);
1628f(&["r", hashes, "\"", symbol, "\"", hashes, suffix])
1629 }
1630 bridge::LitKind::ByteStr => f(&["b\"", symbol, "\"", suffix]),
1631 bridge::LitKind::ByteStrRaw(n) => {
1632let hashes = get_hashes_str(n);
1633f(&["br", hashes, "\"", symbol, "\"", hashes, suffix])
1634 }
1635 bridge::LitKind::CStr => f(&["c\"", symbol, "\"", suffix]),
1636 bridge::LitKind::CStrRaw(n) => {
1637let hashes = get_hashes_str(n);
1638f(&["cr", hashes, "\"", symbol, "\"", hashes, suffix])
1639 }
16401641 bridge::LitKind::Integer | bridge::LitKind::Float | bridge::LitKind::ErrWithGuar => {
1642f(&[symbol, suffix])
1643 }
1644 })
1645 }
16461647/// Returns the unescaped character value if the current literal is a byte character literal.
1648#[unstable(feature = "proc_macro_value", issue = "136652")]
1649pub fn byte_character_value(&self) -> Result<u8, ConversionErrorKind> {
1650self.0.symbol.with(|symbol| match self.0.kind {
1651 bridge::LitKind::Byte => unescape_byte(symbol)
1652 .map_err(|err| ConversionErrorKind::FailedToUnescape(err.into())),
1653_ => Err(ConversionErrorKind::InvalidLiteralKind),
1654 })
1655 }
16561657/// Returns the unescaped character value if the current literal is a character literal.
1658#[unstable(feature = "proc_macro_value", issue = "136652")]
1659pub fn character_value(&self) -> Result<char, ConversionErrorKind> {
1660self.0.symbol.with(|symbol| match self.0.kind {
1661 bridge::LitKind::Char => unescape_char(symbol)
1662 .map_err(|err| ConversionErrorKind::FailedToUnescape(err.into())),
1663_ => Err(ConversionErrorKind::InvalidLiteralKind),
1664 })
1665 }
16661667/// Returns the unescaped string value if the current literal is a string or a string literal.
1668#[unstable(feature = "proc_macro_value", issue = "136652")]
1669pub fn str_value(&self) -> Result<String, ConversionErrorKind> {
1670self.0.symbol.with(|symbol| match self.0.kind {
1671 bridge::LitKind::Str => {
1672if symbol.contains('\\') {
1673let mut buf = String::with_capacity(symbol.len());
1674let mut error = None;
1675// Force-inlining here is aggressive but the closure is
1676 // called on every char in the string, so it can be hot in
1677 // programs with many long strings containing escapes.
1678unescape_str(
1679symbol,
1680#[inline(always)]
1681|_, c| match c {
1682Ok(c) => buf.push(c),
1683Err(err) => {
1684if err.is_fatal() {
1685error = Some(ConversionErrorKind::FailedToUnescape(err.into()));
1686 }
1687 }
1688 },
1689 );
1690if let Some(error) = error { Err(error) } else { Ok(buf) }
1691 } else {
1692Ok(symbol.to_string())
1693 }
1694 }
1695 bridge::LitKind::StrRaw(_) => Ok(symbol.to_string()),
1696_ => Err(ConversionErrorKind::InvalidLiteralKind),
1697 })
1698 }
16991700/// Returns the unescaped string value if the current literal is a c-string or a c-string
1701 /// literal.
1702#[unstable(feature = "proc_macro_value", issue = "136652")]
1703pub fn cstr_value(&self) -> Result<Vec<u8>, ConversionErrorKind> {
1704self.0.symbol.with(|symbol| match self.0.kind {
1705 bridge::LitKind::CStr => {
1706let mut error = None;
1707let mut buf = Vec::with_capacity(symbol.len());
17081709unescape_c_str(symbol, |_span, res| match res {
1710Ok(MixedUnit::Char(c)) => {
1711buf.extend_from_slice(c.get().encode_utf8(&mut [0; 4]).as_bytes())
1712 }
1713Ok(MixedUnit::HighByte(b)) => buf.push(b.get()),
1714Err(err) => {
1715if err.is_fatal() {
1716error = Some(ConversionErrorKind::FailedToUnescape(err.into()));
1717 }
1718 }
1719 });
1720if let Some(error) = error {
1721Err(error)
1722 } else {
1723buf.push(0);
1724Ok(buf)
1725 }
1726 }
1727 bridge::LitKind::CStrRaw(_) => {
1728// Raw strings have no escapes so we can convert the symbol
1729 // directly to a `Lrc<u8>` after appending the terminating NUL
1730 // char.
1731let mut buf = symbol.to_owned().into_bytes();
1732buf.push(0);
1733Ok(buf)
1734 }
1735_ => Err(ConversionErrorKind::InvalidLiteralKind),
1736 })
1737 }
17381739/// Returns the unescaped string value if the current literal is a byte string or a byte string
1740 /// literal.
1741#[unstable(feature = "proc_macro_value", issue = "136652")]
1742pub fn byte_str_value(&self) -> Result<Vec<u8>, ConversionErrorKind> {
1743self.0.symbol.with(|symbol| match self.0.kind {
1744 bridge::LitKind::ByteStr => {
1745let mut buf = Vec::with_capacity(symbol.len());
1746let mut error = None;
17471748unescape_byte_str(symbol, |_, res| match res {
1749Ok(b) => buf.push(b),
1750Err(err) => {
1751if err.is_fatal() {
1752error = Some(ConversionErrorKind::FailedToUnescape(err.into()));
1753 }
1754 }
1755 });
1756if let Some(error) = error { Err(error) } else { Ok(buf) }
1757 }
1758 bridge::LitKind::ByteStrRaw(_) => {
1759// Raw strings have no escapes so we can convert the symbol
1760 // directly to a `Lrc<u8>`.
1761Ok(symbol.to_owned().into_bytes())
1762 }
1763_ => Err(ConversionErrorKind::InvalidLiteralKind),
1764 })
1765 }
17661767#[doc =
"Returns the unescaped `i128` value if the literal is a `i128` or if it\'s an \"unmarked\" integer which doesn\'t overflow."]
#[unstable(feature = "proc_macro_value", issue = "136652")]
pub fn i128_value(&self) -> Result<i128, ConversionErrorKind> {
if self.0.kind != bridge::LitKind::Integer {
return Err(ConversionErrorKind::InvalidLiteralKind);
}
self.with_symbol_and_suffix(|symbol, suffix|
{
match suffix {
"i128" | "" => {
let symbol = strip_underscores(symbol);
let (number, base) = parse_number(&symbol);
i128::from_str_radix(&number,
base as
u32).map_err(|_| ConversionErrorKind::InvalidLiteralKind)
}
_ => Err(ConversionErrorKind::InvalidLiteralKind),
}
})
}integer_values! {
1768 u8 => u8_value,
1769 u16 => u16_value,
1770 u32 => u32_value,
1771 u64 => u64_value,
1772 u128 => u128_value,
1773 i8 => i8_value,
1774 i16 => i16_value,
1775 i32 => i32_value,
1776 i64 => i64_value,
1777 i128 => i128_value,
1778 }17791780#[doc =
"Returns the unescaped `f64` value if the literal is a `f64` or if it\'s an \"unmarked\" float which doesn\'t overflow."]
#[unstable(feature = "proc_macro_value", issue = "136652")]
pub fn f64_value(&self) -> Result<f64, ConversionErrorKind> {
if self.0.kind != bridge::LitKind::Float {
return Err(ConversionErrorKind::InvalidLiteralKind);
}
self.with_symbol_and_suffix(|symbol, suffix|
{
match suffix {
"f64" | "" => {
let number = strip_underscores(symbol);
f64::from_str(&number).map_err(|_|
ConversionErrorKind::InvalidLiteralKind)
}
_ => Err(ConversionErrorKind::InvalidLiteralKind),
}
})
}float_values! {
1781 f16 => f16_value,
1782 f32 => f32_value,
1783 f64 => f64_value,
1784// FIXME: `f128` doesn't implement `FromStr` for the moment so we cannot obtain it from
1785 // a `&str`. To be uncommented when it's added.
1786 // f128 => f128_value,
1787}1788}
17891790#[repr(u32)]
1791#[derive(#[automatically_derived]
impl ::core::cmp::PartialEq for Base {
#[inline]
fn eq(&self, other: &Base) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for Base {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
1792enum Base {
1793 Decimal = 10,
1794 Binary = 2,
1795 Octal = 8,
1796 Hexadecimal = 16,
1797}
17981799fn parse_number(value: &str) -> (&str, Base) {
1800let mut iter = value.as_bytes().iter().copied();
1801let Some(first_digit) = iter.next() else {
1802return ("0", Base::Decimal);
1803 };
1804let Some(second_digit) = iter.next() else {
1805return (value, Base::Decimal);
1806 };
18071808let mut base = Base::Decimal;
1809if first_digit == b'0' {
1810// Attempt to parse encoding base.
1811match second_digit {
1812b'b' => {
1813base = Base::Binary;
1814 }
1815b'o' => {
1816base = Base::Octal;
1817 }
1818b'x' => {
1819base = Base::Hexadecimal;
1820 }
1821_ => {}
1822 }
1823 }
18241825let offset = if base == Base::Decimal { 0 } else { 2 };
18261827 (&value[offset..], base)
1828}
18291830fn strip_underscores(value_s: &str) -> Cow<'_, str> {
1831let value = value_s.as_bytes();
1832if value.iter().copied().all(|c| c != b'_' && c != b'f') {
1833return Cow::Borrowed(value_s);
1834 }
1835let mut output = String::with_capacity(value.len());
1836for c in value.iter().copied() {
1837if c != b'_' {
1838 output.push(c as char);
1839 }
1840 }
1841 Cow::Owned(output)
1842}
18431844/// Parse a single literal from its stringified representation.
1845///
1846/// In order to parse successfully, the input string must not contain anything
1847/// but the literal token. Specifically, it must not contain whitespace or
1848/// comments in addition to the literal.
1849///
1850/// The resulting literal token will have a `Span::call_site()` span.
1851///
1852/// NOTE: some errors may cause panics instead of returning `LexError`. We
1853/// reserve the right to change these errors into `LexError`s later.
1854#[stable(feature = "proc_macro_literal_parse", since = "1.54.0")]
1855impl FromStrfor Literal {
1856type Err = LexError;
18571858fn from_str(src: &str) -> Result<Self, LexError> {
1859match BridgeMethods::literal_from_str(src) {
1860Ok(literal) => Ok(Literal(literal)),
1861Err(msg) => Err(LexError(msg)),
1862 }
1863 }
1864}
18651866/// Prints the literal as a string that should be losslessly convertible
1867/// back into the same literal (except for possible rounding for floating point literals).
1868#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1869impl fmt::Displayfor Literal {
1870fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1871self.with_stringify_parts(|parts| {
1872for part in parts {
1873 fmt::Display::fmt(part, f)?;
1874 }
1875Ok(())
1876 })
1877 }
1878}
18791880#[stable(feature = "proc_macro_lib2", since = "1.29.0")]
1881impl fmt::Debugfor Literal {
1882fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1883f.debug_struct("Literal")
1884// format the kind on one line even in {:#?} mode
1885 .field("kind", &format_args!("{0:?}", self.0.kind)format_args!("{:?}", self.0.kind))
1886 .field("symbol", &self.0.symbol)
1887// format `Some("...")` on one line even in {:#?} mode
1888 .field("suffix", &format_args!("{0:?}", self.0.suffix)format_args!("{:?}", self.0.suffix))
1889 .field("span", &self.0.span)
1890 .finish()
1891 }
1892}
18931894#[unstable(
1895 feature = "proc_macro_tracked_path",
1896 issue = "99515",
1897 implied_by = "proc_macro_tracked_env"
1898)]
1899/// Functionality for adding environment state to the build dependency info.
1900pub mod tracked {
1901use std::env::{self, VarError};
1902use std::ffi::OsStr;
1903use std::path::Path;
19041905use crate::BridgeMethods;
19061907/// Retrieve an environment variable and add it to build dependency info.
1908 /// The build system executing the compiler will know that the variable was accessed during
1909 /// compilation, and will be able to rerun the build when the value of that variable changes.
1910 /// Besides the dependency tracking this function should be equivalent to `env::var` from the
1911 /// standard library, except that the argument must be UTF-8.
1912#[unstable(feature = "proc_macro_tracked_env", issue = "99515")]
1913pub fn env_var<K: AsRef<OsStr> + AsRef<str>>(key: K) -> Result<String, VarError> {
1914let key: &str = key.as_ref();
1915let value = BridgeMethods::injected_env_var(key).map_or_else(|| env::var(key), Ok);
1916BridgeMethods::track_env_var(key, value.as_deref().ok());
1917value1918 }
19191920/// Track a file or directory explicitly.
1921 ///
1922 /// Commonly used for tracking asset preprocessing.
1923#[unstable(feature = "proc_macro_tracked_path", issue = "99515")]
1924pub fn path<P: AsRef<Path>>(path: P) {
1925let path: &str = path.as_ref().to_str().unwrap();
1926BridgeMethods::track_path(path);
1927 }
1928}