Skip to content

Commit 6670134

Browse files
committed
Parse unnamed fields and anonymous structs or unions
Anonymous structs or unions are parsed as a special type of field, instead of a gerenal kind of `Ty`.
1 parent cd05ffa commit 6670134

File tree

23 files changed

+415
-161
lines changed

23 files changed

+415
-161
lines changed

compiler/rustc_ast/src/ast.rs

+147-5
Original file line numberDiff line numberDiff line change
@@ -2060,10 +2060,6 @@ pub enum TyKind {
20602060
Never,
20612061
/// A tuple (`(A, B, C, D,...)`).
20622062
Tup(ThinVec<P<Ty>>),
2063-
/// An anonymous struct type i.e. `struct { foo: Type }`
2064-
AnonStruct(ThinVec<FieldDef>),
2065-
/// An anonymous union type i.e. `union { bar: Type }`
2066-
AnonUnion(ThinVec<FieldDef>),
20672063
/// A path (`module::module::...::Type`), optionally
20682064
/// "qualified", e.g., `<Vec<T> as SomeTrait>::SomeType`.
20692065
///
@@ -2689,6 +2685,93 @@ impl VisibilityKind {
26892685
}
26902686
}
26912687

2688+
#[derive(Clone, Copy, Encodable, Decodable, Debug)]
2689+
pub enum AnonRecordKind {
2690+
Struct,
2691+
Union,
2692+
}
2693+
2694+
impl AnonRecordKind {
2695+
/// Returns the lowercase name.
2696+
pub fn name(self) -> &'static str {
2697+
match self {
2698+
Self::Struct => "struct",
2699+
Self::Union => "union",
2700+
}
2701+
}
2702+
}
2703+
2704+
/// An anonymous struct or union, i.e. `struct { foo: Foo }` or `union { foo: Foo }`.
2705+
#[derive(Clone, Encodable, Decodable, Debug)]
2706+
pub struct AnonRecord {
2707+
pub id: NodeId,
2708+
pub span: Span,
2709+
pub fields: ThinVec<FieldDef>,
2710+
pub kind: AnonRecordKind,
2711+
pub recovered: bool,
2712+
}
2713+
2714+
impl AnonRecord {
2715+
pub fn is_union(&self) -> bool {
2716+
matches!(self.kind, AnonRecordKind::Union)
2717+
}
2718+
pub fn is_struct(&self) -> bool {
2719+
matches!(self.kind, AnonRecordKind::Struct)
2720+
}
2721+
}
2722+
2723+
/// Type of fields in a struct, variant or union.
2724+
#[derive(Clone, Encodable, Decodable, Debug)]
2725+
pub enum FieldTy {
2726+
Ty(P<Ty>),
2727+
/// An anonymous struct or union, i.e. `struct { foo: Foo }` or `union { foo: Foo }`.
2728+
// AnonRecord(P<Item>),
2729+
AnonRecord(P<AnonRecord>),
2730+
}
2731+
2732+
impl From<P<Ty>> for FieldTy {
2733+
fn from(ty: P<Ty>) -> Self {
2734+
Self::Ty(ty)
2735+
}
2736+
}
2737+
2738+
impl From<AnonRecord> for FieldTy {
2739+
fn from(anon_record: AnonRecord) -> Self {
2740+
Self::AnonRecord(P(anon_record))
2741+
}
2742+
}
2743+
2744+
impl FieldTy {
2745+
pub fn id(&self) -> NodeId {
2746+
match self {
2747+
Self::Ty(ty) => ty.id,
2748+
Self::AnonRecord(anon_record) => anon_record.id,
2749+
}
2750+
}
2751+
2752+
pub fn span(&self) -> Span {
2753+
match self {
2754+
Self::Ty(ty) => ty.span,
2755+
Self::AnonRecord(anon_record) => anon_record.span,
2756+
}
2757+
}
2758+
2759+
pub fn as_ty(&self) -> Option<&P<Ty>> {
2760+
match self {
2761+
Self::Ty(ty) => Some(ty),
2762+
_ => None,
2763+
}
2764+
}
2765+
2766+
/// Expects a `Ty`, otherwise panics.
2767+
pub fn expect_ty(&self) -> &P<Ty> {
2768+
let FieldTy::Ty(ty) = &self else {
2769+
panic!("expect a type, found {self:?}");
2770+
};
2771+
ty
2772+
}
2773+
}
2774+
26922775
/// Field definition in a struct, variant or union.
26932776
///
26942777
/// E.g., `bar: usize` as in `struct Foo { bar: usize }`.
@@ -2700,7 +2783,7 @@ pub struct FieldDef {
27002783
pub vis: Visibility,
27012784
pub ident: Option<Ident>,
27022785

2703-
pub ty: P<Ty>,
2786+
pub ty: FieldTy,
27042787
pub is_placeholder: bool,
27052788
}
27062789

