@@ -93,7 +93,6 @@ use rustc_index::IndexVec;
93
93
use rustc_middle:: mir:: interpret:: GlobalAlloc ;
94
94
use rustc_middle:: mir:: visit:: * ;
95
95
use rustc_middle:: mir:: * ;
96
- use rustc_middle:: ty:: adjustment:: PointerCoercion ;
97
96
use rustc_middle:: ty:: layout:: LayoutOf ;
98
97
use rustc_middle:: ty:: { self , Ty , TyCtxt , TypeAndMut } ;
99
98
use rustc_span:: def_id:: DefId ;
@@ -552,6 +551,29 @@ impl<'body, 'tcx> VnState<'body, 'tcx> {
552
551
}
553
552
value. offset ( Size :: ZERO , to, & self . ecx ) . ok ( ) ?
554
553
}
554
+ CastKind :: PointerCoercion ( ty:: adjustment:: PointerCoercion :: Unsize ) => {
555
+ let src = self . evaluated [ value] . as_ref ( ) ?;
556
+ let to = self . ecx . layout_of ( to) . ok ( ) ?;
557
+ let dest = self . ecx . allocate ( to, MemoryKind :: Stack ) . ok ( ) ?;
558
+ self . ecx . unsize_into ( src, to, & dest. clone ( ) . into ( ) ) . ok ( ) ?;
559
+ self . ecx
560
+ . alloc_mark_immutable ( dest. ptr ( ) . provenance . unwrap ( ) . alloc_id ( ) )
561
+ . ok ( ) ?;
562
+ dest. into ( )
563
+ }
564
+ CastKind :: FnPtrToPtr
565
+ | CastKind :: PtrToPtr
566
+ | CastKind :: PointerCoercion (
567
+ ty:: adjustment:: PointerCoercion :: MutToConstPointer
568
+ | ty:: adjustment:: PointerCoercion :: ArrayToPointer
569
+ | ty:: adjustment:: PointerCoercion :: UnsafeFnPointer ,
570
+ ) => {
571
+ let src = self . evaluated [ value] . as_ref ( ) ?;
572
+ let src = self . ecx . read_immediate ( src) . ok ( ) ?;
573
+ let to = self . ecx . layout_of ( to) . ok ( ) ?;
574
+ let ret = self . ecx . ptr_to_ptr ( & src, to) . ok ( ) ?;
575
+ ret. into ( )
576
+ }
555
577
_ => return None ,
556
578
} ,
557
579
} ;
@@ -778,18 +800,8 @@ impl<'body, 'tcx> VnState<'body, 'tcx> {
778
800
779
801
// Operations.
780
802
Rvalue :: Len ( ref mut place) => return self . simplify_len ( place, location) ,
781
- Rvalue :: Cast ( kind, ref mut value, to) => {
782
- let from = value. ty ( self . local_decls , self . tcx ) ;
783
- let value = self . simplify_operand ( value, location) ?;
784
- if let CastKind :: PointerCoercion (
785
- PointerCoercion :: ReifyFnPointer | PointerCoercion :: ClosureFnPointer ( _) ,
786
- ) = kind
787
- {
788
- // Each reification of a generic fn may get a different pointer.
789
- // Do not try to merge them.
790
- return self . new_opaque ( ) ;
791
- }
792
- Value :: Cast { kind, value, from, to }
803
+ Rvalue :: Cast ( ref mut kind, ref mut value, to) => {
804
+ return self . simplify_cast ( kind, value, to, location) ;
793
805
}
794
806
Rvalue :: BinaryOp ( op, box ( ref mut lhs, ref mut rhs) ) => {
795
807
let ty = lhs. ty ( self . local_decls , self . tcx ) ;
@@ -1035,6 +1047,50 @@ impl<'body, 'tcx> VnState<'body, 'tcx> {
1035
1047
}
1036
1048
}
1037
1049
1050
+ fn simplify_cast (
1051
+ & mut self ,
1052
+ kind : & mut CastKind ,
1053
+ operand : & mut Operand < ' tcx > ,
1054
+ to : Ty < ' tcx > ,
1055
+ location : Location ,
1056
+ ) -> Option < VnIndex > {
1057
+ use rustc_middle:: ty:: adjustment:: PointerCoercion :: * ;
1058
+ use CastKind :: * ;
1059
+
1060
+ let mut from = operand. ty ( self . local_decls , self . tcx ) ;
1061
+ let mut value = self . simplify_operand ( operand, location) ?;
1062
+ if from == to {
1063
+ return Some ( value) ;
1064
+ }
1065
+
1066
+ if let CastKind :: PointerCoercion ( ReifyFnPointer | ClosureFnPointer ( _) ) = kind {
1067
+ // Each reification of a generic fn may get a different pointer.
1068
+ // Do not try to merge them.
1069
+ return self . new_opaque ( ) ;
1070
+ }
1071
+
1072
+ if let PtrToPtr | PointerCoercion ( MutToConstPointer ) = kind
1073
+ && let Value :: Cast { kind : inner_kind, value : inner_value, from : inner_from, to : _ } =
1074
+ * self . get ( value)
1075
+ && let PtrToPtr | PointerCoercion ( MutToConstPointer ) = inner_kind
1076
+ {
1077
+ from = inner_from;
1078
+ value = inner_value;
1079
+ * kind = PtrToPtr ;
1080
+ if inner_from == to {
1081
+ return Some ( inner_value) ;
1082
+ }
1083
+ if let Some ( const_) = self . try_as_constant ( value) {
1084
+ * operand = Operand :: Constant ( Box :: new ( const_) ) ;
1085
+ } else if let Some ( local) = self . try_as_local ( value, location) {
1086
+ * operand = Operand :: Copy ( local. into ( ) ) ;
1087
+ self . reused_locals . insert ( local) ;
1088
+ }
1089
+ }
1090
+
1091
+ Some ( self . insert ( Value :: Cast { kind : * kind, value, from, to } ) )
1092
+ }
1093
+
1038
1094
fn simplify_len ( & mut self , place : & mut Place < ' tcx > , location : Location ) -> Option < VnIndex > {
1039
1095
// Trivial case: we are fetching a statically known length.
1040
1096
let place_ty = place. ty ( self . local_decls , self . tcx ) . ty ;
0 commit comments