Skip to content

Commit 02954ae

Browse files
committed
Auto merge of #31977 - bluss:partial-eq-save, r=brson
derive: Avoid emitting provided PartialEq, PartialOrd methods for c-like enums derive: Avoid emitting provided PartialEq, PartialOrd method for c-like enums `ne` is completely symmetrical with the method `eq`, and we can save rust code size and compilation time here if we only emit one of them when possible. One case where it's easy to recognize is when it's a C-like enum. Most other cases can not omit ne, because any value field may have a custom PartialEq implementation.
2 parents fc9e1d0 + edcc02b commit 02954ae

File tree

3 files changed

+43
-11
lines changed

3 files changed

+43
-11
lines changed

src/libsyntax_ext/deriving/cmp/partial_eq.rs

+9-4
Original file line numberDiff line numberDiff line change
@@ -80,17 +80,22 @@ pub fn expand_deriving_partial_eq(cx: &mut ExtCtxt,
8080
} }
8181
}
8282

83+
// avoid defining `ne` if we can
84+
// c-like enums, enums without any fields and structs without fields
85+
// can safely define only `eq`.
86+
let mut methods = vec![md!("eq", cs_eq)];
87+
if !is_type_without_fields(item) {
88+
methods.push(md!("ne", cs_ne));
89+
}
90+
8391
let trait_def = TraitDef {
8492
span: span,
8593
attributes: Vec::new(),
8694
path: path_std!(cx, core::cmp::PartialEq),
8795
additional_bounds: Vec::new(),
8896
generics: LifetimeBounds::empty(),
8997
is_unsafe: false,
90-
methods: vec!(
91-
md!("eq", cs_eq),
92-
md!("ne", cs_ne)
93-
),
98+
methods: methods,
9499
associated_types: Vec::new(),
95100
};
96101
trait_def.expand(cx, mitem, item, push)

src/libsyntax_ext/deriving/cmp/partial_ord.rs

+16-7
Original file line numberDiff line numberDiff line change
@@ -67,20 +67,29 @@ pub fn expand_deriving_partial_ord(cx: &mut ExtCtxt,
6767
}))
6868
};
6969

70+
// avoid defining extra methods if we can
71+
// c-like enums, enums without any fields and structs without fields
72+
// can safely define only `partial_cmp`.
73+
let methods = if is_type_without_fields(item) {
74+
vec![partial_cmp_def]
75+
} else {
76+
vec![
77+
partial_cmp_def,
78+
md!("lt", true, false),
79+
md!("le", true, true),
80+
md!("gt", false, false),
81+
md!("ge", false, true)
82+
]
83+
};
84+
7085
let trait_def = TraitDef {
7186
span: span,
7287
attributes: vec![],
7388
path: path_std!(cx, core::cmp::PartialOrd),
7489
additional_bounds: vec![],
7590
generics: LifetimeBounds::empty(),
7691
is_unsafe: false,
77-
methods: vec![
78-
partial_cmp_def,
79-
md!("lt", true, false),
80-
md!("le", true, true),
81-
md!("gt", false, false),
82-
md!("ge", false, true)
83-
],
92+
methods: methods,
8493
associated_types: Vec::new(),
8594
};
8695
trait_def.expand(cx, mitem, item, push)

src/libsyntax_ext/deriving/generic/mod.rs

+18
Original file line numberDiff line numberDiff line change
@@ -1647,3 +1647,21 @@ pub fn cs_same_method<F>(f: F,
16471647
}
16481648
}
16491649
}
1650+
1651+
/// Return true if the type has no value fields
1652+
/// (for an enum, no variant has any fields)
1653+
pub fn is_type_without_fields(item: &Annotatable) -> bool {
1654+
if let Annotatable::Item(ref item) = *item {
1655+
match item.node {
1656+
ast::ItemKind::Enum(ref enum_def, _) => {
1657+
enum_def.variants.iter().all(|v| v.node.data.fields().is_empty())
1658+
}
1659+
ast::ItemKind::Struct(ref variant_data, _) => {
1660+
variant_data.fields().is_empty()
1661+
}
1662+
_ => false
1663+
}
1664+
} else {
1665+
false
1666+
}
1667+
}

0 commit comments

Comments
 (0)