@@ -2730,6 +2813,27 @@ impl VariantData {
27302813
}
27312814
}
27322815

2816+
/// Return all fields of this variant, with anonymous structs or unions flattened.
2817+
pub fn all_fields(&self) -> AllFields<'_> {
2818+
AllFields { iters: smallvec::smallvec![self.fields().iter()] }
2819+
}
2820+
2821+
/// Return whether this variant contains inner anonymous unions.
2822+
pub fn find_inner_union(&self) -> Option<Span> {
2823+
// We only check the record-like structs
2824+
let VariantData::Struct(fields, ..) = self else { return None };
2825+
fn find_union(fields: &[FieldDef]) -> Option<Span> {
2826+
fields.iter().find_map(|field| {
2827+
let FieldTy::AnonRecord(anon_record) = &field.ty else { return None };
2828+
anon_record
2829+
.is_union()
2830+
.then(|| anon_record.span)
2831+
.or_else(|| find_union(&anon_record.fields))
2832+
})
2833+
}
2834+
find_union(&fields)
2835+
}
2836+
27332837
/// Return the `NodeId` of this variant's constructor, if it has one.
27342838
pub fn ctor_node_id(&self) -> Option<NodeId> {
27352839
match *self {
@@ -2739,6 +2843,44 @@ impl VariantData {
27392843
}
27402844
}
27412845

2846+
/// Iterator of all fields of a `VariantData`.
2847+
///
2848+
/// It iterates on the field tree in preorder, where the unnamed fields with anonymous structs or unions
2849+
/// are flattened to their inner fields.
2850+
pub struct AllFields<'a> {
2851+
iters: smallvec::SmallVec<[std::slice::Iter<'a, FieldDef>; 1]>,
2852+
}
2853+
2854+
impl<'a> Iterator for AllFields<'a> {
2855+
type Item = &'a FieldDef;
2856+
2857+
fn next(&mut self) -> Option<Self::Item> {
2858+
let mut top = self.iters.last_mut()?;
2859+
loop {
2860+
if let Some(field) = top.next() {
2861+
if let FieldTy::AnonRecord(anon_record) = &field.ty {
2862+
self.iters.push(anon_record.fields.iter());
2863+
top = self.iters.last_mut().unwrap();
2864+
continue;
2865+
}
2866+
return Some(field);
2867+
}
2868+
self.iters.pop();
2869+
top = self.iters.last_mut()?;
2870+
}
2871+
}
2872+
2873+
fn size_hint(&self) -> (usize, Option<usize>) {
2874+
match &self.iters[..] {
2875+
[] => (0, Some(0)),
2876+
[single] => (single.size_hint().0, None),
2877+
[first, .., last] => (first.size_hint().0 + last.size_hint().0, None),
2878+
}
2879+
}
2880+
}
2881+
2882+
impl std::iter::FusedIterator for AllFields<'_> {}
2883+
27422884
/// An item definition.
27432885
#[derive(Clone, Encodable, Decodable, Debug)]
27442886
pub struct Item<K = ItemKind> {

compiler/rustc_ast/src/mut_visit.rs

+17-4
Original file line numberDiff line numberDiff line change
@@ -251,6 +251,10 @@ pub trait MutVisitor: Sized {
251251
noop_visit_variant_data(vdata, self);
252252
}
253253

254+
fn visit_anon_record(&mut self, anon_record: &mut AnonRecord) {
255+
noop_visit_anon_record(anon_record, self);
256+
}
257+
254258
fn flat_map_generic_param(&mut self, param: GenericParam) -> SmallVec<[GenericParam; 1]> {
255259
noop_flat_map_generic_param(param, self)
256260
}
@@ -514,9 +518,6 @@ pub fn noop_visit_ty<T: MutVisitor>(ty: &mut P<Ty>, vis: &mut T) {
514518
visit_vec(bounds, |bound| vis.visit_param_bound(bound));
515519
}
516520
TyKind::MacCall(mac) => vis.visit_mac_call(mac),
517-
TyKind::AnonStruct(fields) | TyKind::AnonUnion(fields) => {
518-
fields.flat_map_in_place(|field| vis.flat_map_field_def(field));
519-
}
520521
}
521522
vis.visit_span(span);
522523
visit_lazy_tts(tokens, vis);
@@ -982,6 +983,13 @@ pub fn noop_visit_variant_data<T: MutVisitor>(vdata: &mut VariantData, vis: &mut
982983
}
983984
}
984985

