1use std::fmt;
2use std::ops::Deref;
3
4use rustc_abi::{FIRST_VARIANT, VariantIdx};
5use rustc_data_structures::intern::Interned;
6use rustc_hir::def::Namespace;
7use rustc_macros::{
8 HashStable, Lift, TyDecodable, TyEncodable, TypeFoldable, TypeVisitable, extension,
9};
10
11use super::ScalarInt;
12use crate::mir::interpret::{ErrorHandled, Scalar};
13use crate::ty::print::{FmtPrinter, PrettyPrinter};
14use crate::ty::{self, Ty, TyCtxt, ValTreeKind};
15
16impl<'tcx> ValTreeKindExt<'tcx> for ty::ValTreeKind<TyCtxt<'tcx>> {
fn try_to_scalar(&self) -> Option<Scalar> {
self.try_to_leaf().map(Scalar::Int)
}
}#[extension(pub trait ValTreeKindExt<'tcx>)]
17impl<'tcx> ty::ValTreeKind<TyCtxt<'tcx>> {
18 fn try_to_scalar(&self) -> Option<Scalar> {
19 self.try_to_leaf().map(Scalar::Int)
20 }
21}
22
23#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for ValTree<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for ValTree<'tcx> {
#[inline]
fn clone(&self) -> ValTree<'tcx> {
let _:
::core::clone::AssertParamIsClone<Interned<'tcx,
ty::ValTreeKind<TyCtxt<'tcx>>>>;
*self
}
}Clone, #[automatically_derived]
impl<'tcx> ::core::hash::Hash for ValTree<'tcx> {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
::core::hash::Hash::hash(&self.0, state)
}
}Hash, #[automatically_derived]
impl<'tcx> ::core::cmp::Eq for ValTree<'tcx> {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_receiver_is_total_eq(&self) {
let _:
::core::cmp::AssertParamIsEq<Interned<'tcx,
ty::ValTreeKind<TyCtxt<'tcx>>>>;
}
}Eq, #[automatically_derived]
impl<'tcx> ::core::cmp::PartialEq for ValTree<'tcx> {
#[inline]
fn eq(&self, other: &ValTree<'tcx>) -> bool { self.0 == other.0 }
}PartialEq)]
29#[derive(const _: () =
{
impl<'tcx, '__ctx>
::rustc_data_structures::stable_hasher::HashStable<::rustc_query_system::ich::StableHashingContext<'__ctx>>
for ValTree<'tcx> {
#[inline]
fn hash_stable(&self,
__hcx:
&mut ::rustc_query_system::ich::StableHashingContext<'__ctx>,
__hasher:
&mut ::rustc_data_structures::stable_hasher::StableHasher) {
match *self {
ValTree(ref __binding_0) => {
{ __binding_0.hash_stable(__hcx, __hasher); }
}
}
}
}
};HashStable)]
30pub struct ValTree<'tcx>(pub(crate) Interned<'tcx, ty::ValTreeKind<TyCtxt<'tcx>>>);
33
34impl<'tcx> rustc_type_ir::inherent::ValTree<TyCtxt<'tcx>> for ValTree<'tcx> {
35 fn kind(&self) -> &ty::ValTreeKind<TyCtxt<'tcx>> {
36 &self
37 }
38}
39
40impl<'tcx> ValTree<'tcx> {
41 pub fn zst(tcx: TyCtxt<'tcx>) -> Self {
43 tcx.consts.valtree_zst
44 }
45
46 pub fn is_zst(self) -> bool {
47 #[allow(non_exhaustive_omitted_patterns)] match *self {
ty::ValTreeKind::Branch(box []) => true,
_ => false,
}matches!(*self, ty::ValTreeKind::Branch(box []))
48 }
49
50 pub fn from_raw_bytes(tcx: TyCtxt<'tcx>, bytes: &[u8]) -> Self {
51 let branches = bytes.iter().map(|&b| {
52 ty::Const::new_value(tcx, Self::from_scalar_int(tcx, b.into()), tcx.types.u8)
53 });
54 Self::from_branches(tcx, branches)
55 }
56
57 pub fn from_branches(
58 tcx: TyCtxt<'tcx>,
59 branches: impl IntoIterator<Item = ty::Const<'tcx>>,
60 ) -> Self {
61 tcx.intern_valtree(ty::ValTreeKind::Branch(branches.into_iter().collect()))
62 }
63
64 pub fn from_scalar_int(tcx: TyCtxt<'tcx>, i: ScalarInt) -> Self {
65 tcx.intern_valtree(ty::ValTreeKind::Leaf(i))
66 }
67}
68
69impl<'tcx> Deref for ValTree<'tcx> {
70 type Target = &'tcx ty::ValTreeKind<TyCtxt<'tcx>>;
71
72 #[inline]
73 fn deref(&self) -> &&'tcx ty::ValTreeKind<TyCtxt<'tcx>> {
74 &self.0.0
75 }
76}
77
78impl fmt::Debug for ValTree<'_> {
79 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
80 (**self).fmt(f)
81 }
82}
83
84pub type ConstToValTreeResult<'tcx> = Result<Result<ValTree<'tcx>, Ty<'tcx>>, ErrorHandled>;
89
90#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for Value<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for Value<'tcx> {
#[inline]
fn clone(&self) -> Value<'tcx> {
let _: ::core::clone::AssertParamIsClone<Ty<'tcx>>;
let _: ::core::clone::AssertParamIsClone<ValTree<'tcx>>;
*self
}
}Clone, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for Value<'tcx> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_struct_field2_finish(f, "Value", "ty",
&self.ty, "valtree", &&self.valtree)
}
}Debug, #[automatically_derived]
impl<'tcx> ::core::hash::Hash for Value<'tcx> {
#[inline]
fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
::core::hash::Hash::hash(&self.ty, state);
::core::hash::Hash::hash(&self.valtree, state)
}
}Hash, #[automatically_derived]
impl<'tcx> ::core::cmp::Eq for Value<'tcx> {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_receiver_is_total_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<Ty<'tcx>>;
let _: ::core::cmp::AssertParamIsEq<ValTree<'tcx>>;
}
}Eq, #[automatically_derived]
impl<'tcx> ::core::cmp::PartialEq for Value<'tcx> {
#[inline]
fn eq(&self, other: &Value<'tcx>) -> bool {
self.ty == other.ty && self.valtree == other.valtree
}
}PartialEq)]
96#[derive(const _: () =
{
impl<'tcx, '__ctx>
::rustc_data_structures::stable_hasher::HashStable<::rustc_query_system::ich::StableHashingContext<'__ctx>>
for Value<'tcx> {
#[inline]
fn hash_stable(&self,
__hcx:
&mut ::rustc_query_system::ich::StableHashingContext<'__ctx>,
__hasher:
&mut ::rustc_data_structures::stable_hasher::StableHasher) {
match *self {
Value { ty: ref __binding_0, valtree: ref __binding_1 } => {
{ __binding_0.hash_stable(__hcx, __hasher); }
{ __binding_1.hash_stable(__hcx, __hasher); }
}
}
}
}
};HashStable, const _: () =
{
impl<'tcx, __E: ::rustc_middle::ty::codec::TyEncoder<'tcx>>
::rustc_serialize::Encodable<__E> for Value<'tcx> {
fn encode(&self, __encoder: &mut __E) {
match *self {
Value { ty: ref __binding_0, valtree: ref __binding_1 } => {
::rustc_serialize::Encodable::<__E>::encode(__binding_0,
__encoder);
::rustc_serialize::Encodable::<__E>::encode(__binding_1,
__encoder);
}
}
}
}
};TyEncodable, const _: () =
{
impl<'tcx, __D: ::rustc_middle::ty::codec::TyDecoder<'tcx>>
::rustc_serialize::Decodable<__D> for Value<'tcx> {
fn decode(__decoder: &mut __D) -> Self {
Value {
ty: ::rustc_serialize::Decodable::decode(__decoder),
valtree: ::rustc_serialize::Decodable::decode(__decoder),
}
}
}
};TyDecodable, const _: () =
{
impl<'tcx>
::rustc_middle::ty::TypeFoldable<::rustc_middle::ty::TyCtxt<'tcx>>
for Value<'tcx> {
fn try_fold_with<__F: ::rustc_middle::ty::FallibleTypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
__folder: &mut __F) -> Result<Self, __F::Error> {
Ok(match self {
Value { ty: __binding_0, valtree: __binding_1 } => {
Value {
ty: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_0,
__folder)?,
valtree: ::rustc_middle::ty::TypeFoldable::try_fold_with(__binding_1,
__folder)?,
}
}
})
}
fn fold_with<__F: ::rustc_middle::ty::TypeFolder<::rustc_middle::ty::TyCtxt<'tcx>>>(self,
__folder: &mut __F) -> Self {
match self {
Value { ty: __binding_0, valtree: __binding_1 } => {
Value {
ty: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_0,
__folder),
valtree: ::rustc_middle::ty::TypeFoldable::fold_with(__binding_1,
__folder),
}
}
}
}
}
};TypeFoldable, const _: () =
{
impl<'tcx>
::rustc_middle::ty::TypeVisitable<::rustc_middle::ty::TyCtxt<'tcx>>
for Value<'tcx> {
fn visit_with<__V: ::rustc_middle::ty::TypeVisitor<::rustc_middle::ty::TyCtxt<'tcx>>>(&self,
__visitor: &mut __V) -> __V::Result {
match *self {
Value { ty: ref __binding_0, valtree: ref __binding_1 } => {
{
match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_0,
__visitor)) {
::core::ops::ControlFlow::Continue(()) => {}
::core::ops::ControlFlow::Break(r) => {
return ::rustc_middle::ty::VisitorResult::from_residual(r);
}
}
}
{
match ::rustc_middle::ty::VisitorResult::branch(::rustc_middle::ty::TypeVisitable::visit_with(__binding_1,
__visitor)) {
::core::ops::ControlFlow::Continue(()) => {}
::core::ops::ControlFlow::Break(r) => {
return ::rustc_middle::ty::VisitorResult::from_residual(r);
}
}
}
}
}
<__V::Result as ::rustc_middle::ty::VisitorResult>::output()
}
}
};TypeVisitable, const _: () =
{
impl<'tcx, '__lifted>
::rustc_middle::ty::Lift<::rustc_middle::ty::TyCtxt<'__lifted>>
for Value<'tcx> {
type Lifted = Value<'__lifted>;
fn lift_to_interner(self,
__tcx: ::rustc_middle::ty::TyCtxt<'__lifted>)
-> Option<Value<'__lifted>> {
Some(match self {
Value { ty: __binding_0, valtree: __binding_1 } => {
Value {
ty: __tcx.lift(__binding_0)?,
valtree: __tcx.lift(__binding_1)?,
}
}
})
}
}
};Lift)]
97pub struct Value<'tcx> {
98 pub ty: Ty<'tcx>,
99 pub valtree: ValTree<'tcx>,
100}
101
102impl<'tcx> Value<'tcx> {
103 #[inline]
108 pub fn try_to_bits(self, tcx: TyCtxt<'tcx>, typing_env: ty::TypingEnv<'tcx>) -> Option<u128> {
109 let (ty::Bool | ty::Char | ty::Uint(_) | ty::Int(_) | ty::Float(_)) = self.ty.kind() else {
110 return None;
111 };
112 let scalar = self.try_to_leaf()?;
113 let input = typing_env.with_post_analysis_normalized(tcx).as_query_input(self.ty);
114 let size = tcx.layout_of(input).ok()?.size;
115 Some(scalar.to_bits(size))
116 }
117
118 pub fn try_to_bool(self) -> Option<bool> {
119 if !self.ty.is_bool() {
120 return None;
121 }
122 self.try_to_leaf()?.try_to_bool().ok()
123 }
124
125 pub fn try_to_target_usize(self, tcx: TyCtxt<'tcx>) -> Option<u64> {
126 if !self.ty.is_usize() {
127 return None;
128 }
129 self.try_to_leaf().map(|s| s.to_target_usize(tcx))
130 }
131
132 pub fn try_to_raw_bytes(self, tcx: TyCtxt<'tcx>) -> Option<&'tcx [u8]> {
135 match self.ty.kind() {
136 ty::Ref(_, inner_ty, _) => match inner_ty.kind() {
137 ty::Str => {}
139 ty::Slice(slice_ty) if *slice_ty == tcx.types.u8 => {}
141 _ => return None,
143 },
144 ty::Array(array_ty, _) if *array_ty == tcx.types.u8 => {}
146 _ => return None,
148 }
149
150 let iterator = self.to_branch().into_iter().map(|ct| Some(ct.try_to_leaf()?.to_u8()));
152 let bytes: Vec<u8> = iterator.collect::<Option<Vec<u8>>>()?;
154
155 Some(tcx.arena.alloc_from_iter(bytes))
156 }
157
158 #[inline]
161 pub fn to_leaf(self) -> ScalarInt {
162 match &**self.valtree {
163 ValTreeKind::Leaf(s) => *s,
164 ValTreeKind::Branch(..) => crate::util::bug::bug_fmt(format_args!("expected leaf, got {0:?}", self))bug!("expected leaf, got {:?}", self),
165 }
166 }
167
168 #[inline]
171 pub fn to_branch(self) -> &'tcx [ty::Const<'tcx>] {
172 match &**self.valtree {
173 ValTreeKind::Branch(branch) => &**branch,
174 ValTreeKind::Leaf(..) => crate::util::bug::bug_fmt(format_args!("expected branch, got {0:?}", self))bug!("expected branch, got {:?}", self),
175 }
176 }
177
178 pub fn try_to_leaf(self) -> Option<ScalarInt> {
180 match &**self.valtree {
181 ValTreeKind::Leaf(s) => Some(*s),
182 ValTreeKind::Branch(_) => None,
183 }
184 }
185
186 pub fn try_to_scalar(&self) -> Option<Scalar> {
188 self.try_to_leaf().map(Scalar::Int)
189 }
190
191 pub fn try_to_branch(self) -> Option<&'tcx [ty::Const<'tcx>]> {
193 match &**self.valtree {
194 ValTreeKind::Branch(branch) => Some(&**branch),
195 ValTreeKind::Leaf(_) => None,
196 }
197 }
198
199 pub fn destructure_adt_const(&self) -> ty::DestructuredAdtConst<'tcx> {
201 let fields = self.to_branch();
202
203 let (variant, fields) = match self.ty.kind() {
204 ty::Adt(def, _) if def.variants().is_empty() => {
205 crate::util::bug::bug_fmt(format_args!("unreachable"))bug!("unreachable")
206 }
207 ty::Adt(def, _) if def.is_enum() => {
208 let (head, rest) = fields.split_first().unwrap();
209 (VariantIdx::from_u32(head.to_leaf().to_u32()), rest)
210 }
211 ty::Adt(_, _) => (FIRST_VARIANT, fields),
212 _ => crate::util::bug::bug_fmt(format_args!("destructure_adt_const called on non-ADT type: {0:?}",
self.ty))bug!("destructure_adt_const called on non-ADT type: {:?}", self.ty),
213 };
214
215 ty::DestructuredAdtConst { variant, fields }
216 }
217}
218
219impl<'tcx> rustc_type_ir::inherent::ValueConst<TyCtxt<'tcx>> for Value<'tcx> {
220 fn ty(self) -> Ty<'tcx> {
221 self.ty
222 }
223
224 fn valtree(self) -> ValTree<'tcx> {
225 self.valtree
226 }
227}
228
229impl<'tcx> fmt::Display for Value<'tcx> {
230 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
231 ty::tls::with(move |tcx| {
232 let cv = tcx.lift(*self).unwrap();
233 let mut p = FmtPrinter::new(tcx, Namespace::ValueNS);
234 p.pretty_print_const_valtree(cv, true)?;
235 f.write_str(&p.into_buffer())
236 })
237 }
238}