@@ -956,6 +956,7 @@ impl<'a, Ty> Deref for TyLayout<'a, Ty> {
956
956
}
957
957
}
958
958
959
+ /// Trait for context types that can compute layouts of things.
959
960
pub trait LayoutOf {
960
961
type Ty ;
961
962
type TyLayout ;
@@ -966,6 +967,39 @@ pub trait LayoutOf {
966
967
}
967
968
}
968
969
970
+ /// The `TyLayout` above will always be a `MaybeResult<TyLayout<'_, Self>>`.
971
+ /// We can't add the bound due to the lifetime, but this trait is still useful when
972
+ /// writing code that's generic over the `LayoutOf` impl.
973
+ pub trait MaybeResult < T > {
974
+ type Error ;
975
+
976
+ fn from ( x : Result < T , Self :: Error > ) -> Self ;
977
+ fn to_result ( self ) -> Result < T , Self :: Error > ;
978
+ }
979
+
980
+ impl < T > MaybeResult < T > for T {
981
+ type Error = !;
982
+
983
+ fn from ( x : Result < T , Self :: Error > ) -> Self {
984
+ let Ok ( x) = x;
985
+ x
986
+ }
987
+ fn to_result ( self ) -> Result < T , Self :: Error > {
988
+ Ok ( self )
989
+ }
990
+ }
991
+
992
+ impl < T , E > MaybeResult < T > for Result < T , E > {
993
+ type Error = E ;
994
+
995
+ fn from ( x : Result < T , Self :: Error > ) -> Self {
996
+ x
997
+ }
998
+ fn to_result ( self ) -> Result < T , Self :: Error > {
999
+ self
1000
+ }
1001
+ }
1002
+
969
1003
#[ derive( Copy , Clone , PartialEq , Eq ) ]
970
1004
pub enum PointerKind {
971
1005
/// Most general case, we know no restrictions to tell LLVM.
@@ -1007,10 +1041,14 @@ impl<'a, Ty> TyLayout<'a, Ty> {
1007
1041
where Ty : TyLayoutMethods < ' a , C > , C : LayoutOf < Ty = Ty > {
1008
1042
Ty :: for_variant ( self , cx, variant_index)
1009
1043
}
1044
+
1045
+ /// Callers might want to use `C: LayoutOf<Ty=Ty, TyLayout: MaybeResult<Self>>`
1046
+ /// to allow recursion (see `might_permit_zero_init` below for an example).
1010
1047
pub fn field < C > ( self , cx : & C , i : usize ) -> C :: TyLayout
1011
1048
where Ty : TyLayoutMethods < ' a , C > , C : LayoutOf < Ty = Ty > {
1012
1049
Ty :: field ( self , cx, i)
1013
1050
}
1051
+
1014
1052
pub fn pointee_info_at < C > ( self , cx : & C , offset : Size ) -> Option < PointeeInfo >
1015
1053
where Ty : TyLayoutMethods < ' a , C > , C : LayoutOf < Ty = Ty > {
1016
1054
Ty :: pointee_info_at ( self , cx, offset)
@@ -1033,4 +1071,89 @@ impl<'a, Ty> TyLayout<'a, Ty> {
1033
1071
Abi :: Aggregate { sized } => sized && self . size . bytes ( ) == 0
1034
1072
}
1035
1073
}
1074
+
1075
+ /// Determines if this type permits "raw" initialization by just transmuting some
1076
+ /// memory into an instance of `T`.
1077
+ /// `zero` indicates if the memory is zero-initialized, or alternatively
1078
+ /// left entirely uninitialized.
1079
+ /// This is conservative: in doubt, it will answer `true`.
1080
+ pub fn might_permit_raw_init < C , E > (
1081
+ self ,
1082
+ cx : & C ,
1083
+ zero : bool ,
1084
+ ) -> Result < bool , E >
1085
+ where
1086
+ Self : Copy ,
1087
+ Ty : TyLayoutMethods < ' a , C > ,
1088
+ C : LayoutOf < Ty = Ty , TyLayout : MaybeResult < Self , Error = E > > + HasDataLayout
1089
+ {
1090
+ let scalar_allows_raw_init = move |s : & Scalar | -> bool {
1091
+ if zero {
1092
+ let range = & s. valid_range ;
1093
+ // The range must contain 0.
1094
+ range. contains ( & 0 ) ||
1095
+ ( * range. start ( ) > * range. end ( ) ) // wrap-around allows 0
1096
+ } else {
1097
+ // The range must include all values. `valid_range_exclusive` handles
1098
+ // the wrap-around using target arithmetic; with wrap-around then the full
1099
+ // range is one where `start == end`.
1100
+ let range = s. valid_range_exclusive ( cx) ;
1101
+ range. start == range. end
1102
+ }
1103
+ } ;
1104
+
1105
+ // Abi is the most informative here.
1106
+ let res = match & self . abi {
1107
+ Abi :: Uninhabited => false , // definitely UB
1108
+ Abi :: Scalar ( s) => scalar_allows_raw_init ( s) ,
1109
+ Abi :: ScalarPair ( s1, s2) =>
1110
+ scalar_allows_raw_init ( s1) && scalar_allows_raw_init ( s2) ,
1111
+ Abi :: Vector { element : s, count } =>
1112
+ * count == 0 || scalar_allows_raw_init ( s) ,
1113
+ Abi :: Aggregate { .. } => {
1114
+ match self . variants {
1115
+ Variants :: Multiple { .. } =>
1116
+ if zero {
1117
+ // FIXME(#66151):
1118
+ // could we identify the variant with discriminant 0, check that?
1119
+ true
1120
+ } else {
1121
+ // FIXME(#66151): This needs to have some sort of discriminant,
1122
+ // which cannot be undef. But for now we are conservative.
1123
+ true
1124
+ } ,
1125
+ Variants :: Single { .. } => {
1126
+ // For aggregates, recurse.
1127
+ match self . fields {
1128
+ FieldPlacement :: Union ( ..) => true , // An all-0 unit is fine.
1129
+ FieldPlacement :: Array { .. } =>
1130
+ // FIXME(#66151): The widely use smallvec 0.6 creates uninit arrays
1131
+ // with any element type, so let us not (yet) complain about that.
1132
+ /* count == 0 ||
1133
+ self.field(cx, 0).to_result()?.might_permit_raw_init(cx, zero)? */
1134
+ true ,
1135
+ FieldPlacement :: Arbitrary { .. } => {
1136
+ // FIXME(#66151) cargo depends on sized-chunks 0.3.0 which
1137
+ // has some illegal zero-initialization, so let us not (yet)
1138
+ // complain about aggregates either.
1139
+ /* let mut res = true;
1140
+ // Check that all fields accept zero-init.
1141
+ for idx in 0..offsets.len() {
1142
+ let field = self.field(cx, idx).to_result()?;
1143
+ if !field.might_permit_raw_init(cx, zero)? {
1144
+ res = false;
1145
+ break;
1146
+ }
1147
+ }
1148
+ res */
1149
+ true
1150
+ }
1151
+ }
1152
+ }
1153
+ }
1154
+ }
1155
+ } ;
1156
+ trace ! ( "might_permit_raw_init({:?}, zero={}) = {}" , self . details, zero, res) ;
1157
+ Ok ( res)
1158
+ }
1036
1159
}
0 commit comments