rust/compiler/rustc_span/src/symbol.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

2487 lines
64 KiB
Rust
Raw Normal View History

//! An "interner" is a data structure that associates values with usize tags and
//! allows bidirectional lookup; i.e., given a value, one can easily find the
//! type, and vice versa.
2020-06-02 20:19:49 +03:00
use rustc_arena::DroplessArena;
use rustc_data_structures::fx::FxIndexSet;
use rustc_data_structures::stable_hasher::{
HashStable, StableCompare, StableHasher, ToStableHashKey,
};
use rustc_data_structures::sync::Lock;
use rustc_macros::HashStable_Generic;
use std::fmt;
use std::hash::{Hash, Hasher};
use std::str;
use crate::{with_session_globals, Edition, Span, DUMMY_SP};
#[cfg(test)]
mod tests;
// The proc macro code for this is in `compiler/rustc_macros/src/symbols.rs`.
symbols! {
2023-11-02 15:31:51 +11:00
// If you modify this list, adjust `is_special` and `is_used_keyword`/`is_unused_keyword`.
// But this should rarely be necessary if the keywords are kept in alphabetic order.
Keywords {
// Special reserved identifiers used internally for elided lifetimes,
// unnamed method parameters, crate root module, error recovery etc.
Empty: "",
2019-04-09 09:39:48 +02:00
PathRoot: "{{root}}",
DollarCrate: "$crate",
Underscore: "_",
// Keywords that are used in stable Rust.
2019-04-09 09:39:48 +02:00
As: "as",
Break: "break",
Const: "const",
Continue: "continue",
Crate: "crate",
Else: "else",
Enum: "enum",
Extern: "extern",
False: "false",
Fn: "fn",
For: "for",
If: "if",
Impl: "impl",
In: "in",
Let: "let",
Loop: "loop",
Match: "match",
Mod: "mod",
Move: "move",
Mut: "mut",
Pub: "pub",
Ref: "ref",
Return: "return",
SelfLower: "self",
SelfUpper: "Self",
Static: "static",
Struct: "struct",
Super: "super",
Trait: "trait",
True: "true",
Type: "type",
Unsafe: "unsafe",
Use: "use",
Where: "where",
While: "while",
// Keywords that are used in unstable Rust or reserved for future use.
2019-04-09 09:39:48 +02:00
Abstract: "abstract",
Become: "become",
Box: "box",
2019-04-09 09:39:48 +02:00
Do: "do",
Final: "final",
Macro: "macro",
Override: "override",
Priv: "priv",
Typeof: "typeof",
Unsized: "unsized",
Virtual: "virtual",
Yield: "yield",
// Edition-specific keywords that are used in stable Rust.
Async: "async", // >= 2018 Edition only
Await: "await", // >= 2018 Edition only
2019-04-09 09:39:48 +02:00
Dyn: "dyn", // >= 2018 Edition only
// Edition-specific keywords that are used in unstable Rust or reserved for future use.
2019-04-09 09:39:48 +02:00
Try: "try", // >= 2018 Edition only
// Special lifetime names
2019-04-09 09:39:48 +02:00
UnderscoreLifetime: "'_",
StaticLifetime: "'static",
// Weak keywords, have special meaning only in specific contexts.
2019-04-09 09:39:48 +02:00
Auto: "auto",
Builtin: "builtin",
2019-04-09 09:39:48 +02:00
Catch: "catch",
Default: "default",
Gen: "gen",
MacroRules: "macro_rules",
Raw: "raw",
2023-11-26 15:57:31 +03:00
Reuse: "reuse",
2019-04-09 09:39:48 +02:00
Union: "union",
Yeet: "yeet",
}
2020-07-08 11:54:27 +10:00
// Pre-interned symbols that can be referred to with `rustc_span::sym::*`.
//
// The symbol is the stringified identifier unless otherwise specified, in
// which case the name should mention the non-identifier punctuation.
// E.g. `sym::proc_dash_macro` represents "proc-macro", and it shouldn't be
// called `sym::proc_macro` because then it's easy to mistakenly think it
// represents "proc_macro".
//
2020-09-21 12:14:28 +09:00
// As well as the symbols listed, there are symbols for the strings
// "0", "1", ..., "9", which are accessible via `sym::integer`.
//
// The proc macro will abort if symbols are not in alphabetical order (as
// defined by `impl Ord for str`) or if any symbols are duplicated. Vim
// users can sort the list by selecting it and executing the command
// `:'<,'>!LC_ALL=C sort`.
//
// There is currently no checking that all symbols are used; that would be
// nice to have.
Symbols {
Abi,
AcqRel,
Acquire,
Any,
Arc,
2023-09-26 23:56:38 -04:00
ArcWeak,
Argument,
ArgumentMethods,
2023-09-26 23:56:38 -04:00
ArrayIntoIter,
AsMut,
AsRef,
AssertParamIsClone,
AssertParamIsCopy,
AssertParamIsEq,
2023-11-28 18:18:19 +00:00
AsyncGenFinished,
AsyncGenPending,
AsyncGenReady,
AtomicBool,
AtomicI128,
AtomicI16,
AtomicI32,
AtomicI64,
AtomicI8,
AtomicIsize,
AtomicPtr,
AtomicU128,
AtomicU16,
AtomicU32,
AtomicU64,
AtomicU8,
AtomicUsize,
BTreeEntry,
2021-02-26 21:23:42 -06:00
BTreeMap,
BTreeSet,
BinaryHeap,
2021-01-05 16:46:50 +01:00
Borrow,
BorrowMut,
Break,
C,
2021-05-18 15:33:38 +02:00
CStr,
CallOnceFuture,
CallRefFuture,
Capture,
Center,
Cleanup,
Clone,
2023-09-26 23:56:38 -04:00
Command,
ConstParamTy,
Context,
Continue,
Copy,
Cow,
Debug,
2023-09-26 23:56:38 -04:00
DebugStruct,
Decodable,
Decoder,
Default,
2021-02-16 22:39:05 +01:00
Deref,
DiagMessage,
Diagnostic,
DirBuilder,
2021-10-02 18:51:01 -05:00
Display,
DoubleEndedIterator,
Duration,
Encodable,
Encoder,
Eq,
Equal,
Err,
Error,
File,
FileType,
Fn,
FnMut,
FnOnce,
Formatter,
From,
FromIterator,
2021-12-09 22:42:17 +08:00
FromResidual,
2023-09-26 23:56:38 -04:00
FsOpenOptions,
FsPermissions,
FusedIterator,
Future,
2022-12-27 23:56:46 +00:00
FutureOutput,
GlobalAlloc,
Hash,
HashMap,
HashMapEntry,
HashSet,
Hasher,
Implied,
InCleanup,
IndexOutput,
Input,
2023-09-26 23:56:38 -04:00
Instant,
2021-10-02 18:51:01 -05:00
Into,
2023-04-25 19:48:59 +00:00
IntoFuture,
IntoIterator,
2023-09-26 23:56:38 -04:00
IoLines,
IoRead,
2023-09-26 23:56:38 -04:00
IoSeek,
IoWrite,
2022-09-12 19:03:24 +02:00
IpAddr,
IrTyKind,
Is,
Item,
ItemContext,
2023-09-26 23:56:38 -04:00
IterEmpty,
IterOnce,
IterPeekable,
Iterator,
IteratorItem,
Layout,
Left,
2021-02-26 21:23:42 -06:00
LinkedList,
LintDiagnostic,
LintPass,
2022-09-26 00:55:35 +02:00
LocalKey,
2021-10-02 18:51:01 -05:00
Mutex,
MutexGuard,
N,
2023-09-26 23:56:38 -04:00
NonNull,
2024-01-20 18:25:52 +01:00
NonZero,
None,
Normal,
Ok,
Option,
Ord,
Ordering,
2021-02-16 02:32:21 +00:00
OsStr,
2021-02-15 22:58:03 +00:00
OsString,
Output,
Param,
PartialEq,
PartialOrd,
2021-02-16 02:32:21 +00:00
Path,
2021-02-15 22:58:03 +00:00
PathBuf,
Pending,
2021-10-02 18:51:01 -05:00
Pointer,
Poll,
ProcMacro,
ProceduralMasqueradeDummyType,
Range,
2023-09-26 23:56:38 -04:00
RangeBounds,
RangeFrom,
RangeFull,
RangeInclusive,
RangeTo,
RangeToInclusive,
Rc,
2023-09-26 23:56:38 -04:00
RcWeak,
Ready,
Receiver,
2022-09-26 00:55:35 +02:00
RefCell,
2023-09-26 23:56:38 -04:00
RefCellRef,
RefCellRefMut,
Relaxed,
Release,
Result,
ResumeTy,
Return,
Right,
2022-07-18 14:25:34 +09:00
Rust,
RustcDecodable,
RustcEncodable,
2022-09-27 13:06:31 +02:00
RwLock,
RwLockReadGuard,
RwLockWriteGuard,
2023-08-24 13:30:53 -03:00
Saturating,
Send,
SeqCst,
SliceIndex,
2023-09-26 23:56:38 -04:00
SliceIter,
Some,
2023-10-16 01:05:11 -07:00
SpanCtxt,
2021-10-02 18:51:01 -05:00
String,
StructuralPartialEq,
SubdiagMessage,
Subdiagnostic,
Sync,
T,
Target,
2021-02-15 22:58:03 +00:00
ToOwned,
2021-02-16 02:32:21 +00:00
ToString,
TokenStream,
Trait,
Try,
TryCaptureGeneric,
TryCapturePrintable,
2021-10-02 18:51:01 -05:00
TryFrom,
TryInto,
Ty,
TyCtxt,
TyKind,
Unknown,
2024-01-25 03:50:23 +00:00
Upvars,
Vec,
2021-10-02 18:51:01 -05:00
VecDeque,
Wrapper,
2023-08-24 13:30:53 -03:00
Wrapping,
Yield,
_DECLS,
_Self,
__D,
__H,
__S,
__awaitee,
__try_var,
_d,
_e,
_task_context,
a32,
aarch64_target_feature,
2022-02-10 16:43:45 +00:00
aarch64_ver_target_feature,
abi,
abi_amdgpu_kernel,
abi_avr_interrupt,
abi_c_cmse_nonsecure_call,
abi_efiapi,
abi_msp430_interrupt,
abi_ptx,
feat: `riscv-interrupt-{m,s}` calling conventions Similar to prior support added for the mips430, avr, and x86 targets this change implements the rough equivalent of clang's [`__attribute__((interrupt))`][clang-attr] for riscv targets, enabling e.g. ```rust static mut CNT: usize = 0; pub extern "riscv-interrupt-m" fn isr_m() { unsafe { CNT += 1; } } ``` to produce highly effective assembly like: ```asm pub extern "riscv-interrupt-m" fn isr_m() { 420003a0: 1141 addi sp,sp,-16 unsafe { CNT += 1; 420003a2: c62a sw a0,12(sp) 420003a4: c42e sw a1,8(sp) 420003a6: 3fc80537 lui a0,0x3fc80 420003aa: 63c52583 lw a1,1596(a0) # 3fc8063c <_ZN12esp_riscv_rt3CNT17hcec3e3a214887d53E.0> 420003ae: 0585 addi a1,a1,1 420003b0: 62b52e23 sw a1,1596(a0) } } 420003b4: 4532 lw a0,12(sp) 420003b6: 45a2 lw a1,8(sp) 420003b8: 0141 addi sp,sp,16 420003ba: 30200073 mret ``` (disassembly via `riscv64-unknown-elf-objdump -C -S --disassemble ./esp32c3-hal/target/riscv32imc-unknown-none-elf/release/examples/gpio_interrupt`) This outcome is superior to hand-coded interrupt routines which, lacking visibility into any non-assembly body of the interrupt handler, have to be very conservative and save the [entire CPU state to the stack frame][full-frame-save]. By instead asking LLVM to only save the registers that it uses, we defer the decision to the tool with the best context: it can more accurately account for the cost of spills if it knows that every additional register used is already at the cost of an implicit spill. At the LLVM level, this is apparently [implemented by] marking every register as "[callee-save]," matching the semantics of an interrupt handler nicely (it has to leave the CPU state just as it found it after its `{m|s}ret`). This approach is not suitable for every interrupt handler, as it makes no attempt to e.g. save the state in a user-accessible stack frame. For a full discussion of those challenges and tradeoffs, please refer to [the interrupt calling conventions RFC][rfc]. Inside rustc, this implementation differs from prior art because LLVM does not expose the "all-saved" function flavor as a calling convention directly, instead preferring to use an attribute that allows for differentiating between "machine-mode" and "superivsor-mode" interrupts. Finally, some effort has been made to guide those who may not yet be aware of the differences between machine-mode and supervisor-mode interrupts as to why no `riscv-interrupt` calling convention is exposed through rustc, and similarly for why `riscv-interrupt-u` makes no appearance (as it would complicate future LLVM upgrades). [clang-attr]: https://clang.llvm.org/docs/AttributeReference.html#interrupt-risc-v [full-frame-save]: https://github.com/esp-rs/esp-riscv-rt/blob/9281af2ecffe13e40992917316f36920c26acaf3/src/lib.rs#L440-L469 [implemented by]: https://github.com/llvm/llvm-project/blob/b7fb2a3fec7c187d58a6d338ab512d9173bca987/llvm/lib/Target/RISCV/RISCVRegisterInfo.cpp#L61-L67 [callee-save]: https://github.com/llvm/llvm-project/blob/973f1fe7a8591c7af148e573491ab68cc15b6ecf/llvm/lib/Target/RISCV/RISCVCallingConv.td#L30-L37 [rfc]: https://github.com/rust-lang/rfcs/pull/3246
2023-05-23 15:08:23 -07:00
abi_riscv_interrupt,
abi_sysv64,
abi_thiscall,
abi_unadjusted,
abi_vectorcall,
abi_x86_interrupt,
abort,
add,
add_assign,
add_with_overflow,
address,
adt_const_params,
advanced_slice_patterns,
adx_target_feature,
alias,
align,
align_offset,
alignment,
all,
alloc,
alloc_error_handler,
alloc_layout,
alloc_zeroed,
allocator,
allocator_api,
allocator_internals,
allow,
allow_fail,
allow_internal_unsafe,
allow_internal_unstable,
alu32,
always,
and,
2020-07-14 16:29:44 +10:00
and_then,
2023-04-11 14:24:59 +10:00
anon,
anon_adt,
2022-06-22 22:19:02 +02:00
anonymous_lifetime_in_impl_trait,
any,
append_const_msg,
2019-05-09 17:08:55 -04:00
arbitrary_enum_discriminant,
arbitrary_self_types,
args,
arith_offset,
arm,
arm_target_feature,
array,
2020-08-18 17:02:23 -04:00
as_ptr,
as_ref,
as_str,
asm,
asm_const,
asm_experimental_arch,
2023-12-25 20:53:01 +00:00
asm_goto,
asm_sym,
asm_unwind,
assert,
assert_eq,
2022-01-06 14:50:46 +00:00
assert_eq_macro,
assert_inhabited,
assert_macro,
assert_mem_uninitialized_valid,
2022-01-06 14:50:46 +00:00
assert_ne_macro,
assert_receiver_is_total_eq,
assert_zero_valid,
asserting,
associated_const_equality,
associated_consts,
2019-05-20 00:22:44 +01:00
associated_type_bounds,
associated_type_defaults,
associated_types,
assume,
assume_init,
async_await,
async_call,
async_call_mut,
async_call_once,
async_closure,
async_fn,
async_fn_in_trait,
async_fn_kind_helper,
async_fn_mut,
async_fn_once,
async_fn_track_caller,
async_fn_traits,
2023-12-08 14:51:50 -08:00
async_for_loop,
2023-11-28 18:18:19 +00:00
async_iterator,
2023-12-08 17:00:11 -08:00
async_iterator_poll_next,
atomic,
atomic_mod,
atomics,
att_syntax,
attr,
attr_literals,
attributes,
augmented_assignments,
auto_traits,
automatically_derived,
avx,
avx512_target_feature,
avx512bw,
avx512f,
await_macro,
bang,
begin_panic,
bench,
bin,
bind_by_move_pattern_guards,
bindings_after_at,
bitand,
bitand_assign,
bitor,
bitor_assign,
bitreverse,
bitxor,
bitxor_assign,
black_box,
block,
bool,
borrowck_graphviz_format,
borrowck_graphviz_postflow,
box_new,
box_patterns,
box_syntax,
2021-04-20 19:03:10 +10:00
bpf_target_feature,
braced_empty_structs,
branch,
breakpoint,
bridge,
bswap,
builtin_syntax,
c,
2020-08-18 17:02:23 -04:00
c_str,
2023-03-06 07:10:23 +00:00
c_str_literals,
rustc_target: add "unwind" payloads to `Abi` ### Overview This commit begins the implementation work for RFC 2945. For more information, see the rendered RFC [1] and tracking issue [2]. A boolean `unwind` payload is added to the `C`, `System`, `Stdcall`, and `Thiscall` variants, marking whether unwinding across FFI boundaries is acceptable. The cases where each of these variants' `unwind` member is true correspond with the `C-unwind`, `system-unwind`, `stdcall-unwind`, and `thiscall-unwind` ABI strings introduced in RFC 2945 [3]. ### Feature Gate and Unstable Book This commit adds a `c_unwind` feature gate for the new ABI strings. Tests for this feature gate are included in `src/test/ui/c-unwind/`, which ensure that this feature gate works correctly for each of the new ABIs. A new language features entry in the unstable book is added as well. ### Further Work To Be Done This commit does not proceed to implement the new unwinding ABIs, and is intentionally scoped specifically to *defining* the ABIs and their feature flag. ### One Note on Test Churn This will lead to some test churn, in re-blessing hash tests, as the deleted comment in `src/librustc_target/spec/abi.rs` mentioned, because we can no longer guarantee the ordering of the `Abi` variants. While this is a downside, this decision was made bearing in mind that RFC 2945 states the following, in the "Other `unwind` Strings" section [3]: > More unwind variants of existing ABI strings may be introduced, > with the same semantics, without an additional RFC. Adding a new variant for each of these cases, rather than specifying a payload for a given ABI, would quickly become untenable, and make working with the `Abi` enum prone to mistakes. This approach encodes the unwinding information *into* a given ABI, to account for the future possibility of other `-unwind` ABI strings. ### Ignore Directives `ignore-*` directives are used in two of our `*-unwind` ABI test cases. Specifically, the `stdcall-unwind` and `thiscall-unwind` test cases ignore architectures that do not support `stdcall` and `thiscall`, respectively. These directives are cribbed from `src/test/ui/c-variadic/variadic-ffi-1.rs` for `stdcall`, and `src/test/ui/extern/extern-thiscall.rs` for `thiscall`. This would otherwise fail on some targets, see: https://github.com/rust-lang-ci/rust/commit/fcf697f90206e9c87b39d494f94ab35d976bfc60 ### Footnotes [1]: https://github.com/rust-lang/rfcs/blob/master/text/2945-c-unwind-abi.md [2]: https://github.com/rust-lang/rust/issues/74990 [3]: https://github.com/rust-lang/rfcs/blob/master/text/2945-c-unwind-abi.md#other-unwind-abi-strings
2020-08-27 11:49:18 -04:00
c_unwind,
c_variadic,
c_void,
call,
call_mut,
call_once,
caller_location,
capture_disjoint_fields,
catch_unwind,
cause,
cdylib,
2024-03-01 03:53:26 -05:00
ceilf128,
ceilf16,
ceilf32,
ceilf64,
cfg,
cfg_accessible,
cfg_attr,
cfg_attr_multi,
cfg_doctest,
cfg_eval,
cfg_hide,
cfg_overflow_checks,
2020-07-25 19:02:49 +01:00
cfg_panic,
cfg_relocation_model,
cfg_sanitize,
cfg_sanitizer_cfi,
cfg_target_abi,
cfg_target_compact,
cfg_target_feature,
cfg_target_has_atomic,
cfg_target_has_atomic_equal_alignment,
cfg_target_thread_local,
cfg_target_vendor,
cfg_version,
cfi,
cfi_encoding,
char,
client,
2019-06-24 18:14:04 +02:00
clippy,
2021-07-29 13:43:26 +02:00
clobber_abi,
clone,
clone_closures,
clone_from,
closure,
closure_lifetime_binder,
closure_to_fn_coercion,
Support `#[track_caller]` on closures and generators This PR allows applying a `#[track_caller]` attribute to a closure/generator expression. The attribute as interpreted as applying to the compiler-generated implementation of the corresponding trait method (`FnOnce::call_once`, `FnMut::call_mut`, `Fn::call`, or `Generator::resume`). This feature does not have its own feature gate - however, it requires `#![feature(stmt_expr_attributes)]` in order to actually apply an attribute to a closure or generator. This is implemented in the same way as for functions - an extra location argument is appended to the end of the ABI. For closures, this argument is *not* part of the 'tupled' argument storing the parameters - the final closure argument for `#[track_caller]` closures is no longer a tuple. For direct (monomorphized) calls, the necessary support was already implemented - we just needeed to adjust some assertions around checking the ABI and argument count to take closures into account. For calls through a trait object, more work was needed. When creating a `ReifyShim`, we need to create a shim for the trait method (e.g. `FnOnce::call_mut`) - unlike normal functions, closures are never invoked directly, and always go through a trait method. Additional handling was needed for `InstanceDef::ClosureOnceShim`. In order to pass location information throgh a direct (monomorphized) call to `FnOnce::call_once` on an `FnMut` closure, we need to make `ClosureOnceShim` aware of `#[tracked_caller]`. A new field `track_caller` is added to `ClosureOnceShim` - this is used by `InstanceDef::requires_caller` location, allowing codegen to pass through the extra location argument. Since `ClosureOnceShim.track_caller` is only used by codegen, we end up generating two identical MIR shims - one for `track_caller == true`, and one for `track_caller == false`. However, these two shims are used by the entire crate (i.e. it's two shims total, not two shims per unique closure), so this shouldn't a big deal.
2021-06-27 14:01:11 -05:00
closure_track_caller,
cmp,
cmp_max,
cmp_min,
cmp_ord_max,
cmp_ord_min,
cmp_partialeq_eq,
cmp_partialeq_ne,
cmp_partialord_cmp,
cmp_partialord_ge,
cmp_partialord_gt,
cmp_partialord_le,
cmp_partialord_lt,
cmpxchg16b_target_feature,
cmse_nonsecure_entry,
coerce_unsized,
cold,
collapse_debuginfo,
column,
compare_bytes,
compare_exchange,
compare_exchange_weak,
compile_error,
compiler,
compiler_builtins,
compiler_fence,
concat,
concat_bytes,
concat_idents,
conservative_impl_trait,
console,
2020-12-01 15:39:25 +05:30
const_allocate,
const_async_blocks,
2022-12-21 14:51:02 +00:00
const_closures,
const_compare_raw_pointers,
const_constructor,
2021-12-25 22:35:11 +09:00
const_deallocate,
const_eval_limit,
2021-10-12 05:06:37 +00:00
const_eval_select,
const_evaluatable_checked,
const_extern_fn,
const_fn,
const_fn_floating_point_arithmetic,
const_fn_fn_ptr_basics,
const_fn_trait_bound,
2020-07-11 08:55:05 -07:00
const_fn_transmute,
const_fn_union,
const_fn_unsize,
const_for,
const_format_args,
const_generics,
const_generics_defaults,
const_if_match,
2020-10-05 20:30:13 -07:00
const_impl_trait,
const_in_array_repeat_expressions,
const_indexing,
const_let,
2019-12-10 12:41:58 -08:00
const_loop,
2019-11-21 12:38:14 -05:00
const_mut_refs,
const_panic,
const_panic_fmt,
2023-02-17 13:44:35 +00:00
const_param_ty,
const_precise_live_drops,
const_ptr_cast,
const_raw_ptr_deref,
const_raw_ptr_to_usize_cast,
const_refs_to_cell,
const_refs_to_static,
const_trait,
const_trait_bound_opt_out,
2020-01-02 15:45:48 -08:00
const_trait_impl,
const_try,
constant,
constructor,
2023-09-26 23:56:38 -04:00
convert_identity,
copy,
copy_closures,
copy_nonoverlapping,
2024-03-01 03:53:26 -05:00
copysignf128,
copysignf16,
copysignf32,
copysignf64,
core,
core_panic,
core_panic_2015_macro,
core_panic_2021_macro,
core_panic_macro,
2023-10-19 21:46:28 +00:00
coroutine,
coroutine_clone,
coroutine_resume,
2023-10-19 21:46:28 +00:00
coroutine_state,
coroutines,
2024-03-01 03:53:26 -05:00
cosf128,
cosf16,
cosf32,
cosf64,
count,
2023-08-09 15:57:16 +01:00
coverage,
coverage_attribute,
cr,
crate_id,
crate_in_paths,
crate_local,
crate_name,
crate_type,
crate_visibility_modifier,
crt_dash_static: "crt-static",
csky_target_feature,
2020-09-21 16:32:28 -04:00
cstring_type,
ctlz,
ctlz_nonzero,
ctpop,
cttz,
cttz_nonzero,
custom_attribute,
custom_code_classes_in_docs,
custom_derive,
custom_inner_attributes,
2022-08-03 04:30:13 -07:00
custom_mir,
custom_test_frameworks,
d,
d32,
2022-01-06 14:50:46 +00:00
dbg_macro,
dead_code,
dealloc,
debug,
2022-01-06 14:50:46 +00:00
debug_assert_eq_macro,
debug_assert_macro,
2022-01-06 14:50:46 +00:00
debug_assert_ne_macro,
debug_assertions,
debug_struct,
Optimize the code produced by `derive(Debug)`. This commit adds new methods that combine sequences of existing formatting methods. - `Formatter::debug_{tuple,struct}_field[12345]_finish`, equivalent to a `Formatter::debug_{tuple,struct}` + N x `Debug{Tuple,Struct}::field` + `Debug{Tuple,Struct}::finish` call sequence. - `Formatter::debug_{tuple,struct}_fields_finish` is similar, but can handle any number of fields by using arrays. These new methods are all marked as `doc(hidden)` and unstable. They are intended for the compiler's own use. Special-casing up to 5 fields gives significantly better performance results than always using arrays (as was tried in #95637). The commit also changes the `Debug` deriving code to use these new methods. For example, where the old `Debug` code for a struct with two fields would be like this: ``` fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result { match *self { Self { f1: ref __self_0_0, f2: ref __self_0_1, } => { let debug_trait_builder = &mut ::core::fmt::Formatter::debug_struct(f, "S2"); let _ = ::core::fmt::DebugStruct::field(debug_trait_builder, "f1", &&(*__self_0_0)); let _ = ::core::fmt::DebugStruct::field(debug_trait_builder, "f2", &&(*__self_0_1)); ::core::fmt::DebugStruct::finish(debug_trait_builder) } } } ``` the new code is like this: ``` fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result { match *self { Self { f1: ref __self_0_0, f2: ref __self_0_1, } => ::core::fmt::Formatter::debug_struct_field2_finish( f, "S2", "f1", &&(*__self_0_0), "f2", &&(*__self_0_1), ), } } ``` This shrinks the code produced for `Debug` instances considerably, reducing compile times and binary sizes. Co-authored-by: Scott McMurray <scottmcm@users.noreply.github.com>
2022-06-17 17:08:36 +10:00
debug_struct_fields_finish,
debug_tuple,
Optimize the code produced by `derive(Debug)`. This commit adds new methods that combine sequences of existing formatting methods. - `Formatter::debug_{tuple,struct}_field[12345]_finish`, equivalent to a `Formatter::debug_{tuple,struct}` + N x `Debug{Tuple,Struct}::field` + `Debug{Tuple,Struct}::finish` call sequence. - `Formatter::debug_{tuple,struct}_fields_finish` is similar, but can handle any number of fields by using arrays. These new methods are all marked as `doc(hidden)` and unstable. They are intended for the compiler's own use. Special-casing up to 5 fields gives significantly better performance results than always using arrays (as was tried in #95637). The commit also changes the `Debug` deriving code to use these new methods. For example, where the old `Debug` code for a struct with two fields would be like this: ``` fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result { match *self { Self { f1: ref __self_0_0, f2: ref __self_0_1, } => { let debug_trait_builder = &mut ::core::fmt::Formatter::debug_struct(f, "S2"); let _ = ::core::fmt::DebugStruct::field(debug_trait_builder, "f1", &&(*__self_0_0)); let _ = ::core::fmt::DebugStruct::field(debug_trait_builder, "f2", &&(*__self_0_1)); ::core::fmt::DebugStruct::finish(debug_trait_builder) } } } ``` the new code is like this: ``` fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result { match *self { Self { f1: ref __self_0_0, f2: ref __self_0_1, } => ::core::fmt::Formatter::debug_struct_field2_finish( f, "S2", "f1", &&(*__self_0_0), "f2", &&(*__self_0_1), ), } } ``` This shrinks the code produced for `Debug` instances considerably, reducing compile times and binary sizes. Co-authored-by: Scott McMurray <scottmcm@users.noreply.github.com>
2022-06-17 17:08:36 +10:00
debug_tuple_fields_finish,
debugger_visualizer,
decl_macro,
declare_lint_pass,
decode,
default_alloc_error_handler,
2023-09-26 23:56:38 -04:00
default_fn,
default_lib_allocator,
2021-07-04 12:24:20 +08:00
default_method_body_is_const,
default_type_parameter_fallback,
default_type_params,
delayed_bug_from_inside_query,
deny,
deprecated,
deprecated_safe,
deprecated_suggestion,
deref,
deref_method,
deref_mut,
2023-09-26 23:56:38 -04:00
deref_mut_method,
deref_patterns,
2024-03-21 14:37:43 -04:00
deref_pure,
deref_target,
derive,
2022-09-20 11:55:07 +00:00
derive_const,
derive_default_enum,
destruct,
destructuring_assignment,
diagnostic,
diagnostic_namespace,
direct,
discriminant_kind,
discriminant_type,
2020-03-08 14:24:32 +01:00
discriminant_value,
dispatch_from_dyn,
div,
div_assign,
diverging_block_default,
2023-01-09 20:51:01 -03:00
do_not_recommend,
doc,
doc_alias,
doc_auto_cfg,
doc_cfg,
doc_cfg_hide,
doc_keyword,
doc_masked,
doc_notable_trait,
2021-07-10 22:06:10 -04:00
doc_primitive,
doc_spotlight,
doctest,
document_private_items,
dotdot: "..",
dotdot_in_tuple_patterns,
dotdoteq_in_patterns,
dreg,
dreg_low16,
dreg_low8,
drop,
drop_in_place,
drop_types_in_const,
dropck_eyepatch,
dropck_parametricity,
dylib,
dyn_metadata,
dyn_star,
dyn_trait,
dynamic_no_pic: "dynamic-no-pic",
e,
edition_panic,
2023-06-30 13:53:23 +00:00
effects,
eh_catch_typeinfo,
eh_personality,
emit,
emit_enum,
emit_enum_variant,
emit_enum_variant_arg,
emit_struct,
emit_struct_field,
enable,
encode,
end,
env,
env_CFG_RELEASE: env!("CFG_RELEASE"),
2022-01-06 14:50:46 +00:00
eprint_macro,
eprintln_macro,
eq,
ermsb_target_feature,
exact_div,
except,
exchange_malloc,
exclusive_range_pattern,
exhaustive_integer_patterns,
exhaustive_patterns,
existential_type,
2024-03-01 03:53:26 -05:00
exp2f128,
exp2f16,
exp2f32,
exp2f64,
2020-08-18 17:02:23 -04:00
expect,
expected,
2024-03-01 03:53:26 -05:00
expf128,
expf16,
expf32,
expf64,
explicit_generic_args_with_impl_trait,
explicit_tail_calls,
export_name,
expr,
extended_key_value_attributes,
extended_varargs_abi_support,
extern_absolute_paths,
extern_crate_item_prelude,
extern_crate_self,
extern_in_paths,
extern_prelude,
extern_types,
external,
external_doc,
f,
f128,
2024-03-01 03:53:26 -05:00
f128_nan,
f16,
2024-03-01 03:53:26 -05:00
f16_nan,
f16c_target_feature,
2019-05-19 19:56:45 +03:00
f32,
f32_legacy_const_digits,
f32_legacy_const_epsilon,
f32_legacy_const_infinity,
f32_legacy_const_mantissa_dig,
f32_legacy_const_max,
f32_legacy_const_max_10_exp,
f32_legacy_const_max_exp,
f32_legacy_const_min,
f32_legacy_const_min_10_exp,
f32_legacy_const_min_exp,
f32_legacy_const_min_positive,
f32_legacy_const_nan,
f32_legacy_const_neg_infinity,
f32_legacy_const_radix,
f32_nan,
2019-05-19 19:56:45 +03:00
f64,
f64_legacy_const_digits,
f64_legacy_const_epsilon,
f64_legacy_const_infinity,
f64_legacy_const_mantissa_dig,
f64_legacy_const_max,
f64_legacy_const_max_10_exp,
f64_legacy_const_max_exp,
f64_legacy_const_min,
f64_legacy_const_min_10_exp,
f64_legacy_const_min_exp,
f64_legacy_const_min_positive,
f64_legacy_const_nan,
f64_legacy_const_neg_infinity,
f64_legacy_const_radix,
f64_nan,
2024-03-01 03:53:26 -05:00
fabsf128,
fabsf16,
fabsf32,
fabsf64,
fadd_algebraic,
fadd_fast,
fake_variadic,
fallback,
fdiv_algebraic,
fdiv_fast,
feature,
fence,
2021-12-03 01:20:25 +01:00
ferris: "🦀",
fetch_update,
2020-08-18 17:02:23 -04:00
ffi,
ffi_const,
ffi_pure,
ffi_returns_twice,
field,
field_init_shorthand,
file,
float,
float_to_int_unchecked,
2024-03-01 03:53:26 -05:00
floorf128,
floorf16,
floorf32,
floorf64,
2024-03-01 03:53:26 -05:00
fmaf128,
fmaf16,
fmaf32,
fmaf64,
fmt,
fmul_algebraic,
fmul_fast,
fn_align,
fn_delegation,
fn_must_use,
fn_mut,
fn_once,
fn_once_output,
2022-07-20 14:32:58 +02:00
fn_ptr_addr,
fn_ptr_trait,
forbid,
forget,
format,
format_alignment,
format_args,
2019-10-12 15:07:13 +01:00
format_args_capture,
2022-01-06 14:50:46 +00:00
format_args_macro,
format_args_nl,
format_argument,
format_arguments,
format_count,
format_macro,
format_placeholder,
format_unsafe_arg,
freeze,
freeze_impls,
freg,
frem_algebraic,
frem_fast,
from,
from_desugaring,
from_fn,
from_iter,
2023-09-26 23:56:38 -04:00
from_iter_fn,
from_output,
from_residual,
from_size_align_unchecked,
2023-09-26 23:56:38 -04:00
from_str_method,
from_usize,
from_yeet,
2023-09-26 23:56:38 -04:00
fs_create_dir,
fsub_algebraic,
fsub_fast,
fundamental,
fused_iterator,
future,
future_trait,
gdb_script_file,
ge,
gen_blocks,
gen_future,
gen_kill,
generator_clone,
generators,
2021-05-06 15:33:44 +00:00
generic_arg_infer,
2022-06-02 09:00:04 -03:00
generic_assert,
generic_associated_types,
generic_associated_types_extended,
generic_const_exprs,
2023-05-04 16:08:33 +02:00
generic_const_items,
generic_param_attrs,
get_context,
global_alloc_ty,
global_allocator,
global_asm,
globs,
gt,
half_open_range_patterns,
half_open_range_patterns_in_slices,
hash,
hexagon_target_feature,
hidden,
homogeneous_aggregate,
2023-06-30 13:53:23 +00:00
host,
html_favicon_url,
html_logo_url,
html_no_source,
html_playground_url,
html_root_url,
2021-01-22 18:32:38 -08:00
hwaddress,
i,
i128,
i128_legacy_const_max,
i128_legacy_const_min,
i128_legacy_fn_max_value,
i128_legacy_fn_min_value,
i128_legacy_mod,
i128_type,
i16,
i16_legacy_const_max,
i16_legacy_const_min,
i16_legacy_fn_max_value,
i16_legacy_fn_min_value,
i16_legacy_mod,
i32,
i32_legacy_const_max,
i32_legacy_const_min,
i32_legacy_fn_max_value,
i32_legacy_fn_min_value,
i32_legacy_mod,
i64,
i64_legacy_const_max,
i64_legacy_const_min,
i64_legacy_fn_max_value,
i64_legacy_fn_min_value,
i64_legacy_mod,
i8,
i8_legacy_const_max,
i8_legacy_const_min,
i8_legacy_fn_max_value,
i8_legacy_fn_min_value,
i8_legacy_mod,
ident,
if_let,
2020-07-14 07:53:23 +00:00
if_let_guard,
if_while_or_patterns,
ignore,
impl_header_lifetime_elision,
impl_lint_pass,
impl_trait_in_assoc_type,
impl_trait_in_bindings,
impl_trait_in_fn_trait_return,
impl_trait_projections,
2023-06-16 23:45:01 +00:00
implement_via_object,
implied_by,
import,
2022-07-12 13:52:35 -07:00
import_name_type,
import_shadowing,
imported_main,
in_band_lifetimes,
include,
include_bytes,
2022-01-06 14:50:46 +00:00
include_bytes_macro,
include_str,
2022-01-06 14:50:46 +00:00
include_str_macro,
inclusive_range_syntax,
index,
index_mut,
infer_outlives_requirements,
infer_static_outlives_requirements,
inherent_associated_types,
inherit,
inlateout,
inline,
2020-09-21 16:10:31 -03:00
inline_const,
inline_const_pat,
inout,
instruction_set,
2023-11-02 15:31:51 +11:00
integer_: "integer", // underscore to avoid clashing with the function `sym::integer` below
integral,
2023-12-08 17:00:11 -08:00
into_async_iter_into_iter,
into_future,
into_iter,
intra_doc_pointers,
intrinsics,
2023-09-26 23:56:38 -04:00
intrinsics_unaligned_volatile_load,
intrinsics_unaligned_volatile_store,
io_stderr,
io_stdout,
irrefutable_let_patterns,
is_val_statically_known,
isa_attribute,
isize,
isize_legacy_const_max,
isize_legacy_const_min,
isize_legacy_fn_max_value,
isize_legacy_fn_min_value,
isize_legacy_mod,
issue,
issue_5723_bootstrap,
issue_tracker_base_url,
item,
item_like_imports,
iter,
iter_mut,
iter_repeat,
2023-10-23 09:41:30 +00:00
iterator,
2022-12-09 22:31:16 -08:00
iterator_collect_fn,
kcfi,
keyword,
kind,
kreg,
kreg0,
label,
label_break_value,
2024-02-12 09:06:45 -03:00
lahfsahf_target_feature,
lang,
lang_items,
large_assignments,
lateout,
lazy_normalization_consts,
2023-06-19 13:50:22 +02:00
lazy_type_alias,
le,
2021-06-20 16:09:42 +02:00
len,
2019-05-11 15:11:04 +02:00
let_chains,
2021-06-23 16:40:06 -05:00
let_else,
lhs,
lib,
libc,
lifetime,
2023-10-19 19:00:18 +00:00
lifetime_capture_rules_2024,
lifetimes,
likely,
line,
link,
link_arg_attribute,
link_args,
link_cfg,
link_llvm_intrinsics,
link_name,
2019-08-27 22:42:44 +08:00
link_ordinal,
link_section,
linkage,
linker,
lint_reasons,
literal,
load,
loaded_from_disk,
local,
local_inner_macros,
2024-03-01 03:53:26 -05:00
log10f128,
log10f16,
log10f32,
log10f64,
2024-03-01 03:53:26 -05:00
log2f128,
log2f16,
log2f32,
log2f64,
log_syntax,
2024-03-01 03:53:26 -05:00
logf128,
logf16,
logf32,
logf64,
loongarch_target_feature,
loop_break_value,
lt,
macro_at_most_once_rep,
macro_attributes_in_derive_output,
macro_escape,
macro_export,
macro_lifetime_matcher,
macro_literal_matcher,
macro_metavar_expr,
macro_reexport,
macro_use,
macro_vis_matcher,
macros_in_extern,
main,
managed_boxes,
manually_drop,
2020-07-14 16:29:44 +10:00
map,
map_err,
marker,
marker_trait_attr,
masked,
match_beginning_vert,
match_default_bindings,
matches_macro,
2024-03-01 03:53:26 -05:00
maxnumf128,
maxnumf16,
maxnumf32,
maxnumf64,
may_dangle,
may_unwind,
maybe_uninit,
maybe_uninit_uninit,
maybe_uninit_zeroed,
mem_discriminant,
mem_drop,
mem_forget,
mem_replace,
mem_size_of,
mem_size_of_val,
2023-09-26 23:56:38 -04:00
mem_swap,
mem_uninitialized,
2021-07-31 12:14:30 -04:00
mem_variant_count,
mem_zeroed,
member_constraints,
memory,
memtag,
message,
meta,
metadata_type,
min_align_of,
min_align_of_val,
min_const_fn,
2020-08-05 18:27:54 +02:00
min_const_generics,
min_const_unsafe_fn,
2024-01-08 21:48:22 +01:00
min_exhaustive_patterns,
min_specialization,
min_type_alias_impl_trait,
2024-03-01 03:53:26 -05:00
minnumf128,
minnumf16,
minnumf32,
minnumf64,
mips_target_feature,
2024-01-20 19:01:57 +00:00
mir_assume,
mir_basic_block,
mir_call,
mir_cast_transmute,
mir_checked,
mir_copy_for_deref,
mir_debuginfo,
mir_deinit,
mir_discriminant,
mir_drop,
mir_field,
mir_goto,
mir_len,
mir_make_place,
mir_move,
mir_offset,
mir_retag,
mir_return,
mir_return_to,
mir_set_discriminant,
mir_static,
mir_static_mut,
mir_storage_dead,
mir_storage_live,
mir_unreachable,
mir_unwind_cleanup,
mir_unwind_continue,
mir_unwind_resume,
mir_unwind_terminate,
mir_unwind_terminate_reason,
mir_unwind_unreachable,
mir_variant,
miri,
mmx_reg,
modifiers,
module,
module_path,
more_qualified_paths,
more_struct_aliases,
movbe_target_feature,
move_ref_pattern,
move_size_limit,
mul,
mul_assign,
mul_with_overflow,
multiple_supertrait_upcastable,
2021-09-04 19:36:51 -07:00
must_not_suspend,
must_use,
2024-03-26 01:23:26 -04:00
mut_ref,
naked,
naked_functions,
name,
names,
native_link_modifiers,
native_link_modifiers_as_needed,
native_link_modifiers_bundle,
native_link_modifiers_verbatim,
native_link_modifiers_whole_archive,
natvis_file,
ne,
2024-03-01 03:53:26 -05:00
nearbyintf128,
nearbyintf16,
nearbyintf32,
nearbyintf64,
needs_allocator,
needs_drop,
needs_panic_runtime,
neg,
negate_unsigned,
2023-04-25 05:15:50 +00:00
negative_bounds,
negative_impls,
neon,
nested,
never,
2023-11-22 02:30:43 +01:00
never_patterns,
never_type,
never_type_fallback,
new,
new_binary,
new_const,
new_debug,
new_display,
new_lower_exp,
new_lower_hex,
new_octal,
new_pointer,
new_unchecked,
new_upper_exp,
new_upper_hex,
new_v1,
new_v1_formatted,
next,
niko,
nll,
no,
no_builtins,
no_core,
no_coverage,
no_crate_inject,
no_debug,
no_default_passes,
no_implicit_prelude,
no_inline,
no_link,
no_main,
no_mangle,
no_sanitize,
no_stack_check,
no_start,
no_std,
nomem,
non_ascii_idents,
non_exhaustive,
non_exhaustive_omitted_patterns_lint,
non_lifetime_binders,
non_modrs_mods,
none,
nontemporal_store,
2021-02-16 22:39:05 +01:00
noop_method_borrow,
noop_method_clone,
2021-02-16 22:39:05 +01:00
noop_method_deref,
noreturn,
nostack,
not,
notable_trait,
note,
object_safe_for_dispatch,
of,
2023-08-09 15:57:16 +01:00
off,
offset,
2022-09-11 00:37:49 -07:00
offset_of,
2023-11-03 13:16:47 +00:00
offset_of_enum,
2023-12-05 22:15:26 +00:00
offset_of_nested,
ok_or_else,
omit_gdb_pretty_printer_section,
on,
on_unimplemented,
opaque,
opt_out_copy,
optimize,
optimize_attribute,
optin_builtin_traits,
option,
option_env,
options,
or,
or_patterns,
2023-09-26 23:56:38 -04:00
ord_cmp_method,
other,
out,
overflow_checks,
overlapping_marker_traits,
owned_box,
packed,
packed_bundled_libs,
panic,
panic_2015,
panic_2021,
panic_abort,
panic_bounds_check,
panic_cannot_unwind,
panic_const_add_overflow,
panic_const_async_fn_resumed,
panic_const_async_fn_resumed_panic,
panic_const_async_gen_fn_resumed,
panic_const_async_gen_fn_resumed_panic,
panic_const_coroutine_resumed,
panic_const_coroutine_resumed_panic,
panic_const_div_by_zero,
panic_const_div_overflow,
panic_const_gen_fn_none,
panic_const_gen_fn_none_panic,
panic_const_mul_overflow,
panic_const_neg_overflow,
panic_const_rem_by_zero,
panic_const_rem_overflow,
panic_const_shl_overflow,
panic_const_shr_overflow,
panic_const_sub_overflow,
panic_fmt,
panic_handler,
panic_impl,
panic_implementation,
panic_in_cleanup,
panic_info,
panic_location,
panic_misaligned_pointer_dereference,
panic_nounwind,
panic_runtime,
panic_str,
panic_unwind,
panicking,
param_attrs,
parent_label,
partial_cmp,
partial_ord,
passes,
pat,
2021-04-14 20:34:51 -05:00
pat_param,
path,
pattern_complexity,
pattern_parentheses,
phantom_data,
pic,
pie,
pin,
platform_intrinsics,
plugin,
plugin_registrar,
plugins,
pointee_trait,
pointer,
2023-02-07 18:02:20 +00:00
pointer_like,
2020-04-05 17:55:19 -07:00
poll,
2023-11-28 18:18:19 +00:00
poll_next,
post_dash_lto: "post-lto",
postfix_match,
powerpc_target_feature,
2024-03-01 03:53:26 -05:00
powf128,
powf16,
powf32,
powf64,
2024-03-01 03:53:26 -05:00
powif128,
powif16,
powif32,
powif64,
pre_dash_lto: "pre-lto",
precise_pointer_size_matching,
pref_align_of,
prefetch_read_data,
prefetch_read_instruction,
prefetch_write_data,
prefetch_write_instruction,
preg,
prelude,
prelude_import,
preserves_flags,
2024-02-12 09:06:45 -03:00
prfchw_target_feature,
2022-01-06 14:50:46 +00:00
print_macro,
println_macro,
proc_dash_macro: "proc-macro",
proc_macro,
proc_macro_attribute,
proc_macro_derive,
proc_macro_expr,
proc_macro_gen,
proc_macro_hygiene,
proc_macro_internals,
proc_macro_mod,
proc_macro_non_items,
proc_macro_path_invoc,
2023-09-26 23:56:38 -04:00
process_exit,
profiler_builtins,
profiler_runtime,
ptr,
ptr_cast,
ptr_cast_const,
ptr_cast_mut,
ptr_const_is_null,
2023-09-26 23:56:38 -04:00
ptr_copy,
ptr_copy_nonoverlapping,
ptr_eq,
ptr_from_ref,
ptr_guaranteed_cmp,
ptr_is_null,
ptr_mask,
2021-04-06 22:01:00 +02:00
ptr_null,
ptr_null_mut,
ptr_offset_from,
ptr_offset_from_unsigned,
2023-09-26 23:56:38 -04:00
ptr_read,
ptr_read_unaligned,
ptr_read_volatile,
ptr_replace,
ptr_slice_from_raw_parts,
ptr_slice_from_raw_parts_mut,
ptr_swap,
ptr_swap_nonoverlapping,
2023-06-15 17:43:23 +02:00
ptr_unique,
ptr_write,
2023-09-26 23:56:38 -04:00
ptr_write_bytes,
ptr_write_unaligned,
ptr_write_volatile,
2020-10-21 00:46:29 +03:00
pub_macro_rules,
pub_restricted,
public,
pure,
pushpop_unsafe,
qreg,
qreg_low4,
qreg_low8,
quad_precision_float,
question_mark,
quote,
range_inclusive_new,
2019-08-27 22:42:44 +08:00
raw_dylib,
raw_eq,
raw_identifiers,
raw_ref_op,
re_rebalance_coherence,
read_enum,
read_enum_variant,
read_enum_variant_arg,
read_struct,
read_struct_field,
read_via_copy,
readonly,
realloc,
reason,
receiver,
recursion_limit,
reexport_test_harness_main,
2024-04-02 10:54:29 -05:00
ref_pat_everywhere,
ref_unwind_safe_trait,
reference,
reflect,
reg,
reg16,
reg32,
reg64,
reg_abcd,
2023-04-05 18:42:36 -07:00
reg_addr,
reg_byte,
2023-04-05 18:42:36 -07:00
reg_data,
2021-11-24 23:04:27 -05:00
reg_iw,
reg_nonzero,
2021-11-24 23:04:27 -05:00
reg_pair,
reg_ptr,
reg_upper,
register_attr,
register_tool,
relaxed_adts,
relaxed_struct_unsize,
relocation_model,
rem,
rem_assign,
repr,
repr128,
repr_align,
repr_align_enum,
repr_packed,
repr_simd,
repr_transparent,
require,
residual,
result,
resume,
return_position_impl_trait_in_trait,
2023-03-04 02:23:36 +00:00
return_type_notation,
rhs,
2024-03-01 03:53:26 -05:00
rintf128,
rintf16,
rintf32,
rintf64,
2020-03-01 09:52:04 +00:00
riscv_target_feature,
rlib,
ropi,
ropi_rwpi: "ropi-rwpi",
rotate_left,
rotate_right,
2024-03-01 03:53:26 -05:00
roundevenf128,
roundevenf16,
roundevenf32,
roundevenf64,
2024-03-01 03:53:26 -05:00
roundf128,
roundf16,
roundf32,
roundf64,
rt,
rtm_target_feature,
rust,
rust_2015,
rust_2018,
rust_2018_preview,
rust_2021,
2022-02-28 18:13:24 -05:00
rust_2024,
rust_begin_unwind,
rust_cold_cc,
rust_eh_catch_typeinfo,
rust_eh_personality,
rust_logo,
2019-05-14 15:58:22 +02:00
rustc,
2023-08-25 23:01:06 +02:00
rustc_abi,
rustc_allocator,
rustc_allocator_zeroed,
rustc_allow_const_fn_unstable,
rustc_allow_incoherent_impl,
rustc_allowed_through_unstable_modules,
rustc_attrs,
rustc_box,
rustc_builtin_macro,
rustc_capture_analysis,
rustc_clean,
rustc_coherence_is_core,
2023-02-14 10:17:19 +01:00
rustc_coinductive,
rustc_confusables,
rustc_const_panic_str,
rustc_const_stable,
rustc_const_unstable,
rustc_conversion_suggestion,
rustc_deallocator,
rustc_def_path,
rustc_default_body_unstable,
2022-11-12 23:37:52 +00:00
rustc_deny_explicit_impl,
rustc_diagnostic_item,
rustc_diagnostic_macros,
rustc_dirty,
2021-10-12 05:06:37 +00:00
rustc_do_not_const_check,
rustc_doc_primitive,
rustc_dummy,
rustc_dump_env_program_clauses,
rustc_dump_program_clauses,
rustc_dump_user_args,
2021-07-17 15:44:19 +08:00
rustc_dump_vtable,
2022-09-12 10:57:34 +03:00
rustc_effective_visibility,
rustc_error,
rustc_evaluate_where_clauses,
rustc_expected_cgu_reuse,
rustc_has_incoherent_inherent_impls,
rustc_hidden_type_of_opaques,
rustc_if_this_changed,
rustc_inherit_overflow_checks,
rustc_insignificant_dtor,
rustc_intrinsic,
rustc_intrinsic_must_be_overridden,
rustc_layout,
rustc_layout_scalar_valid_range_end,
rustc_layout_scalar_valid_range_start,
2021-02-23 15:12:28 +00:00
rustc_legacy_const_generics,
rustc_lint_diagnostics,
rustc_lint_opt_deny_field_access,
rustc_lint_opt_ty,
2022-01-05 13:02:16 +01:00
rustc_lint_query_instability,
rustc_macro_transparency,
2021-04-08 21:37:38 +08:00
rustc_main,
rustc_mir,
rustc_must_implement_one_of,
rustc_never_returns_null_ptr,
rustc_never_type_options,
rustc_no_mir_inline,
rustc_nonnull_optimization_guaranteed,
rustc_nounwind,
rustc_object_lifetime_default,
rustc_on_unimplemented,
rustc_outlives,
rustc_paren_sugar,
rustc_partition_codegened,
rustc_partition_reused,
rustc_pass_by_value,
rustc_peek,
rustc_peek_definite_init,
2020-04-10 11:00:11 -07:00
rustc_peek_liveness,
rustc_peek_maybe_init,
rustc_peek_maybe_uninit,
rustc_polymorphize_error,
rustc_preserve_ub_checks,
rustc_private,
rustc_proc_macro_decls,
rustc_promotable,
rustc_reallocator,
rustc_regions,
rustc_reservation_impl,
rustc_safe_intrinsic,
rustc_serialize,
rustc_skip_array_during_method_dispatch,
rustc_specialization_trait,
rustc_std_internal_symbol,
rustc_strict_coherence,
rustc_symbol_name,
rustc_test_marker,
rustc_then_this_would_need,
rustc_trivial_field_reads,
rustc_unsafe_specialization_marker,
rustc_variance,
2023-08-08 19:59:44 +00:00
rustc_variance_of_opaques,
rustdoc,
2021-10-30 17:44:50 +02:00
rustdoc_internals,
rustdoc_missing_doc_code_examples,
rustfmt,
rvalue_static_promotion,
rwpi,
2021-06-17 09:45:19 +09:00
s,
safety,
sanitize,
sanitizer_cfi_generalize_pointers,
sanitizer_cfi_normalize_integers,
sanitizer_runtime,
saturating_add,
2023-08-24 13:30:53 -03:00
saturating_div,
saturating_sub,
self_in_typedefs,
self_struct_ctor,
semitransparent,
shadow_call_stack,
shl,
shl_assign,
should_panic,
shr,
shr_assign,
sig_dfl,
sig_ign,
simd,
simd_add,
simd_and,
2022-04-12 11:00:55 -04:00
simd_arith_offset,
2021-12-30 01:18:44 +00:00
simd_as,
simd_bitmask,
simd_bitreverse,
2023-07-27 23:04:14 -04:00
simd_bswap,
simd_cast,
simd_cast_ptr,
simd_ceil,
simd_ctlz,
simd_cttz,
simd_div,
simd_eq,
simd_expose_provenance,
simd_extract,
simd_fabs,
simd_fcos,
simd_fexp,
simd_fexp2,
simd_ffi,
simd_flog,
simd_flog10,
simd_flog2,
simd_floor,
simd_fma,
simd_fmax,
simd_fmin,
simd_fpow,
simd_fpowi,
simd_fsin,
simd_fsqrt,
simd_gather,
simd_ge,
simd_gt,
simd_insert,
simd_le,
simd_lt,
simd_masked_load,
simd_masked_store,
simd_mul,
simd_ne,
2021-03-19 02:16:21 +08:00
simd_neg,
simd_or,
simd_reduce_add_ordered,
simd_reduce_add_unordered,
simd_reduce_all,
simd_reduce_and,
simd_reduce_any,
simd_reduce_max,
simd_reduce_min,
simd_reduce_mul_ordered,
simd_reduce_mul_unordered,
simd_reduce_or,
simd_reduce_xor,
simd_rem,
simd_round,
simd_saturating_add,
simd_saturating_sub,
simd_scatter,
simd_select,
simd_select_bitmask,
simd_shl,
simd_shr,
simd_shuffle,
simd_shuffle_generic,
simd_sub,
simd_trunc,
2024-03-23 23:00:53 +01:00
simd_with_exposed_provenance,
simd_xor,
since,
2024-03-01 03:53:26 -05:00
sinf128,
sinf16,
sinf32,
sinf64,
size,
size_of,
size_of_val,
sized,
skip,
slice,
2023-09-26 23:56:38 -04:00
slice_from_raw_parts,
slice_from_raw_parts_mut,
2021-06-20 16:09:42 +02:00
slice_len_fn,
slice_patterns,
slicing_syntax,
soft,
specialization,
speed,
spotlight,
2024-03-01 03:53:26 -05:00
sqrtf128,
sqrtf16,
sqrtf32,
sqrtf64,
sreg,
sreg_low16,
sse,
sse4a_target_feature,
stable,
staged_api,
start,
state,
static_in_const,
static_nobundle,
static_recursion,
staticlib,
std,
std_panic,
std_panic_2015_macro,
std_panic_macro,
stmt,
stmt_expr_attributes,
stop_after_dataflow,
store,
str,
str_from_utf8,
str_from_utf8_mut,
str_from_utf8_unchecked,
str_from_utf8_unchecked_mut,
2022-03-24 17:05:47 +01:00
str_split_whitespace,
str_trim,
str_trim_end,
str_trim_start,
strict_provenance,
2022-11-11 14:31:07 +00:00
string_deref_patterns,
stringify,
struct_field_attributes,
struct_inherit,
struct_variant,
structural_match,
structural_peq,
sub,
sub_assign,
sub_with_overflow,
suggestion,
sym,
sync,
2024-02-27 16:02:19 +00:00
synthetic,
t32,
target,
target_abi,
target_arch,
target_endian,
target_env,
target_family,
target_feature,
target_feature_11,
target_has_atomic,
target_has_atomic_equal_alignment,
target_has_atomic_load_store,
target_os,
target_pointer_width,
target_thread_local,
target_vendor,
tbm_target_feature,
termination,
termination_trait,
termination_trait_test,
test,
test_2018_feature,
test_accepted_feature,
test_case,
test_removed_feature,
test_runner,
test_unstable_lint,
thread,
thread_local,
2022-01-06 14:50:46 +00:00
thread_local_macro,
2023-03-05 20:19:41 -08:00
three_way_compare,
thumb2,
thumb_mode: "thumb-mode",
2022-05-16 20:15:06 -04:00
tmm_reg,
2023-09-26 23:56:38 -04:00
to_owned_method,
2022-07-15 14:29:15 +09:00
to_string,
2023-09-26 23:56:38 -04:00
to_string_method,
2022-07-15 14:29:15 +09:00
to_vec,
2022-01-06 14:50:46 +00:00
todo_macro,
tool_attributes,
tool_lints,
trace_macros,
track_caller,
trait_alias,
2019-10-31 10:52:05 +08:00
trait_upcasting,
transmute,
2023-04-04 08:41:44 +00:00
transmute_generic_consts,
transmute_opts,
transmute_trait,
transmute_unchecked,
transparent,
transparent_enums,
transparent_unions,
trivial_bounds,
2024-03-01 03:53:26 -05:00
truncf128,
truncf16,
truncf32,
truncf64,
try_blocks,
try_capture,
2021-05-25 23:27:26 -04:00
try_from,
2023-09-26 23:56:38 -04:00
try_from_fn,
2021-05-25 23:27:26 -04:00
try_into,
try_trait_v2,
tt,
tuple,
tuple_indexing,
2022-07-30 01:53:29 +00:00
tuple_trait,
two_phase,
ty,
type_alias_enum_variants,
type_alias_impl_trait,
type_ascribe,
type_ascription,
type_changing_struct_update,
type_id,
type_length_limit,
type_macros,
type_name,
type_privacy_lints,
typed_swap,
u128,
u128_legacy_const_max,
u128_legacy_const_min,
u128_legacy_fn_max_value,
u128_legacy_fn_min_value,
u128_legacy_mod,
u16,
u16_legacy_const_max,
u16_legacy_const_min,
u16_legacy_fn_max_value,
u16_legacy_fn_min_value,
u16_legacy_mod,
u32,
u32_legacy_const_max,
u32_legacy_const_min,
u32_legacy_fn_max_value,
u32_legacy_fn_min_value,
u32_legacy_mod,
u64,
u64_legacy_const_max,
u64_legacy_const_min,
u64_legacy_fn_max_value,
u64_legacy_fn_min_value,
u64_legacy_mod,
u8,
u8_legacy_const_max,
u8_legacy_const_min,
u8_legacy_fn_max_value,
u8_legacy_fn_min_value,
u8_legacy_mod,
ub_checks,
unaligned_volatile_load,
unaligned_volatile_store,
unboxed_closures,
unchecked_add,
unchecked_div,
unchecked_mul,
unchecked_rem,
unchecked_shl,
unchecked_shr,
unchecked_sub,
underscore_const_names,
underscore_imports,
underscore_lifetimes,
uniform_paths,
2022-01-06 14:50:46 +00:00
unimplemented_macro,
unit,
universal_impl_trait,
unix,
unix_sigpipe,
unlikely,
unmarked_api,
unnamed_fields,
unpin,
unreachable,
unreachable_2015,
unreachable_2015_macro,
unreachable_2021,
unreachable_code,
unreachable_display,
2022-01-06 14:50:46 +00:00
unreachable_macro,
unrestricted_attribute_tokens,
2020-05-03 23:11:34 +02:00
unsafe_block_in_unsafe_fn,
unsafe_cell,
unsafe_cell_raw_get,
unsafe_no_drop_flag,
unsafe_pin_internals,
unsize,
2020-10-16 17:46:59 -03:00
unsized_fn_params,
unsized_locals,
unsized_tuple_coercion,
unstable,
unstable_location_reason_default: "this crate is being loaded from the sysroot, an \
unstable location; did you mean to load this crate \
from crates.io via `Cargo.toml` instead?",
untagged_unions,
unused_imports,
unwind,
unwind_attributes,
unwind_safe_trait,
2020-08-18 17:02:23 -04:00
unwrap,
unwrap_or,
use_extern_macros,
use_nested_groups,
used,
used_with_arg,
using,
usize,
usize_legacy_const_max,
usize_legacy_const_min,
usize_legacy_fn_max_value,
usize_legacy_fn_min_value,
usize_legacy_mod,
va_arg,
va_copy,
va_end,
va_list,
va_start,
val,
validity,
values,
var,
variant_count,
vec,
2022-01-06 14:50:46 +00:00
vec_macro,
version,
vfp2,
vis,
visible_private_types,
volatile,
volatile_copy_memory,
volatile_copy_nonoverlapping_memory,
volatile_load,
volatile_set_memory,
volatile_store,
vreg,
vreg_low16,
vtable_align,
vtable_size,
warn,
wasip2,
rustc: Add a new `wasm` ABI This commit implements the idea of a new ABI for the WebAssembly target, one called `"wasm"`. This ABI is entirely of my own invention and has no current precedent, but I think that the addition of this ABI might help solve a number of issues with the WebAssembly targets. When `wasm32-unknown-unknown` was first added to Rust I naively "implemented an abi" for the target. I then went to write `wasm-bindgen` which accidentally relied on details of this ABI. Turns out the ABI definition didn't match C, which is causing issues for C/Rust interop. Currently the compiler has a "wasm32 bindgen compat" ABI which is the original implementation I added, and it's purely there for, well, `wasm-bindgen`. Another issue with the WebAssembly target is that it's not clear to me when and if the default C ABI will change to account for WebAssembly's multi-value feature (a feature that allows functions to return multiple values). Even if this does happen, though, it seems like the C ABI will be guided based on the performance of WebAssembly code and will likely not match even what the current wasm-bindgen-compat ABI is today. This leaves a hole in Rust's expressivity in binding WebAssembly where given a particular import type, Rust may not be able to import that signature with an updated C ABI for multi-value. To fix these issues I had the idea of a new ABI for WebAssembly, one called `wasm`. The definition of this ABI is "what you write maps straight to wasm". The goal here is that whatever you write down in the parameter list or in the return values goes straight into the function's signature in the WebAssembly file. This special ABI is for intentionally matching the ABI of an imported function from the environment or exporting a function with the right signature. With the addition of a new ABI, this enables rustc to: * Eventually remove the "wasm-bindgen compat hack". Once this ABI is stable wasm-bindgen can switch to using it everywhere. Afterwards the wasm32-unknown-unknown target can have its default ABI updated to match C. * Expose the ability to precisely match an ABI signature for a WebAssembly function, regardless of what the C ABI that clang chooses turns out to be. * Continue to evolve the definition of the default C ABI to match what clang does on all targets, since the purpose of that ABI will be explicitly matching C rather than generating particular function imports/exports. Naturally this is implemented as an unstable feature initially, but it would be nice for this to get stabilized (if it works) in the near-ish future to remove the wasm32-unknown-unknown incompatibility with the C ABI. Doing this, however, requires the feature to be on stable because wasm-bindgen works with stable Rust.
2021-04-01 16:08:29 -07:00
wasm_abi,
wasm_import_module,
wasm_target_feature,
while_let,
windows,
windows_subsystem,
with_negative_coherence,
wrapping_add,
2023-08-24 13:30:53 -03:00
wrapping_div,
wrapping_mul,
2023-08-24 13:30:53 -03:00
wrapping_rem,
wrapping_rem_euclid,
wrapping_sub,
2021-04-20 19:03:10 +10:00
wreg,
write_bytes,
2022-01-06 14:50:46 +00:00
write_macro,
write_str,
write_via_move,
2022-01-06 14:50:46 +00:00
writeln_macro,
x87_reg,
xer,
xmm_reg,
yeet_desugar_details,
yeet_expr,
yes,
yield_expr,
ymm_reg,
zmm_reg,
}
}
#[derive(Copy, Clone, Eq, HashStable_Generic, Encodable, Decodable)]
2017-03-17 04:04:41 +00:00
pub struct Ident {
pub name: Symbol,
pub span: Span,
2017-03-17 04:04:41 +00:00
}
impl Ident {
#[inline]
2019-05-17 10:44:51 +10:00
/// Constructs a new identifier from a symbol and a span.
pub const fn new(name: Symbol, span: Span) -> Ident {
Ident { name, span }
}
/// Constructs a new identifier with a dummy span.
#[inline]
pub const fn with_dummy_span(name: Symbol) -> Ident {
Ident::new(name, DUMMY_SP)
2017-03-17 04:04:41 +00:00
}
#[inline]
pub fn empty() -> Ident {
Ident::with_dummy_span(kw::Empty)
}
2019-09-14 21:16:51 +01:00
/// Maps a string to an identifier with a dummy span.
2017-03-17 04:04:41 +00:00
pub fn from_str(string: &str) -> Ident {
Ident::with_dummy_span(Symbol::intern(string))
2017-03-17 04:04:41 +00:00
}
2019-05-17 10:44:51 +10:00
/// Maps a string and a span to an identifier.
pub fn from_str_and_span(string: &str, span: Span) -> Ident {
Ident::new(Symbol::intern(string), span)
}
2019-02-08 14:53:55 +01:00
/// Replaces `lo` and `hi` with those from `span`, but keep hygiene context.
2018-03-18 16:47:09 +03:00
pub fn with_span_pos(self, span: Span) -> Ident {
Ident::new(self.name, span.with_ctxt(self.span.ctxt()))
}
pub fn without_first_quote(self) -> Ident {
Ident::new(Symbol::intern(self.as_str().trim_start_matches('\'')), self.span)
2017-12-06 04:28:01 -05:00
}
2018-06-28 00:12:17 +03:00
/// "Normalize" ident for use in comparisons using "item hygiene".
/// Identifiers with same string value become same if they came from the same macro 2.0 macro
/// (e.g., `macro` item, but not `macro_rules` item) and stay different if they came from
/// different macro 2.0 macros.
2018-06-28 00:12:17 +03:00
/// Technically, this operation strips all non-opaque marks from ident's syntactic context.
pub fn normalize_to_macros_2_0(self) -> Ident {
Ident::new(self.name, self.span.normalize_to_macros_2_0())
}
2018-06-30 01:53:17 +03:00
/// "Normalize" ident for use in comparisons using "local variable hygiene".
/// Identifiers with same string value become same if they came from the same non-transparent
/// macro (e.g., `macro` or `macro_rules!` items) and stay different if they came from different
2018-06-30 01:53:17 +03:00
/// non-transparent macros.
/// Technically, this operation strips all transparent marks from ident's syntactic context.
2022-09-28 14:45:31 +10:00
#[inline]
pub fn normalize_to_macro_rules(self) -> Ident {
Ident::new(self.name, self.span.normalize_to_macro_rules())
2018-06-24 19:54:23 +03:00
}
/// Access the underlying string. This is a slowish operation because it
/// requires locking the symbol interner.
///
/// Note that the lifetime of the return value is a lie. See
/// `Symbol::as_str()` for details.
pub fn as_str(&self) -> &str {
2018-05-26 15:12:38 +03:00
self.name.as_str()
}
}
impl PartialEq for Ident {
2022-09-28 14:45:31 +10:00
#[inline]
fn eq(&self, rhs: &Self) -> bool {
self.name == rhs.name && self.span.eq_ctxt(rhs.span)
}
}
impl Hash for Ident {
fn hash<H: Hasher>(&self, state: &mut H) {
self.name.hash(state);
self.span.ctxt().hash(state);
2017-03-17 04:04:41 +00:00
}
}
impl fmt::Debug for Ident {
2019-02-04 03:42:27 +09:00
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2020-02-23 01:29:36 +03:00
fmt::Display::fmt(self, f)?;
fmt::Debug::fmt(&self.span.ctxt(), f)
2017-03-17 04:04:41 +00:00
}
}
2020-02-23 01:29:36 +03:00
/// This implementation is supposed to be used in error messages, so it's expected to be identical
/// to printing the original identifier token written in source code (`token_to_string`),
/// except that AST identifiers don't keep the rawness flag, so we have to guess it.
2017-03-17 04:04:41 +00:00
impl fmt::Display for Ident {
2019-02-04 03:42:27 +09:00
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2020-02-23 01:29:36 +03:00
fmt::Display::fmt(&IdentPrinter::new(self.name, self.is_raw_guess(), None), f)
2017-03-17 04:04:41 +00:00
}
}
2022-12-10 04:48:37 +02:00
/// The most general type to print identifiers.
///
2020-02-23 01:29:36 +03:00
/// AST pretty-printer is used as a fallback for turning AST structures into token streams for
/// proc macros. Additionally, proc macros may stringify their input and expect it survive the
/// stringification (especially true for proc macro derives written between Rust 1.15 and 1.30).
/// So we need to somehow pretty-print `$crate` in a way preserving at least some of its
/// hygiene data, most importantly name of the crate it refers to.
/// As a result we print `$crate` as `crate` if it refers to the local crate
/// and as `::other_crate_name` if it refers to some other crate.
/// Note, that this is only done if the ident token is printed from inside of AST pretty-printing,
2020-02-23 01:29:36 +03:00
/// but not otherwise. Pretty-printing is the only way for proc macros to discover token contents,
/// so we should not perform this lossy conversion if the top level call to the pretty-printer was
/// done for a token stream or a single token.
pub struct IdentPrinter {
symbol: Symbol,
is_raw: bool,
/// Span used for retrieving the crate name to which `$crate` refers to,
/// if this field is `None` then the `$crate` conversion doesn't happen.
convert_dollar_crate: Option<Span>,
}
impl IdentPrinter {
/// The most general `IdentPrinter` constructor. Do not use this.
pub fn new(symbol: Symbol, is_raw: bool, convert_dollar_crate: Option<Span>) -> IdentPrinter {
IdentPrinter { symbol, is_raw, convert_dollar_crate }
}
/// This implementation is supposed to be used when printing identifiers
/// as a part of pretty-printing for larger AST pieces.
/// Do not use this either.
pub fn for_ast_ident(ident: Ident, is_raw: bool) -> IdentPrinter {
IdentPrinter::new(ident.name, is_raw, Some(ident.span))
}
}
impl fmt::Display for IdentPrinter {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
if self.is_raw {
f.write_str("r#")?;
2020-12-03 17:06:58 -05:00
} else if self.symbol == kw::DollarCrate {
if let Some(span) = self.convert_dollar_crate {
let converted = span.ctxt().dollar_crate_name();
if !converted.is_path_segment_keyword() {
f.write_str("::")?;
2020-02-23 01:29:36 +03:00
}
2020-12-03 17:06:58 -05:00
return fmt::Display::fmt(&converted, f);
2020-02-23 01:29:36 +03:00
}
}
fmt::Display::fmt(&self.symbol, f)
}
}
/// An newtype around `Ident` that calls [Ident::normalize_to_macro_rules] on
/// construction.
// FIXME(matthewj, petrochenkov) Use this more often, add a similar
// `ModernIdent` struct and use that as well.
#[derive(Copy, Clone, Eq, PartialEq, Hash)]
pub struct MacroRulesNormalizedIdent(Ident);
impl MacroRulesNormalizedIdent {
pub fn new(ident: Ident) -> Self {
Self(ident.normalize_to_macro_rules())
}
}
impl fmt::Debug for MacroRulesNormalizedIdent {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Debug::fmt(&self.0, f)
}
}
impl fmt::Display for MacroRulesNormalizedIdent {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Display::fmt(&self.0, f)
}
}
2019-08-31 16:40:20 +01:00
/// An interned string.
2019-05-17 10:44:51 +10:00
///
2019-08-31 16:40:20 +01:00
/// Internally, a `Symbol` is implemented as an index, and all operations
/// (including hashing, equality, and ordering) operate on that index. The use
/// of `rustc_index::newtype_index!` means that `Option<Symbol>` only takes up 4 bytes,
/// because `rustc_index::newtype_index!` reserves the last 256 values for tagging purposes.
///
/// Note that `Symbol` cannot directly be a `rustc_index::newtype_index!` because it
2019-05-17 10:44:51 +10:00
/// implements `fmt::Debug`, `Encodable`, and `Decodable` in special ways.
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct Symbol(SymbolIndex);
rustc_index::newtype_index! {
#[orderable]
struct SymbolIndex {}
}
impl Symbol {
const fn new(n: u32) -> Self {
2020-03-10 13:44:53 -07:00
Symbol(SymbolIndex::from_u32(n))
}
/// for use in Decoder only
pub fn new_from_decoded(n: u32) -> Self {
Self::new(n)
}
/// Maps a string to its interned representation.
pub fn intern(string: &str) -> Self {
2021-09-15 18:44:17 +02:00
with_session_globals(|session_globals| session_globals.symbol_interner.intern(string))
}
/// Access the underlying string. This is a slowish operation because it
/// requires locking the symbol interner.
///
/// Note that the lifetime of the return value is a lie. It's not the same
/// as `&self`, but actually tied to the lifetime of the underlying
/// interner. Interners are long-lived, and there are very few of them, and
/// this function is typically used for short-lived things, so in practice
/// it works out ok.
pub fn as_str(&self) -> &str {
with_session_globals(|session_globals| unsafe {
std::mem::transmute::<&str, &str>(session_globals.symbol_interner.get(*self))
2016-11-19 05:55:28 +00:00
})
}
pub fn as_u32(self) -> u32 {
self.0.as_u32()
}
pub fn is_empty(self) -> bool {
self == kw::Empty
}
/// This method is supposed to be used in error messages, so it's expected to be
/// identical to printing the original identifier token written in source code
/// (`token_to_string`, `Ident::to_string`), except that symbols don't keep the rawness flag
/// or edition, so we have to guess the rawness using the global edition.
pub fn to_ident_string(self) -> String {
Ident::with_dummy_span(self).to_string()
}
}
impl fmt::Debug for Symbol {
2019-02-04 03:42:27 +09:00
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Debug::fmt(self.as_str(), f)
}
}
impl fmt::Display for Symbol {
2019-02-04 03:42:27 +09:00
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
fmt::Display::fmt(self.as_str(), f)
}
}
2022-10-27 16:14:48 -05:00
// takes advantage of `str::to_string` specialization
impl ToString for Symbol {
fn to_string(&self) -> String {
self.as_str().to_string()
}
}
impl<CTX> HashStable<CTX> for Symbol {
#[inline]
fn hash_stable(&self, hcx: &mut CTX, hasher: &mut StableHasher) {
self.as_str().hash_stable(hcx, hasher);
}
}
impl<CTX> ToStableHashKey<CTX> for Symbol {
type KeyType = String;
#[inline]
fn to_stable_hash_key(&self, _: &CTX) -> String {
self.as_str().to_string()
}
}
impl StableCompare for Symbol {
const CAN_USE_UNSTABLE_SORT: bool = true;
fn stable_cmp(&self, other: &Self) -> std::cmp::Ordering {
self.as_str().cmp(other.as_str())
}
}
pub(crate) struct Interner(Lock<InternerInner>);
// The `&'static str`s in this type actually point into the arena.
//
// This type is private to prevent accidentally constructing more than one
// `Interner` on the same thread, which makes it easy to mix up `Symbol`s
// between `Interner`s.
struct InternerInner {
2018-05-10 16:27:46 +02:00
arena: DroplessArena,
strings: FxIndexSet<&'static str>,
}
impl Interner {
fn prefill(init: &[&'static str]) -> Self {
Interner(Lock::new(InternerInner {
arena: Default::default(),
strings: init.iter().copied().collect(),
}))
}
#[inline]
2021-09-15 18:44:31 +02:00
fn intern(&self, string: &str) -> Symbol {
let mut inner = self.0.lock();
if let Some(idx) = inner.strings.get_index_of(string) {
return Symbol::new(idx as u32);
}
let string: &str = inner.arena.alloc_str(string);
// SAFETY: we can extend the arena allocation to `'static` because we
// only access these while the arena is still alive.
2018-05-10 16:27:46 +02:00
let string: &'static str = unsafe { &*(string as *const str) };
// This second hash table lookup can be avoided by using `RawEntryMut`,
// but this code path isn't hot enough for it to be worth it. See
// #91445 for details.
let (idx, is_new) = inner.strings.insert_full(string);
debug_assert!(is_new); // due to the get_index_of check above
Symbol::new(idx as u32)
}
/// Get the symbol as a string.
///
/// [`Symbol::as_str()`] should be used in preference to this function.
2021-09-15 18:44:31 +02:00
fn get(&self, symbol: Symbol) -> &str {
self.0.lock().strings.get_index(symbol.0.as_usize()).unwrap()
}
}
2019-05-11 17:41:37 +03:00
// This module has a very short name because it's used a lot.
/// This module contains all the defined keyword `Symbol`s.
///
/// Given that `kw` is imported, use them like `kw::keyword_name`.
/// For example `kw::Loop` or `kw::Break`.
2019-05-11 17:41:37 +03:00
pub mod kw {
pub use super::kw_generated::*;
2019-04-09 09:36:17 +02:00
}
// This module has a very short name because it's used a lot.
/// This module contains all the defined non-keyword `Symbol`s.
///
/// Given that `sym` is imported, use them like `sym::symbol_name`.
/// For example `sym::rustfmt` or `sym::u8`.
pub mod sym {
2019-04-09 09:36:17 +02:00
use super::Symbol;
2020-12-30 19:37:39 +01:00
#[doc(inline)]
pub use super::sym_generated::*;
// Used from a macro in `librustc_feature/accepted.rs`
pub use super::kw::MacroRules as macro_rules;
/// Get the symbol for an integer.
///
/// The first few non-negative integers each have a static symbol and therefore
/// are fast.
pub fn integer<N: TryInto<usize> + Copy + itoa::Integer>(n: N) -> Symbol {
if let Result::Ok(idx) = n.try_into() {
if idx < 10 {
return Symbol::new(super::SYMBOL_DIGITS_BASE + idx as u32);
}
}
let mut buffer = itoa::Buffer::new();
let printed = buffer.format(n);
Symbol::intern(printed)
}
2019-04-09 09:36:17 +02:00
}
impl Symbol {
fn is_special(self) -> bool {
self <= kw::Underscore
2018-12-02 03:05:19 +03:00
}
fn is_used_keyword_always(self) -> bool {
self >= kw::As && self <= kw::While
}
fn is_used_keyword_conditional(self, edition: impl FnOnce() -> Edition) -> bool {
(self >= kw::Async && self <= kw::Dyn) && edition() >= Edition::Edition2018
}
fn is_unused_keyword_always(self) -> bool {
self >= kw::Abstract && self <= kw::Yield
}
fn is_unused_keyword_conditional(self, edition: impl Copy + FnOnce() -> Edition) -> bool {
self == kw::Try && edition().at_least_rust_2018()
|| self == kw::Gen && edition().at_least_rust_2024()
}
pub fn is_reserved(self, edition: impl Copy + FnOnce() -> Edition) -> bool {
self.is_special()
|| self.is_used_keyword_always()
|| self.is_unused_keyword_always()
|| self.is_used_keyword_conditional(edition)
|| self.is_unused_keyword_conditional(edition)
}
/// A keyword or reserved identifier that can be used as a path segment.
pub fn is_path_segment_keyword(self) -> bool {
self == kw::Super
|| self == kw::SelfLower
|| self == kw::SelfUpper
|| self == kw::Crate
|| self == kw::PathRoot
|| self == kw::DollarCrate
}
/// Returns `true` if the symbol is `true` or `false`.
pub fn is_bool_lit(self) -> bool {
self == kw::True || self == kw::False
}
/// Returns `true` if this symbol can be a raw identifier.
pub fn can_be_raw(self) -> bool {
self != kw::Empty && self != kw::Underscore && !self.is_path_segment_keyword()
}
/// Is this symbol was interned in compiler's `symbols!` macro
pub fn is_preinterned(self) -> bool {
self.as_u32() < PREINTERNED_SYMBOLS_COUNT
}
}
impl Ident {
/// Returns `true` for reserved identifiers used internally for elided lifetimes,
/// unnamed method parameters, crate root module, error recovery etc.
pub fn is_special(self) -> bool {
self.name.is_special()
}
/// Returns `true` if the token is a keyword used in the language.
pub fn is_used_keyword(self) -> bool {
2018-12-02 03:05:19 +03:00
// Note: `span.edition()` is relatively expensive, don't call it unless necessary.
self.name.is_used_keyword_always()
|| self.name.is_used_keyword_conditional(|| self.span.edition())
}
/// Returns `true` if the token is a keyword reserved for possible future use.
pub fn is_unused_keyword(self) -> bool {
// Note: `span.edition()` is relatively expensive, don't call it unless necessary.
self.name.is_unused_keyword_always()
|| self.name.is_unused_keyword_conditional(|| self.span.edition())
}
/// Returns `true` if the token is either a special identifier or a keyword.
pub fn is_reserved(self) -> bool {
// Note: `span.edition()` is relatively expensive, don't call it unless necessary.
self.name.is_reserved(|| self.span.edition())
}
/// A keyword or reserved identifier that can be used as a path segment.
pub fn is_path_segment_keyword(self) -> bool {
self.name.is_path_segment_keyword()
2019-02-27 22:06:26 +03:00
}
/// We see this identifier in a normal identifier position, like variable name or a type.
/// How was it written originally? Did it use the raw form? Let's try to guess.
pub fn is_raw_guess(self) -> bool {
self.name.can_be_raw() && self.is_reserved()
}
}