986+
pub fn noop_visit_anon_record<T: MutVisitor>(anon_record: &mut AnonRecord, vis: &mut T) {
987+
let AnonRecord { span, id, fields, recovered: _recovered, kind: _kind } = anon_record;
988+
vis.visit_span(span);
989+
vis.visit_id(id);
990+
fields.flat_map_in_place(|field| vis.flat_map_field_def(field));
991+
}
992+
985993
pub fn noop_visit_trait_ref<T: MutVisitor>(TraitRef { path, ref_id }: &mut TraitRef, vis: &mut T) {
986994
vis.visit_path(path);
987995
vis.visit_id(ref_id);
@@ -1003,7 +1011,12 @@ pub fn noop_flat_map_field_def<T: MutVisitor>(
10031011
visit_opt(ident, |ident| visitor.visit_ident(ident));
10041012
visitor.visit_vis(vis);
10051013
visitor.visit_id(id);
1006-
visitor.visit_ty(ty);
1014+
match ty {
1015+
FieldTy::Ty(ty) => {
1016+
visitor.visit_ty(ty);
1017+
}
1018+
FieldTy::AnonRecord(anon_record) => visitor.visit_anon_record(anon_record),
1019+
}
10071020
visit_attrs(attrs, visitor);
10081021
smallvec![fd]
10091022
}

compiler/rustc_ast/src/visit.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -441,9 +441,6 @@ pub fn walk_ty<'a, V: Visitor<'a>>(visitor: &mut V, typ: &'a Ty) {
441441
TyKind::Infer | TyKind::ImplicitSelf | TyKind::Err => {}
442442
TyKind::MacCall(mac) => visitor.visit_mac_call(mac),
443443
TyKind::Never | TyKind::CVarArgs => {}
444-
TyKind::AnonStruct(ref fields, ..) | TyKind::AnonUnion(ref fields, ..) => {
445-
walk_list!(visitor, visit_field_def, fields)
446-
}
447444
}
448445
}
449446

@@ -716,7 +713,14 @@ pub fn walk_field_def<'a, V: Visitor<'a>>(visitor: &mut V, field: &'a FieldDef)
716713
if let Some(ident) = field.ident {
717714
visitor.visit_ident(ident);
718715
}
719-
visitor.visit_ty(&field.ty);
716+
match &field.ty {
717+
FieldTy::Ty(ty) => {
718+
visitor.visit_ty(ty);
719+
}
720+
FieldTy::AnonRecord(anon_record) => {
721+
walk_list!(visitor, visit_field_def, &anon_record.fields);
722+
}
723+
}
720724
walk_list!(visitor, visit_attribute, &field.attrs);
721725
}
722726

compiler/rustc_ast_lowering/src/item.rs

+29-11
Original file line numberDiff line numberDiff line change
@@ -708,17 +708,35 @@ impl<'hir> LoweringContext<'_, 'hir> {
708708
}
709709

