@@ -393,34 +393,47 @@ fn find_type_parameters(ty: &ast::Ty,
393
393
}
394
394
395
395
impl < ' a > TraitDef < ' a > {
396
- pub fn expand ( & self ,
396
+ pub fn expand ( self ,
397
397
cx : & mut ExtCtxt ,
398
398
mitem : & ast:: MetaItem ,
399
399
item : & ' a Annotatable ,
400
400
push : & mut FnMut ( Annotatable ) ) {
401
401
self . expand_ext ( cx, mitem, item, push, false ) ;
402
402
}
403
403
404
- pub fn expand_ext ( & self ,
404
+ pub fn expand_ext ( self ,
405
405
cx : & mut ExtCtxt ,
406
406
mitem : & ast:: MetaItem ,
407
407
item : & ' a Annotatable ,
408
408
push : & mut FnMut ( Annotatable ) ,
409
409
from_scratch : bool ) {
410
410
match * item {
411
411
Annotatable :: Item ( ref item) => {
412
+ let is_packed = item. attrs . iter ( ) . any ( |attr| {
413
+ attr:: find_repr_attrs ( & cx. parse_sess . span_diagnostic , attr)
414
+ . contains ( & attr:: ReprPacked )
415
+ } ) ;
416
+ let use_temporaries = is_packed;
412
417
let newitem = match item. node {
413
418
ast:: ItemKind :: Struct ( ref struct_def, ref generics) => {
414
- self . expand_struct_def ( cx, & struct_def, item. ident , generics, from_scratch)
419
+ self . expand_struct_def ( cx, & struct_def, item. ident , generics, from_scratch,
420
+ use_temporaries)
415
421
}
416
422
ast:: ItemKind :: Enum ( ref enum_def, ref generics) => {
423
+ // We ignore `use_temporaries` here, because
424
+ // `repr(packed)` enums cause an error later on.
425
+ //
426
+ // This can only cause further compilation errors
427
+ // downstream in blatantly illegal code, so it
428
+ // is fine.
417
429
self . expand_enum_def ( cx, enum_def, & item. attrs ,
418
430
item. ident , generics, from_scratch)
419
431
}
420
432
ast:: ItemKind :: Union ( ref struct_def, ref generics) => {
421
433
if self . supports_unions {
422
434
self . expand_struct_def ( cx, & struct_def, item. ident ,
423
- generics, from_scratch)
435
+ generics, from_scratch,
436
+ use_temporaries)
424
437
} else {
425
438
cx. span_err ( mitem. span ,
426
439
"this trait cannot be derived for unions" ) ;
@@ -674,7 +687,8 @@ impl<'a> TraitDef<'a> {
674
687
struct_def : & ' a VariantData ,
675
688
type_ident : Ident ,
676
689
generics : & Generics ,
677
- from_scratch : bool )
690
+ from_scratch : bool ,
691
+ use_temporaries : bool )
678
692
-> P < ast:: Item > {
679
693
let field_tys: Vec < P < ast:: Ty > > = struct_def. fields ( )
680
694
. iter ( )
@@ -700,7 +714,8 @@ impl<'a> TraitDef<'a> {
700
714
struct_def,
701
715
type_ident,
702
716
& self_args[ ..] ,
703
- & nonself_args[ ..] )
717
+ & nonself_args[ ..] ,
718
+ use_temporaries)
704
719
} ;
705
720
706
721
method_def. create_method ( cx,
@@ -957,14 +972,31 @@ impl<'a> MethodDef<'a> {
957
972
/// }
958
973
/// }
959
974
/// }
975
+ ///
976
+ /// // or if A is repr(packed) - note fields are matched by-value
977
+ /// // instead of by-reference.
978
+ /// impl PartialEq for A {
979
+ /// fn eq(&self, __arg_1: &A) -> bool {
980
+ /// match *self {
981
+ /// A {x: __self_0_0, y: __self_0_1} => {
982
+ /// match __arg_1 {
983
+ /// A {x: __self_1_0, y: __self_1_1} => {
984
+ /// __self_0_0.eq(&__self_1_0) && __self_0_1.eq(&__self_1_1)
985
+ /// }
986
+ /// }
987
+ /// }
988
+ /// }
989
+ /// }
990
+ /// }
960
991
/// ```
961
992
fn expand_struct_method_body < ' b > ( & self ,
962
993
cx : & mut ExtCtxt ,
963
994
trait_ : & TraitDef < ' b > ,
964
995
struct_def : & ' b VariantData ,
965
996
type_ident : Ident ,
966
997
self_args : & [ P < Expr > ] ,
967
- nonself_args : & [ P < Expr > ] )
998
+ nonself_args : & [ P < Expr > ] ,
999
+ use_temporaries : bool )
968
1000
-> P < Expr > {
969
1001
970
1002
let mut raw_fields = Vec :: new ( ) ; // Vec<[fields of self],
@@ -976,7 +1008,8 @@ impl<'a> MethodDef<'a> {
976
1008
struct_path,
977
1009
struct_def,
978
1010
& format ! ( "__self_{}" , i) ,
979
- ast:: Mutability :: Immutable ) ;
1011
+ ast:: Mutability :: Immutable ,
1012
+ use_temporaries) ;
980
1013
patterns. push ( pat) ;
981
1014
raw_fields. push ( ident_expr) ;
982
1015
}
@@ -1139,7 +1172,6 @@ impl<'a> MethodDef<'a> {
1139
1172
self_args : Vec < P < Expr > > ,
1140
1173
nonself_args : & [ P < Expr > ] )
1141
1174
-> P < Expr > {
1142
-
1143
1175
let sp = trait_. span ;
1144
1176
let variants = & enum_def. variants ;
1145
1177
@@ -1511,12 +1543,18 @@ impl<'a> TraitDef<'a> {
1511
1543
fn create_subpatterns ( & self ,
1512
1544
cx : & mut ExtCtxt ,
1513
1545
field_paths : Vec < ast:: SpannedIdent > ,
1514
- mutbl : ast:: Mutability )
1546
+ mutbl : ast:: Mutability ,
1547
+ use_temporaries : bool )
1515
1548
-> Vec < P < ast:: Pat > > {
1516
1549
field_paths. iter ( )
1517
1550
. map ( |path| {
1551
+ let binding_mode = if use_temporaries {
1552
+ ast:: BindingMode :: ByValue ( ast:: Mutability :: Immutable )
1553
+ } else {
1554
+ ast:: BindingMode :: ByRef ( mutbl)
1555
+ } ;
1518
1556
cx. pat ( path. span ,
1519
- PatKind :: Ident ( ast :: BindingMode :: ByRef ( mutbl ) , ( * path) . clone ( ) , None ) )
1557
+ PatKind :: Ident ( binding_mode , ( * path) . clone ( ) , None ) )
1520
1558
} )
1521
1559
. collect ( )
1522
1560
}
@@ -1527,8 +1565,10 @@ impl<'a> TraitDef<'a> {
1527
1565
struct_path : ast:: Path ,
1528
1566
struct_def : & ' a VariantData ,
1529
1567
prefix : & str ,
1530
- mutbl : ast:: Mutability )
1531
- -> ( P < ast:: Pat > , Vec < ( Span , Option < Ident > , P < Expr > , & ' a [ ast:: Attribute ] ) > ) {
1568
+ mutbl : ast:: Mutability ,
1569
+ use_temporaries : bool )
1570
+ -> ( P < ast:: Pat > , Vec < ( Span , Option < Ident > , P < Expr > , & ' a [ ast:: Attribute ] ) > )
1571
+ {
1532
1572
let mut paths = Vec :: new ( ) ;
1533
1573
let mut ident_exprs = Vec :: new ( ) ;
1534
1574
for ( i, struct_field) in struct_def. fields ( ) . iter ( ) . enumerate ( ) {
@@ -1538,12 +1578,18 @@ impl<'a> TraitDef<'a> {
1538
1578
span : sp,
1539
1579
node : ident,
1540
1580
} ) ;
1541
- let val = cx. expr_deref ( sp, cx. expr_path ( cx. path_ident ( sp, ident) ) ) ;
1581
+ let val = cx. expr_path ( cx. path_ident ( sp, ident) ) ;
1582
+ let val = if use_temporaries {
1583
+ val
1584
+ } else {
1585
+ cx. expr_deref ( sp, val)
1586
+ } ;
1542
1587
let val = cx. expr ( sp, ast:: ExprKind :: Paren ( val) ) ;
1588
+
1543
1589
ident_exprs. push ( ( sp, struct_field. ident , val, & struct_field. attrs [ ..] ) ) ;
1544
1590
}
1545
1591
1546
- let subpats = self . create_subpatterns ( cx, paths, mutbl) ;
1592
+ let subpats = self . create_subpatterns ( cx, paths, mutbl, use_temporaries ) ;
1547
1593
let pattern = match * struct_def {
1548
1594
VariantData :: Struct ( ..) => {
1549
1595
let field_pats = subpats. into_iter ( )
@@ -1587,7 +1633,9 @@ impl<'a> TraitDef<'a> {
1587
1633
let variant_ident = variant. node . name ;
1588
1634
let sp = variant. span . with_ctxt ( self . span . ctxt ( ) ) ;
1589
1635
let variant_path = cx. path ( sp, vec ! [ enum_ident, variant_ident] ) ;
1590
- self . create_struct_pattern ( cx, variant_path, & variant. node . data , prefix, mutbl)
1636
+ let use_temporaries = false ; // enums can't be repr(packed)
1637
+ self . create_struct_pattern ( cx, variant_path, & variant. node . data , prefix, mutbl,
1638
+ use_temporaries)
1591
1639
}
1592
1640
}
1593
1641
0 commit comments