Skip to content

Commit 28ed8b1

Browse files
committed
Fix #[derive] for empty tuple structs/variants
1 parent 7ac11ca commit 28ed8b1

File tree

5 files changed

+32
-10
lines changed

5 files changed

+32
-10
lines changed

src/libsyntax/ext/build.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -844,7 +844,7 @@ impl<'a> AstBuilder for ExtCtxt<'a> {
844844
}
845845
fn pat_enum(&self, span: Span, path: ast::Path, subpats: Vec<P<ast::Pat>>) -> P<ast::Pat> {
846846
let pat = if subpats.is_empty() {
847-
PatKind::Path(None, path)
847+
PatKind::Struct(path, Vec::new(), false)
848848
} else {
849849
PatKind::TupleStruct(path, subpats, None)
850850
};

src/libsyntax_ext/deriving/decodable.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -110,7 +110,7 @@ fn decodable_substructure(cx: &mut ExtCtxt,
110110
return match *substr.fields {
111111
StaticStruct(_, ref summary) => {
112112
let nfields = match *summary {
113-
Unnamed(ref fields) => fields.len(),
113+
Unnamed(ref fields, _) => fields.len(),
114114
Named(ref fields) => fields.len(),
115115
};
116116
let read_struct_field = cx.ident_of("read_struct_field");
@@ -193,9 +193,9 @@ fn decode_static_fields<F>(cx: &mut ExtCtxt,
193193
where F: FnMut(&mut ExtCtxt, Span, InternedString, usize) -> P<Expr>
194194
{
195195
match *fields {
196-
Unnamed(ref fields) => {
196+
Unnamed(ref fields, is_tuple) => {
197197
let path_expr = cx.expr_path(outer_pat_path);
198-
if fields.is_empty() {
198+
if !is_tuple {
199199
path_expr
200200
} else {
201201
let fields = fields.iter()

src/libsyntax_ext/deriving/default.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -57,8 +57,8 @@ fn default_substructure(cx: &mut ExtCtxt, trait_span: Span, substr: &Substructur
5757
return match *substr.fields {
5858
StaticStruct(_, ref summary) => {
5959
match *summary {
60-
Unnamed(ref fields) => {
61-
if fields.is_empty() {
60+
Unnamed(ref fields, is_tuple) => {
61+
if !is_tuple {
6262
cx.expr_ident(trait_span, substr.type_ident)
6363
} else {
6464
let exprs = fields.iter().map(|sp| default_call(*sp)).collect();

src/libsyntax_ext/deriving/generic/mod.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -294,8 +294,8 @@ pub struct FieldInfo<'a> {
294294

295295
/// Fields for a static method
296296
pub enum StaticFields {
297-
/// Tuple structs/enum variants like this.
298-
Unnamed(Vec<Span>),
297+
/// Tuple and unit structs/enum variants like this.
298+
Unnamed(Vec<Span>, bool /*is tuple*/),
299299
/// Normal structs/struct variants.
300300
Named(Vec<(Ident, Span)>),
301301
}
@@ -1470,7 +1470,7 @@ impl<'a> TraitDef<'a> {
14701470
(_, false) => Named(named_idents),
14711471
// empty structs
14721472
_ if struct_def.is_struct() => Named(named_idents),
1473-
_ => Unnamed(just_spans),
1473+
_ => Unnamed(just_spans, struct_def.is_tuple()),
14741474
}
14751475
}
14761476

src/test/run-pass-fulldeps/empty-struct-braces-derive.rs

+23-1
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,9 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11-
// `#[derive(Trait)]` works for empty structs/variants with braces
11+
// `#[derive(Trait)]` works for empty structs/variants with braces or parens.
1212

13+
#![feature(relaxed_adts)]
1314
#![feature(rustc_private)]
1415

1516
extern crate serialize as rustc_serialize;
@@ -18,11 +19,16 @@ extern crate serialize as rustc_serialize;
1819
Default, Debug, RustcEncodable, RustcDecodable)]
1920
struct S {}
2021

22+
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash,
23+
Default, Debug, RustcEncodable, RustcDecodable)]
24+
struct Z();
25+
2126
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash,
2227
Debug, RustcEncodable, RustcDecodable)]
2328
enum E {
2429
V {},
2530
U,
31+
W(),
2632
}
2733

2834
fn main() {
@@ -34,11 +40,27 @@ fn main() {
3440
assert!(!(s < s1));
3541
assert_eq!(format!("{:?}", s), "S");
3642

43+
let z = Z();
44+
let z1 = z;
45+
let z2 = z.clone();
46+
assert_eq!(z, z1);
47+
assert_eq!(z, z2);
48+
assert!(!(z < z1));
49+
assert_eq!(format!("{:?}", z), "Z");
50+
3751
let e = E::V {};
3852
let e1 = e;
3953
let e2 = e.clone();
4054
assert_eq!(e, e1);
4155
assert_eq!(e, e2);
4256
assert!(!(e < e1));
4357
assert_eq!(format!("{:?}", e), "V");
58+
59+
let e = E::W();
60+
let e1 = e;
61+
let e2 = e.clone();
62+
assert_eq!(e, e1);
63+
assert_eq!(e, e2);
64+
assert!(!(e < e1));
65+
assert_eq!(format!("{:?}", e), "W");
4466
}

0 commit comments

Comments
 (0)