Improve pretty printing of valtrees for references
This commit is contained in:
parent
94e93749ab
commit
cd88bb332c
9 changed files with 144 additions and 102 deletions
|
@ -5,7 +5,6 @@ use rustc_middle::mir;
|
||||||
use rustc_middle::mir::interpret::{EvalToValTreeResult, GlobalId};
|
use rustc_middle::mir::interpret::{EvalToValTreeResult, GlobalId};
|
||||||
use rustc_middle::ty::{self, TyCtxt};
|
use rustc_middle::ty::{self, TyCtxt};
|
||||||
use rustc_span::{source_map::DUMMY_SP, symbol::Symbol};
|
use rustc_span::{source_map::DUMMY_SP, symbol::Symbol};
|
||||||
use rustc_target::abi::VariantIdx;
|
|
||||||
|
|
||||||
use crate::interpret::{
|
use crate::interpret::{
|
||||||
intern_const_alloc_recursive, ConstValue, InternKind, InterpCx, InterpResult, MemPlaceMeta,
|
intern_const_alloc_recursive, ConstValue, InternKind, InterpCx, InterpResult, MemPlaceMeta,
|
||||||
|
@ -91,83 +90,6 @@ pub(crate) fn eval_to_valtree<'tcx>(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Tries to destructure constants of type Array or Adt into the constants
|
|
||||||
/// of its fields.
|
|
||||||
pub(crate) fn try_destructure_const<'tcx>(
|
|
||||||
tcx: TyCtxt<'tcx>,
|
|
||||||
const_: ty::Const<'tcx>,
|
|
||||||
) -> Option<ty::DestructuredConst<'tcx>> {
|
|
||||||
if let ty::ConstKind::Value(valtree) = const_.kind() {
|
|
||||||
let branches = match valtree {
|
|
||||||
ty::ValTree::Branch(b) => b,
|
|
||||||
_ => return None,
|
|
||||||
};
|
|
||||||
|
|
||||||
let (fields, variant) = match const_.ty().kind() {
|
|
||||||
ty::Array(inner_ty, _) | ty::Slice(inner_ty) => {
|
|
||||||
// construct the consts for the elements of the array/slice
|
|
||||||
let field_consts = branches
|
|
||||||
.iter()
|
|
||||||
.map(|b| {
|
|
||||||
tcx.mk_const(ty::ConstS { kind: ty::ConstKind::Value(*b), ty: *inner_ty })
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
debug!(?field_consts);
|
|
||||||
|
|
||||||
(field_consts, None)
|
|
||||||
}
|
|
||||||
ty::Adt(def, _) if def.variants().is_empty() => bug!("unreachable"),
|
|
||||||
ty::Adt(def, substs) => {
|
|
||||||
let variant_idx = if def.is_enum() {
|
|
||||||
VariantIdx::from_u32(branches[0].unwrap_leaf().try_to_u32().ok()?)
|
|
||||||
} else {
|
|
||||||
VariantIdx::from_u32(0)
|
|
||||||
};
|
|
||||||
let fields = &def.variant(variant_idx).fields;
|
|
||||||
let mut field_consts = Vec::with_capacity(fields.len());
|
|
||||||
|
|
||||||
// Note: First element inValTree corresponds to variant of enum
|
|
||||||
let mut valtree_idx = if def.is_enum() { 1 } else { 0 };
|
|
||||||
for field in fields {
|
|
||||||
let field_ty = field.ty(tcx, substs);
|
|
||||||
let field_valtree = branches[valtree_idx]; // first element of branches is variant
|
|
||||||
let field_const = tcx.mk_const(ty::ConstS {
|
|
||||||
kind: ty::ConstKind::Value(field_valtree),
|
|
||||||
ty: field_ty,
|
|
||||||
});
|
|
||||||
field_consts.push(field_const);
|
|
||||||
valtree_idx += 1;
|
|
||||||
}
|
|
||||||
debug!(?field_consts);
|
|
||||||
|
|
||||||
(field_consts, Some(variant_idx))
|
|
||||||
}
|
|
||||||
ty::Tuple(elem_tys) => {
|
|
||||||
let fields = elem_tys
|
|
||||||
.iter()
|
|
||||||
.enumerate()
|
|
||||||
.map(|(i, elem_ty)| {
|
|
||||||
let elem_valtree = branches[i];
|
|
||||||
tcx.mk_const(ty::ConstS {
|
|
||||||
kind: ty::ConstKind::Value(elem_valtree),
|
|
||||||
ty: elem_ty,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
(fields, None)
|
|
||||||
}
|
|
||||||
_ => bug!("cannot destructure constant {:?}", const_),
|
|
||||||
};
|
|
||||||
|
|
||||||
let fields = tcx.arena.alloc_from_iter(fields.into_iter());
|
|
||||||
|
|
||||||
Some(ty::DestructuredConst { variant, fields })
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[instrument(skip(tcx), level = "debug")]
|
#[instrument(skip(tcx), level = "debug")]
|
||||||
pub(crate) fn try_destructure_mir_constant<'tcx>(
|
pub(crate) fn try_destructure_mir_constant<'tcx>(
|
||||||
tcx: TyCtxt<'tcx>,
|
tcx: TyCtxt<'tcx>,
|
||||||
|
|
|
@ -42,7 +42,6 @@ pub fn provide(providers: &mut Providers) {
|
||||||
providers.eval_to_const_value_raw = const_eval::eval_to_const_value_raw_provider;
|
providers.eval_to_const_value_raw = const_eval::eval_to_const_value_raw_provider;
|
||||||
providers.eval_to_allocation_raw = const_eval::eval_to_allocation_raw_provider;
|
providers.eval_to_allocation_raw = const_eval::eval_to_allocation_raw_provider;
|
||||||
providers.const_caller_location = const_eval::const_caller_location;
|
providers.const_caller_location = const_eval::const_caller_location;
|
||||||
providers.try_destructure_const = |tcx, val| const_eval::try_destructure_const(tcx, val);
|
|
||||||
providers.eval_to_valtree = |tcx, param_env_and_value| {
|
providers.eval_to_valtree = |tcx, param_env_and_value| {
|
||||||
let (param_env, raw) = param_env_and_value.into_parts();
|
let (param_env, raw) = param_env_and_value.into_parts();
|
||||||
const_eval::eval_to_valtree(tcx, param_env, raw)
|
const_eval::eval_to_valtree(tcx, param_env, raw)
|
||||||
|
|
|
@ -205,14 +205,8 @@ impl<'tcx> TyCtxtEnsure<'tcx> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'tcx> TyCtxt<'tcx> {
|
impl<'tcx> TyCtxt<'tcx> {
|
||||||
/// Destructure a type-level constant ADT or array into its variant index and its field values.
|
|
||||||
/// Panics if the destructuring fails, use `try_destructure_const` for fallible version.
|
|
||||||
pub fn destructure_const(self, const_: ty::Const<'tcx>) -> ty::DestructuredConst<'tcx> {
|
|
||||||
self.try_destructure_const(const_).unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Destructure a mir constant ADT or array into its variant index and its field values.
|
/// Destructure a mir constant ADT or array into its variant index and its field values.
|
||||||
/// Panics if the destructuring fails, use `try_destructure_const` for fallible version.
|
/// Panics if the destructuring fails, use `try_destructure_mir_constant` for fallible version.
|
||||||
pub fn destructure_mir_constant(
|
pub fn destructure_mir_constant(
|
||||||
self,
|
self,
|
||||||
param_env: ty::ParamEnv<'tcx>,
|
param_env: ty::ParamEnv<'tcx>,
|
||||||
|
|
|
@ -978,11 +978,8 @@ rustc_queries! {
|
||||||
desc { "converting type-level constant value to mir constant value"}
|
desc { "converting type-level constant value to mir constant value"}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Destructure a constant ADT or array into its variant index and its
|
/// Destructure a type-level constant ADT or array into its variant index and its field values.
|
||||||
/// field values or return `None` if constant is invalid.
|
query destructure_const(key: ty::Const<'tcx>) -> ty::DestructuredConst<'tcx> {
|
||||||
///
|
|
||||||
/// Use infallible `TyCtxt::destructure_const` when you know that constant is valid.
|
|
||||||
query try_destructure_const(key: ty::Const<'tcx>) -> Option<ty::DestructuredConst<'tcx>> {
|
|
||||||
desc { "destructuring type level constant"}
|
desc { "destructuring type level constant"}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1447,7 +1447,11 @@ pub trait PrettyPrinter<'tcx>:
|
||||||
p!(write("{:?}", String::from_utf8_lossy(bytes)));
|
p!(write("{:?}", String::from_utf8_lossy(bytes)));
|
||||||
return Ok(self);
|
return Ok(self);
|
||||||
}
|
}
|
||||||
_ => {}
|
_ => {
|
||||||
|
p!("&");
|
||||||
|
p!(pretty_print_const_valtree(valtree, *inner_ty, print_ty));
|
||||||
|
return Ok(self);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
(ty::ValTree::Branch(_), ty::Array(t, _)) if *t == u8_type => {
|
(ty::ValTree::Branch(_), ty::Array(t, _)) if *t == u8_type => {
|
||||||
let bytes = valtree.try_to_raw_bytes(self.tcx(), *t).unwrap_or_else(|| {
|
let bytes = valtree.try_to_raw_bytes(self.tcx(), *t).unwrap_or_else(|| {
|
||||||
|
@ -1459,16 +1463,8 @@ pub trait PrettyPrinter<'tcx>:
|
||||||
}
|
}
|
||||||
// Aggregates, printed as array/tuple/struct/variant construction syntax.
|
// Aggregates, printed as array/tuple/struct/variant construction syntax.
|
||||||
(ty::ValTree::Branch(_), ty::Array(..) | ty::Tuple(..) | ty::Adt(..)) => {
|
(ty::ValTree::Branch(_), ty::Array(..) | ty::Tuple(..) | ty::Adt(..)) => {
|
||||||
let Some(contents) = self.tcx().try_destructure_const(
|
let contents =
|
||||||
ty::Const::from_value(self.tcx(), valtree, ty)
|
self.tcx().destructure_const(ty::Const::from_value(self.tcx(), valtree, ty));
|
||||||
) else {
|
|
||||||
// Fall back to debug pretty printing for invalid constants.
|
|
||||||
p!(write("{:?}", valtree));
|
|
||||||
if print_ty {
|
|
||||||
p!(": ", print(ty));
|
|
||||||
}
|
|
||||||
return Ok(self);
|
|
||||||
};
|
|
||||||
let fields = contents.fields.iter().copied();
|
let fields = contents.fields.iter().copied();
|
||||||
match *ty.kind() {
|
match *ty.kind() {
|
||||||
ty::Array(..) => {
|
ty::Array(..) => {
|
||||||
|
|
84
compiler/rustc_ty_utils/src/consts.rs
Normal file
84
compiler/rustc_ty_utils/src/consts.rs
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
use rustc_middle::ty::{self, TyCtxt};
|
||||||
|
use rustc_target::abi::VariantIdx;
|
||||||
|
|
||||||
|
/// Tries to destructure constants of type Array or Adt into the constants
|
||||||
|
/// of its fields.
|
||||||
|
pub(crate) fn destructure_const<'tcx>(
|
||||||
|
tcx: TyCtxt<'tcx>,
|
||||||
|
const_: ty::Const<'tcx>,
|
||||||
|
) -> ty::DestructuredConst<'tcx> {
|
||||||
|
if let ty::ConstKind::Value(valtree) = const_.kind() {
|
||||||
|
let branches = match valtree {
|
||||||
|
ty::ValTree::Branch(b) => b,
|
||||||
|
_ => bug!("cannot destructure constant {:?}", const_),
|
||||||
|
};
|
||||||
|
|
||||||
|
let (fields, variant) = match const_.ty().kind() {
|
||||||
|
ty::Array(inner_ty, _) | ty::Slice(inner_ty) => {
|
||||||
|
// construct the consts for the elements of the array/slice
|
||||||
|
let field_consts = branches
|
||||||
|
.iter()
|
||||||
|
.map(|b| {
|
||||||
|
tcx.mk_const(ty::ConstS { kind: ty::ConstKind::Value(*b), ty: *inner_ty })
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
debug!(?field_consts);
|
||||||
|
|
||||||
|
(field_consts, None)
|
||||||
|
}
|
||||||
|
ty::Adt(def, _) if def.variants().is_empty() => bug!("unreachable"),
|
||||||
|
ty::Adt(def, substs) => {
|
||||||
|
let variant_idx = if def.is_enum() {
|
||||||
|
VariantIdx::from_u32(branches[0].unwrap_leaf().try_to_u32().unwrap())
|
||||||
|
} else {
|
||||||
|
VariantIdx::from_u32(0)
|
||||||
|
};
|
||||||
|
let fields = &def.variant(variant_idx).fields;
|
||||||
|
let mut field_consts = Vec::with_capacity(fields.len());
|
||||||
|
|
||||||
|
// Note: First element inValTree corresponds to variant of enum
|
||||||
|
let mut valtree_idx = if def.is_enum() { 1 } else { 0 };
|
||||||
|
for field in fields {
|
||||||
|
let field_ty = field.ty(tcx, substs);
|
||||||
|
let field_valtree = branches[valtree_idx]; // first element of branches is variant
|
||||||
|
let field_const = tcx.mk_const(ty::ConstS {
|
||||||
|
kind: ty::ConstKind::Value(field_valtree),
|
||||||
|
ty: field_ty,
|
||||||
|
});
|
||||||
|
field_consts.push(field_const);
|
||||||
|
valtree_idx += 1;
|
||||||
|
}
|
||||||
|
debug!(?field_consts);
|
||||||
|
|
||||||
|
(field_consts, Some(variant_idx))
|
||||||
|
}
|
||||||
|
ty::Tuple(elem_tys) => {
|
||||||
|
let fields = elem_tys
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(i, elem_ty)| {
|
||||||
|
let elem_valtree = branches[i];
|
||||||
|
tcx.mk_const(ty::ConstS {
|
||||||
|
kind: ty::ConstKind::Value(elem_valtree),
|
||||||
|
ty: elem_ty,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
(fields, None)
|
||||||
|
}
|
||||||
|
_ => bug!("cannot destructure constant {:?}", const_),
|
||||||
|
};
|
||||||
|
|
||||||
|
let fields = tcx.arena.alloc_from_iter(fields.into_iter());
|
||||||
|
|
||||||
|
ty::DestructuredConst { variant, fields }
|
||||||
|
} else {
|
||||||
|
bug!("cannot destructure constant {:?}", const_)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn provide(providers: &mut ty::query::Providers) {
|
||||||
|
*providers =
|
||||||
|
ty::query::Providers { destructure_const, ..*providers };
|
||||||
|
}
|
|
@ -18,6 +18,7 @@ use rustc_middle::ty::query::Providers;
|
||||||
|
|
||||||
mod assoc;
|
mod assoc;
|
||||||
mod common_traits;
|
mod common_traits;
|
||||||
|
pub mod consts;
|
||||||
pub mod instance;
|
pub mod instance;
|
||||||
mod needs_drop;
|
mod needs_drop;
|
||||||
pub mod representability;
|
pub mod representability;
|
||||||
|
@ -26,6 +27,7 @@ mod ty;
|
||||||
pub fn provide(providers: &mut Providers) {
|
pub fn provide(providers: &mut Providers) {
|
||||||
assoc::provide(providers);
|
assoc::provide(providers);
|
||||||
common_traits::provide(providers);
|
common_traits::provide(providers);
|
||||||
|
consts::provide(providers);
|
||||||
needs_drop::provide(providers);
|
needs_drop::provide(providers);
|
||||||
ty::provide(providers);
|
ty::provide(providers);
|
||||||
instance::provide(providers);
|
instance::provide(providers);
|
||||||
|
|
28
src/test/ui/const-generics/issue-66451.rs
Normal file
28
src/test/ui/const-generics/issue-66451.rs
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
#![feature(adt_const_params)]
|
||||||
|
#![allow(incomplete_features)]
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Eq)]
|
||||||
|
struct Foo {
|
||||||
|
value: i32,
|
||||||
|
nested: &'static Bar<i32>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Eq)]
|
||||||
|
struct Bar<T>(T);
|
||||||
|
|
||||||
|
struct Test<const F: Foo>;
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let x: Test<{
|
||||||
|
Foo {
|
||||||
|
value: 3,
|
||||||
|
nested: &Bar(4),
|
||||||
|
}
|
||||||
|
}> = Test;
|
||||||
|
let y: Test<{
|
||||||
|
Foo {
|
||||||
|
value: 3,
|
||||||
|
nested: &Bar(5),
|
||||||
|
}
|
||||||
|
}> = x; //~ ERROR mismatched types
|
||||||
|
}
|
20
src/test/ui/const-generics/issue-66451.stderr
Normal file
20
src/test/ui/const-generics/issue-66451.stderr
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
error[E0308]: mismatched types
|
||||||
|
--> $DIR/issue-66451.rs:27:10
|
||||||
|
|
|
||||||
|
LL | let y: Test<{
|
||||||
|
| ____________-
|
||||||
|
LL | | Foo {
|
||||||
|
LL | | value: 3,
|
||||||
|
LL | | nested: &Bar(5),
|
||||||
|
LL | | }
|
||||||
|
LL | | }> = x;
|
||||||
|
| | - ^ expected `Foo { value: 3_i32, nested: &Bar::<i32>(5_i32) }`, found `Foo { value: 3_i32, nested: &Bar::<i32>(4_i32) }`
|
||||||
|
| |______|
|
||||||
|
| expected due to this
|
||||||
|
|
|
||||||
|
= note: expected struct `Test<Foo { value: 3_i32, nested: &Bar::<i32>(5_i32) }>`
|
||||||
|
found struct `Test<Foo { value: 3_i32, nested: &Bar::<i32>(4_i32) }>`
|
||||||
|
|
||||||
|
error: aborting due to previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0308`.
|
Loading…
Add table
Add a link
Reference in a new issue