710710
fn lower_field_def(&mut self, (index, f): (usize, &FieldDef)) -> hir::FieldDef<'hir> {
711-
let ty = if let TyKind::Path(qself, path) = &f.ty.kind {
712-
let t = self.lower_path_ty(
713-
&f.ty,
714-
qself,
715-
path,
716-
ParamMode::ExplicitNamed, // no `'_` in declarations (Issue #61124)
717-
&ImplTraitContext::Disallowed(ImplTraitPosition::FieldTy),
718-
);
719-
self.arena.alloc(t)
720-
} else {
721-
self.lower_ty(&f.ty, &ImplTraitContext::Disallowed(ImplTraitPosition::FieldTy))
711+
let ty = match &f.ty {
712+
FieldTy::Ty(ty) if let TyKind::Path(qself, path) = &ty.kind => {
713+
let t = self.lower_path_ty(
714+
ty,
715+
qself,
716+
path,
717+
ParamMode::ExplicitNamed, // no `'_` in declarations (Issue #61124)
718+
&ImplTraitContext::Disallowed(ImplTraitPosition::FieldTy),
719+
);
720+
self.arena.alloc(t)
721+
}
722+
FieldTy::Ty(ty) => {
723+
self.lower_ty(ty, &ImplTraitContext::Disallowed(ImplTraitPosition::FieldTy))
724+
}
725+
FieldTy::AnonRecord(anon_record) => {
726+
let struct_or_union = anon_record.kind.name();
727+
let hir_id = self.lower_node_id(anon_record.id);
728+
let span = anon_record.span;
729+
// FIXME(unnamed_fields): IMPLEMENTATION IN PROGRESS
730+
#[allow(rustc::untranslatable_diagnostic)]
731+
#[allow(rustc::diagnostic_outside_of_impl)]
732+
let kind = hir::TyKind::Err(
733+
self.tcx
734+
.sess
735+
.span_err(span, format!("anonymous {struct_or_union}s are unimplemented")),
736+
);
737+
let ty = hir::Ty { hir_id, kind, span };
738+
self.arena.alloc(ty)
739+
}
722740
};
723741
let hir_id = self.lower_node_id(f.id);
724742
self.lower_attrs(hir_id, &f.attrs);

compiler/rustc_ast_lowering/src/lib.rs

+1-12
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
#![cfg_attr(not(bootstrap), doc(rust_logo))]
3636
#![feature(box_patterns)]
3737
#![feature(let_chains)]
38+
#![feature(if_let_guard)]
3839
#![feature(never_type)]
3940
#![recursion_limit = "256"]
4041
#![deny(rustc::untranslatable_diagnostic)]
@@ -1290,18 +1291,6 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
12901291
TyKind::Err => {
12911292
hir::TyKind::Err(self.tcx.sess.delay_span_bug(t.span, "TyKind::Err lowered"))
12921293
}
1293-
// FIXME(unnamed_fields): IMPLEMENTATION IN PROGRESS
1294-
#[allow(rustc::untranslatable_diagnostic)]
1295-
#[allow(rustc::diagnostic_outside_of_impl)]
1296-
TyKind::AnonStruct(ref _fields) => hir::TyKind::Err(
1297-
self.tcx.sess.span_err(t.span, "anonymous structs are unimplemented"),
1298-
),
1299-
// FIXME(unnamed_fields): IMPLEMENTATION IN PROGRESS
1300-
#[allow(rustc::untranslatable_diagnostic)]
1301-
#[allow(rustc::diagnostic_outside_of_impl)]
1302-
TyKind::AnonUnion(ref _fields) => hir::TyKind::Err(
1303-
self.tcx.sess.span_err(t.span, "anonymous unions are unimplemented"),
1304-
),
13051294
TyKind::Slice(ty) => hir::TyKind::Slice(self.lower_ty(ty, itctx)),
13061295
TyKind::Ptr(mt) => hir::TyKind::Ptr(self.lower_mt(mt, itctx)),
13071296
TyKind::Ref(region, mt) => {

0 commit comments

Comments
 (0)