Skip to content

Commit 5918318

Browse files
committed
Auto merge of #56225 - alexreg:type_alias_enum_variants, r=petrochenkov
Implement RFC 2338, "Type alias enum variants" This PR implements [RFC 2338](rust-lang/rfcs#2338), allowing one to write code like the following. ```rust #![feature(type_alias_enum_variants)] enum Foo { Bar(i32), Baz { i: i32 }, } type Alias = Foo; fn main() { let t = Alias::Bar(0); let t = Alias::Baz { i: 0 }; match t { Alias::Bar(_i) => {} Alias::Baz { i: _i } => {} } } ``` Since `Self` can be considered a type alias in this context, it also enables using `Self::Variant` as both a constructor and pattern. Fixes issues #56199 and #56611. N.B., after discussing the syntax for type arguments on enum variants with @petrochenkov and @eddyb (there are also a few comments on the [tracking issue](#49683)), the consensus seems to be treat the syntax as follows, which ought to be backwards-compatible. ```rust Option::<u8>::None; // OK Option::None::<u8>; // OK, but lint in near future (hard error next edition?) Alias::<u8>::None; // OK Alias::None::<u8>; // Error ``` I do not know if this will need an FCP, but let's start one if so.
2 parents 0071157 + a4fa7ef commit 5918318

File tree

111 files changed

+1605
-758
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

111 files changed

+1605
-758
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
# `type_alias_enum_variants`
2+
3+
The tracking issue for this feature is: [#49683]
4+
5+
[#49683]: https://github.com/rust-lang/rust/issues/49683
6+
7+
------------------------
8+
9+
The `type_alias_enum_variants` feature enables the use of variants on type
10+
aliases that refer to enums, as both a constructor and a pattern. That is,
11+
it allows for the syntax `EnumAlias::Variant`, which behaves exactly the same
12+
as `Enum::Variant` (assuming that `EnumAlias` is an alias for some enum type
13+
`Enum`).
14+
15+
Note that since `Self` exists as a type alias, this feature also enables the
16+
use of the syntax `Self::Variant` within an impl block for an enum type.
17+
18+
```rust
19+
#![feature(type_alias_enum_variants)]
20+
21+
enum Foo {
22+
Bar(i32),
23+
Baz { i: i32 },
24+
}
25+
26+
type Alias = Foo;
27+
28+
fn main() {
29+
let t = Alias::Bar(0);
30+
let t = Alias::Baz { i: 0 };
31+
match t {
32+
Alias::Bar(_i) => {}
33+
Alias::Baz { i: _i } => {}
34+
}
35+
}
36+
```

src/libproc_macro/bridge/rpc.rs

+15-9
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
//! Serialization for client<->server communication.
1+
//! Serialization for client-server communication.
22
33
use std::any::Any;
44
use std::char;
@@ -71,15 +71,18 @@ macro_rules! rpc_encode_decode {
7171
(enum $name:ident $(<$($T:ident),+>)* { $($variant:ident $(($field:ident))*),* $(,)* }) => {
7272
impl<S, $($($T: Encode<S>),+)*> Encode<S> for $name $(<$($T),+>)* {
7373
fn encode(self, w: &mut Writer, s: &mut S) {
74-
// HACK(eddyb) `Tag` enum duplicated between the
74+
// HACK(eddyb): `Tag` enum duplicated between the
7575
// two impls as there's no other place to stash it.
76-
#[repr(u8)] enum Tag { $($variant),* }
7776
#[allow(non_upper_case_globals)]
78-
impl Tag { $(const $variant: u8 = Tag::$variant as u8;)* }
77+
mod tag {
78+
#[repr(u8)] enum Tag { $($variant),* }
79+
80+
$(pub const $variant: u8 = Tag::$variant as u8;)*
81+
}
7982

8083
match self {
8184
$($name::$variant $(($field))* => {
82-
<Tag>::$variant.encode(w, s);
85+
tag::$variant.encode(w, s);
8386
$($field.encode(w, s);)*
8487
})*
8588
}
@@ -90,14 +93,17 @@ macro_rules! rpc_encode_decode {
9093
for $name $(<$($T),+>)*
9194
{
9295
fn decode(r: &mut Reader<'a>, s: &mut S) -> Self {
93-
// HACK(eddyb) `Tag` enum duplicated between the
96+
// HACK(eddyb): `Tag` enum duplicated between the
9497
// two impls as there's no other place to stash it.
95-
#[repr(u8)] enum Tag { $($variant),* }
9698
#[allow(non_upper_case_globals)]
97-
impl Tag { $(const $variant: u8 = Tag::$variant as u8;)* }
99+
mod tag {
100+
#[repr(u8)] enum Tag { $($variant),* }
101+
102+
$(pub const $variant: u8 = Tag::$variant as u8;)*
103+
}
98104

99105
match u8::decode(r, s) {
100-
$(<Tag>::$variant => {
106+
$(tag::$variant => {
101107
$(let $field = DecodeMut::decode(r, s);)*
102108
$name::$variant $(($field))*
103109
})*

src/librustc/hir/intravisit.rs

+7-6
Original file line numberDiff line numberDiff line change
@@ -454,7 +454,7 @@ pub fn walk_trait_ref<'v, V>(visitor: &mut V, trait_ref: &'v TraitRef)
454454

455455
pub fn walk_item<'v, V: Visitor<'v>>(visitor: &mut V, item: &'v Item) {
456456
visitor.visit_vis(&item.vis);
457-
visitor.visit_name(item.span, item.name);
457+
visitor.visit_ident(item.ident);
458458
match item.node {
459459
ItemKind::ExternCrate(orig_name) => {
460460
visitor.visit_id(item.id);
@@ -472,7 +472,7 @@ pub fn walk_item<'v, V: Visitor<'v>>(visitor: &mut V, item: &'v Item) {
472472
visitor.visit_nested_body(body);
473473
}
474474
ItemKind::Fn(ref declaration, header, ref generics, body_id) => {
475-
visitor.visit_fn(FnKind::ItemFn(item.name,
475+
visitor.visit_fn(FnKind::ItemFn(item.ident.name,
476476
generics,
477477
header,
478478
&item.vis,
@@ -528,7 +528,8 @@ pub fn walk_item<'v, V: Visitor<'v>>(visitor: &mut V, item: &'v Item) {
528528
ItemKind::Union(ref struct_definition, ref generics) => {
529529
visitor.visit_generics(generics);
530530
visitor.visit_id(item.id);
531-
visitor.visit_variant_data(struct_definition, item.name, generics, item.id, item.span);
531+
visitor.visit_variant_data(struct_definition, item.ident.name, generics, item.id,
532+
item.span);
532533
}
533534
ItemKind::Trait(.., ref generics, ref bounds, ref trait_item_refs) => {
534535
visitor.visit_id(item.id);
@@ -569,9 +570,9 @@ pub fn walk_variant<'v, V: Visitor<'v>>(visitor: &mut V,
569570
variant: &'v Variant,
570571
generics: &'v Generics,
571572
parent_item_id: NodeId) {
572-
visitor.visit_name(variant.span, variant.node.name);
573+
visitor.visit_ident(variant.node.ident);
573574
visitor.visit_variant_data(&variant.node.data,
574-
variant.node.name,
575+
variant.node.ident.name,
575576
generics,
576577
parent_item_id,
577578
variant.span);
@@ -720,7 +721,7 @@ pub fn walk_pat<'v, V: Visitor<'v>>(visitor: &mut V, pattern: &'v Pat) {
720721
pub fn walk_foreign_item<'v, V: Visitor<'v>>(visitor: &mut V, foreign_item: &'v ForeignItem) {
721722
visitor.visit_id(foreign_item.id);
722723
visitor.visit_vis(&foreign_item.vis);
723-
visitor.visit_name(foreign_item.span, foreign_item.name);
724+
visitor.visit_ident(foreign_item.ident);
724725

725726
match foreign_item.node {
726727
ForeignItemKind::Fn(ref function_declaration, ref param_names, ref generics) => {

src/librustc/hir/lowering.rs

+27-27
Original file line numberDiff line numberDiff line change
@@ -1360,7 +1360,7 @@ impl<'a> LoweringContext<'a> {
13601360
let exist_ty_item = hir::Item {
13611361
id: exist_ty_id.node_id,
13621362
hir_id: exist_ty_id.hir_id,
1363-
name: keywords::Invalid.name(),
1363+
ident: keywords::Invalid.ident(),
13641364
attrs: Default::default(),
13651365
node: exist_ty_item_kind,
13661366
vis: respan(span.shrink_to_lo(), hir::VisibilityKind::Inherited),
@@ -1563,7 +1563,7 @@ impl<'a> LoweringContext<'a> {
15631563
fn lower_variant(&mut self, v: &Variant) -> hir::Variant {
15641564
Spanned {
15651565
node: hir::VariantKind {
1566-
name: v.node.ident.name,
1566+
ident: v.node.ident,
15671567
attrs: self.lower_attrs(&v.node.attrs),
15681568
data: self.lower_variant_data(&v.node.data),
15691569
disr_expr: v.node.disr_expr.as_ref().map(|e| self.lower_anon_const(e)),
@@ -2737,7 +2737,7 @@ impl<'a> LoweringContext<'a> {
27372737
fn lower_item_kind(
27382738
&mut self,
27392739
id: NodeId,
2740-
name: &mut Name,
2740+
ident: &mut Ident,
27412741
attrs: &hir::HirVec<Attribute>,
27422742
vis: &mut hir::Visibility,
27432743
i: &ItemKind,
@@ -2751,7 +2751,7 @@ impl<'a> LoweringContext<'a> {
27512751
span: use_tree.span,
27522752
};
27532753

2754-
self.lower_use_tree(use_tree, &prefix, id, vis, name, attrs)
2754+
self.lower_use_tree(use_tree, &prefix, id, vis, ident, attrs)
27552755
}
27562756
ItemKind::Static(ref t, m, ref e) => {
27572757
let value = self.lower_body(None, |this| this.lower_expr(e));
@@ -2943,7 +2943,7 @@ impl<'a> LoweringContext<'a> {
29432943
prefix: &Path,
29442944
id: NodeId,
29452945
vis: &mut hir::Visibility,
2946-
name: &mut Name,
2946+
ident: &mut Ident,
29472947
attrs: &hir::HirVec<Attribute>,
29482948
) -> hir::ItemKind {
29492949
debug!("lower_use_tree(tree={:?})", tree);
@@ -2959,28 +2959,28 @@ impl<'a> LoweringContext<'a> {
29592959

29602960
match tree.kind {
29612961
UseTreeKind::Simple(rename, id1, id2) => {
2962-
*name = tree.ident().name;
2962+
*ident = tree.ident();
29632963

2964-
// First apply the prefix to the path
2964+
// First, apply the prefix to the path.
29652965
let mut path = Path {
29662966
segments,
29672967
span: path.span,
29682968
};
29692969

2970-
// Correctly resolve `self` imports
2970+
// Correctly resolve `self` imports.
29712971
if path.segments.len() > 1
29722972
&& path.segments.last().unwrap().ident.name == keywords::SelfLower.name()
29732973
{
29742974
let _ = path.segments.pop();
29752975
if rename.is_none() {
2976-
*name = path.segments.last().unwrap().ident.name;
2976+
*ident = path.segments.last().unwrap().ident;
29772977
}
29782978
}
29792979

29802980
let parent_def_index = self.current_hir_id_owner.last().unwrap().0;
29812981
let mut defs = self.expect_full_def_from_use(id);
2982-
// we want to return *something* from this function, so hang onto the first item
2983-
// for later
2982+
// We want to return *something* from this function, so hold onto the first item
2983+
// for later.
29842984
let ret_def = defs.next().unwrap_or(Def::Err);
29852985

29862986
// Here, we are looping over namespaces, if they exist for the definition
@@ -2990,7 +2990,7 @@ impl<'a> LoweringContext<'a> {
29902990
// two imports.
29912991
for (def, &new_node_id) in defs.zip([id1, id2].iter()) {
29922992
let vis = vis.clone();
2993-
let name = name.clone();
2993+
let ident = ident.clone();
29942994
let mut path = path.clone();
29952995
for seg in &mut path.segments {
29962996
seg.id = self.sess.next_node_id();
@@ -3031,7 +3031,7 @@ impl<'a> LoweringContext<'a> {
30313031
hir::Item {
30323032
id: new_id.node_id,
30333033
hir_id: new_id.hir_id,
3034-
name: name,
3034+
ident,
30353035
attrs: attrs.clone(),
30363036
node: item,
30373037
vis,
@@ -3057,8 +3057,8 @@ impl<'a> LoweringContext<'a> {
30573057
hir::ItemKind::Use(path, hir::UseKind::Glob)
30583058
}
30593059
UseTreeKind::Nested(ref trees) => {
3060-
// Nested imports are desugared into simple
3061-
// imports. So if we start with
3060+
// Nested imports are desugared into simple imports.
3061+
// So, if we start with
30623062
//
30633063
// ```
30643064
// pub(x) use foo::{a, b};
@@ -3079,14 +3079,14 @@ impl<'a> LoweringContext<'a> {
30793079
// `self.items`. However, the structure of this
30803080
// function also requires us to return one item, and
30813081
// for that we return the `{}` import (called the
3082-
// "`ListStem`").
3082+
// `ListStem`).
30833083

30843084
let prefix = Path {
30853085
segments,
30863086
span: prefix.span.to(path.span),
30873087
};
30883088

3089-
// Add all the nested PathListItems to the HIR.
3089+
// Add all the nested `PathListItem`s to the HIR.
30903090
for &(ref use_tree, id) in trees {
30913091
self.allocate_hir_id_counter(id, &use_tree);
30923092

@@ -3096,10 +3096,10 @@ impl<'a> LoweringContext<'a> {
30963096
} = self.lower_node_id(id);
30973097

30983098
let mut vis = vis.clone();
3099-
let mut name = name.clone();
3099+
let mut ident = ident.clone();
31003100
let mut prefix = prefix.clone();
31013101

3102-
// Give the segments new ids since they are being cloned.
3102+
// Give the segments new node-ids since they are being cloned.
31033103
for seg in &mut prefix.segments {
31043104
seg.id = self.sess.next_node_id();
31053105
}
@@ -3114,7 +3114,7 @@ impl<'a> LoweringContext<'a> {
31143114
&prefix,
31153115
new_id,
31163116
&mut vis,
3117-
&mut name,
3117+
&mut ident,
31183118
attrs);
31193119

31203120
let vis_kind = match vis.node {
@@ -3138,7 +3138,7 @@ impl<'a> LoweringContext<'a> {
31383138
hir::Item {
31393139
id: new_id,
31403140
hir_id: new_hir_id,
3141-
name,
3141+
ident,
31423142
attrs: attrs.clone(),
31433143
node: item,
31443144
vis,
@@ -3165,7 +3165,7 @@ impl<'a> LoweringContext<'a> {
31653165
*vis = respan(prefix.span.shrink_to_lo(), hir::VisibilityKind::Inherited);
31663166
}
31673167
hir::VisibilityKind::Restricted { .. } => {
3168-
// do nothing here, as described in the comment on the match
3168+
// Do nothing here, as described in the comment on the match.
31693169
}
31703170
}
31713171

@@ -3413,15 +3413,15 @@ impl<'a> LoweringContext<'a> {
34133413
}
34143414

34153415
pub fn lower_item(&mut self, i: &Item) -> Option<hir::Item> {
3416-
let mut name = i.ident.name;
3416+
let mut ident = i.ident;
34173417
let mut vis = self.lower_visibility(&i.vis, None);
34183418
let attrs = self.lower_attrs(&i.attrs);
34193419
if let ItemKind::MacroDef(ref def) = i.node {
34203420
if !def.legacy || attr::contains_name(&i.attrs, "macro_export") ||
34213421
attr::contains_name(&i.attrs, "rustc_doc_only_macro") {
34223422
let body = self.lower_token_stream(def.stream());
34233423
self.exported_macros.push(hir::MacroDef {
3424-
name,
3424+
name: ident.name,
34253425
vis,
34263426
attrs,
34273427
id: i.id,
@@ -3433,14 +3433,14 @@ impl<'a> LoweringContext<'a> {
34333433
return None;
34343434
}
34353435

3436-
let node = self.lower_item_kind(i.id, &mut name, &attrs, &mut vis, &i.node);
3436+
let node = self.lower_item_kind(i.id, &mut ident, &attrs, &mut vis, &i.node);
34373437

34383438
let LoweredNodeId { node_id, hir_id } = self.lower_node_id(i.id);
34393439

34403440
Some(hir::Item {
34413441
id: node_id,
34423442
hir_id,
3443-
name,
3443+
ident,
34443444
attrs,
34453445
node,
34463446
vis,
@@ -3453,7 +3453,7 @@ impl<'a> LoweringContext<'a> {
34533453
let def_id = self.resolver.definitions().local_def_id(node_id);
34543454
hir::ForeignItem {
34553455
id: node_id,
3456-
name: i.ident.name,
3456+
ident: i.ident,
34573457
attrs: self.lower_attrs(&i.attrs),
34583458
node: match i.node {
34593459
ForeignItemKind::Fn(ref fdec, ref generics) => {

src/librustc/hir/map/blocks.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -228,7 +228,7 @@ impl<'a> FnLikeNode<'a> {
228228
ast::ItemKind::Fn(ref decl, header, ref generics, block) =>
229229
item_fn(ItemFnParts {
230230
id: i.id,
231-
name: i.name,
231+
name: i.ident.name,
232232
decl: &decl,
233233
body: block,
234234
vis: &i.vis,

0 commit comments

Comments
 (0)