@@ -1126,75 +1126,43 @@ impl<'a> MethodDef<'a> {
1126
1126
/// A1,
1127
1127
/// A2(i32)
1128
1128
/// }
1129
- ///
1130
- /// // is equivalent to
1131
- ///
1132
- /// impl PartialEq for A {
1129
+ /// ```
1130
+ /// is equivalent to:
1131
+ /// ```
1132
+ /// impl ::core::cmp::PartialEq for A {
1133
+ /// #[inline]
1133
1134
/// fn eq(&self, other: &A) -> bool {
1134
- /// use A::*;
1135
- /// match (&*self, &*other) {
1136
- /// (&A1, &A1) => true,
1137
- /// (&A2(ref self_0),
1138
- /// &A2(ref __arg_1_0)) => (*self_0).eq(&(*__arg_1_0)),
1139
- /// _ => {
1140
- /// let __self_vi = match *self { A1 => 0, A2(..) => 1 };
1141
- /// let __arg_1_vi = match *other { A1 => 0, A2(..) => 1 };
1142
- /// false
1135
+ /// {
1136
+ /// let __self_vi = ::core::intrinsics::discriminant_value(&*self);
1137
+ /// let __arg_1_vi = ::core::intrinsics::discriminant_value(&*other);
1138
+ /// if true && __self_vi == __arg_1_vi {
1139
+ /// match (&*self, &*other) {
1140
+ /// (&A::A2(ref __self_0), &A::A2(ref __arg_1_0)) =>
1141
+ /// (*__self_0) == (*__arg_1_0),
1142
+ /// _ => true,
1143
+ /// }
1144
+ /// } else {
1145
+ /// false // catch-all handler
1143
1146
/// }
1144
1147
/// }
1145
1148
/// }
1146
1149
/// }
1147
1150
/// ```
1148
- ///
1149
- /// (Of course `__self_vi` and `__arg_1_vi` are unused for
1150
- /// `PartialEq`, and those subcomputations will hopefully be removed
1151
- /// as their results are unused. The point of `__self_vi` and
1152
- /// `__arg_1_vi` is for `PartialOrd`; see #15503.)
1153
- fn expand_enum_method_body < ' b > (
1154
- & self ,
1155
- cx : & mut ExtCtxt < ' _ > ,
1156
- trait_ : & TraitDef < ' b > ,
1157
- enum_def : & ' b EnumDef ,
1158
- type_ident : Ident ,
1159
- self_args : Vec < P < Expr > > ,
1160
- nonself_args : & [ P < Expr > ] ,
1161
- ) -> P < Expr > {
1162
- self . build_enum_match_tuple ( cx, trait_, enum_def, type_ident, self_args, nonself_args)
1163
- }
1164
-
1165
1151
/// Creates a match for a tuple of all `self_args`, where either all
1166
1152
/// variants match, or it falls into a catch-all for when one variant
1167
1153
/// does not match.
1168
-
1154
+ ///
1169
1155
/// There are N + 1 cases because is a case for each of the N
1170
1156
/// variants where all of the variants match, and one catch-all for
1171
1157
/// when one does not match.
1172
-
1158
+ ///
1173
1159
/// As an optimization we generate code which checks whether all variants
1174
1160
/// match first which makes llvm see that C-like enums can be compiled into
1175
1161
/// a simple equality check (for PartialEq).
1176
-
1162
+ ///
1177
1163
/// The catch-all handler is provided access the variant index values
1178
1164
/// for each of the self-args, carried in precomputed variables.
1179
-
1180
- /// ```{.text}
1181
- /// let __self0_vi = std::intrinsics::discriminant_value(&self);
1182
- /// let __self1_vi = std::intrinsics::discriminant_value(&arg1);
1183
- /// let __self2_vi = std::intrinsics::discriminant_value(&arg2);
1184
- ///
1185
- /// if __self0_vi == __self1_vi && __self0_vi == __self2_vi && ... {
1186
- /// match (...) {
1187
- /// (Variant1, Variant1, ...) => Body1
1188
- /// (Variant2, Variant2, ...) => Body2,
1189
- /// ...
1190
- /// _ => ::core::intrinsics::unreachable()
1191
- /// }
1192
- /// }
1193
- /// else {
1194
- /// ... // catch-all remainder can inspect above variant index values.
1195
- /// }
1196
- /// ```
1197
- fn build_enum_match_tuple < ' b > (
1165
+ fn expand_enum_method_body < ' b > (
1198
1166
& self ,
1199
1167
cx : & mut ExtCtxt < ' _ > ,
1200
1168
trait_ : & TraitDef < ' b > ,
@@ -1392,37 +1360,32 @@ impl<'a> MethodDef<'a> {
1392
1360
//
1393
1361
// i.e., for `enum E<T> { A, B(1), C(T, T) }`, and a deriving
1394
1362
// with three Self args, builds three statements:
1395
- //
1396
1363
// ```
1397
- // let __self0_vi = std::intrinsics::discriminant_value(&self);
1398
- // let __self1_vi = std::intrinsics::discriminant_value(&arg1);
1399
- // let __self2_vi = std::intrinsics::discriminant_value(&arg2);
1364
+ // let __self_vi = std::intrinsics::discriminant_value(&self);
1365
+ // let __arg_1_vi = std::intrinsics::discriminant_value(&arg1);
1366
+ // let __arg_2_vi = std::intrinsics::discriminant_value(&arg2);
1400
1367
// ```
1401
1368
let mut index_let_stmts: Vec < ast:: Stmt > = Vec :: with_capacity ( vi_idents. len ( ) + 1 ) ;
1402
1369
1403
- // We also build an expression which checks whether all discriminants are equal
1404
- // discriminant_test = __self0_vi == __self1_vi && __self0_vi == __self2_vi && ...
1370
+ // We also build an expression which checks whether all discriminants are equal:
1371
+ // `__self_vi == __arg_1_vi && __self_vi == __arg_2_vi && ...`
1405
1372
let mut discriminant_test = cx. expr_bool ( span, true ) ;
1406
-
1407
- let mut first_ident = None ;
1408
- for ( & ident, self_arg) in iter:: zip ( & vi_idents, & self_args) {
1373
+ for ( i, ( & ident, self_arg) ) in iter:: zip ( & vi_idents, & self_args) . enumerate ( ) {
1409
1374
let self_addr = cx. expr_addr_of ( span, self_arg. clone ( ) ) ;
1410
1375
let variant_value =
1411
1376
deriving:: call_intrinsic ( cx, span, sym:: discriminant_value, vec ! [ self_addr] ) ;
1412
1377
let let_stmt = cx. stmt_let ( span, false , ident, variant_value) ;
1413
1378
index_let_stmts. push ( let_stmt) ;
1414
1379
1415
- match first_ident {
1416
- Some ( first) => {
1417
- let first_expr = cx. expr_ident ( span, first) ;
1418
- let id = cx. expr_ident ( span, ident) ;
1419
- let test = cx. expr_binary ( span, BinOpKind :: Eq , first_expr, id) ;
1420
- discriminant_test =
1421
- cx. expr_binary ( span, BinOpKind :: And , discriminant_test, test)
1422
- }
1423
- None => {
1424
- first_ident = Some ( ident) ;
1425
- }
1380
+ if i > 0 {
1381
+ let id0 = cx. expr_ident ( span, vi_idents[ 0 ] ) ;
1382
+ let id = cx. expr_ident ( span, ident) ;
1383
+ let test = cx. expr_binary ( span, BinOpKind :: Eq , id0, id) ;
1384
+ discriminant_test = if i == 1 {
1385
+ test
1386
+ } else {
1387
+ cx. expr_binary ( span, BinOpKind :: And , discriminant_test, test)
1388
+ } ;
1426
1389
}
1427
1390
}
1428
1391
@@ -1453,7 +1416,7 @@ impl<'a> MethodDef<'a> {
1453
1416
// }
1454
1417
// }
1455
1418
// else {
1456
- // <delegated expression referring to __self0_vi , et al.>
1419
+ // <delegated expression referring to __self_vi , et al.>
1457
1420
// }
1458
1421
let all_match = cx. expr_match ( span, match_arg, match_arms) ;
1459
1422
let arm_expr = cx. expr_if ( span, discriminant_test, all_match, Some ( arm_expr) ) ;
0 commit comments