Skip to content

Commit bc4f117

Browse files
authored
Rollup merge of #94020 - tmiasko:pp, r=oli-obk
Support pretty printing of invalid constants Make it possible to pretty print invalid constants by introducing a fallible variant of `destructure_const` and falling back to debug formatting when it fails. Closes #93688.
2 parents a5a1ffb + 92d20c4 commit bc4f117

File tree

7 files changed

+117
-29
lines changed

7 files changed

+117
-29
lines changed

compiler/rustc_const_eval/src/const_eval/mod.rs

+18-20
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,8 @@ use rustc_middle::{
1111
use rustc_span::{source_map::DUMMY_SP, symbol::Symbol};
1212

1313
use crate::interpret::{
14-
intern_const_alloc_recursive, ConstValue, InternKind, InterpCx, MPlaceTy, MemPlaceMeta, Scalar,
14+
intern_const_alloc_recursive, ConstValue, InternKind, InterpCx, InterpResult, MPlaceTy,
15+
MemPlaceMeta, Scalar,
1516
};
1617

1718
mod error;
@@ -132,42 +133,39 @@ fn const_to_valtree_inner<'tcx>(
132133
}
133134
}
134135

135-
/// This function uses `unwrap` copiously, because an already validated constant
136-
/// must have valid fields and can thus never fail outside of compiler bugs. However, it is
137-
/// invoked from the pretty printer, where it can receive enums with no variants and e.g.
138-
/// `read_discriminant` needs to be able to handle that.
139-
pub(crate) fn destructure_const<'tcx>(
136+
/// This function should never fail for validated constants. However, it is also invoked from the
137+
/// pretty printer which might attempt to format invalid constants and in that case it might fail.
138+
pub(crate) fn try_destructure_const<'tcx>(
140139
tcx: TyCtxt<'tcx>,
141140
param_env: ty::ParamEnv<'tcx>,
142141
val: ty::Const<'tcx>,
143-
) -> mir::DestructuredConst<'tcx> {
142+
) -> InterpResult<'tcx, mir::DestructuredConst<'tcx>> {
144143
trace!("destructure_const: {:?}", val);
145144
let ecx = mk_eval_cx(tcx, DUMMY_SP, param_env, false);
146-
let op = ecx.const_to_op(val, None).unwrap();
145+
let op = ecx.const_to_op(val, None)?;
147146

148147
// We go to `usize` as we cannot allocate anything bigger anyway.
149148
let (field_count, variant, down) = match val.ty().kind() {
150149
ty::Array(_, len) => (usize::try_from(len.eval_usize(tcx, param_env)).unwrap(), None, op),
151-
ty::Adt(def, _) if def.variants.is_empty() => {
152-
return mir::DestructuredConst { variant: None, fields: &[] };
153-
}
154150
ty::Adt(def, _) => {
155-
let variant = ecx.read_discriminant(&op).unwrap().1;
156-
let down = ecx.operand_downcast(&op, variant).unwrap();
151+
let variant = ecx.read_discriminant(&op)?.1;
152+
let down = ecx.operand_downcast(&op, variant)?;
157153
(def.variants[variant].fields.len(), Some(variant), down)
158154
}
159155
ty::Tuple(substs) => (substs.len(), None, op),
160156
_ => bug!("cannot destructure constant {:?}", val),
161157
};
162158

163-
let fields_iter = (0..field_count).map(|i| {
164-
let field_op = ecx.operand_field(&down, i).unwrap();
165-
let val = op_to_const(&ecx, &field_op);
166-
ty::Const::from_value(tcx, val, field_op.layout.ty)
167-
});
168-
let fields = tcx.arena.alloc_from_iter(fields_iter);
159+
let fields = (0..field_count)
160+
.map(|i| {
161+
let field_op = ecx.operand_field(&down, i)?;
162+
let val = op_to_const(&ecx, &field_op);
163+
Ok(ty::Const::from_value(tcx, val, field_op.layout.ty))
164+
})
165+
.collect::<InterpResult<'tcx, Vec<_>>>()?;
166+
let fields = tcx.arena.alloc_from_iter(fields);
169167

170-
mir::DestructuredConst { variant, fields }
168+
Ok(mir::DestructuredConst { variant, fields })
171169
}
172170

173171
pub(crate) fn deref_const<'tcx>(

compiler/rustc_const_eval/src/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -41,9 +41,9 @@ pub fn provide(providers: &mut Providers) {
4141
providers.eval_to_const_value_raw = const_eval::eval_to_const_value_raw_provider;
4242
providers.eval_to_allocation_raw = const_eval::eval_to_allocation_raw_provider;
4343
providers.const_caller_location = const_eval::const_caller_location;
44-
providers.destructure_const = |tcx, param_env_and_value| {
44+
providers.try_destructure_const = |tcx, param_env_and_value| {
4545
let (param_env, value) = param_env_and_value.into_parts();
46-
const_eval::destructure_const(tcx, param_env, value)
46+
const_eval::try_destructure_const(tcx, param_env, value).ok()
4747
};
4848
providers.const_to_valtree = |tcx, param_env_and_value| {
4949
let (param_env, raw) = param_env_and_value.into_parts();

compiler/rustc_middle/src/mir/interpret/queries.rs

+8
Original file line numberDiff line numberDiff line change
@@ -98,4 +98,12 @@ impl<'tcx> TyCtxt<'tcx> {
9898
let raw_const = self.eval_to_allocation_raw(param_env.and(gid))?;
9999
Ok(self.global_alloc(raw_const.alloc_id).unwrap_memory())
100100
}
101+
102+
/// Destructure a constant ADT or array into its variant index and its field values.
103+
pub fn destructure_const(
104+
self,
105+
param_env_and_val: ty::ParamEnvAnd<'tcx, ty::Const<'tcx>>,
106+
) -> mir::DestructuredConst<'tcx> {
107+
self.try_destructure_const(param_env_and_val).unwrap()
108+
}
101109
}

compiler/rustc_middle/src/query/mod.rs

+5-3
Original file line numberDiff line numberDiff line change
@@ -924,10 +924,12 @@ rustc_queries! {
924924
}
925925

926926
/// Destructure a constant ADT or array into its variant index and its
927-
/// field values.
928-
query destructure_const(
927+
/// field values or return `None` if constant is invalid.
928+
///
929+
/// Use infallible `TyCtxt::destructure_const` when you know that constant is valid.
930+
query try_destructure_const(
929931
key: ty::ParamEnvAnd<'tcx, ty::Const<'tcx>>
930-
) -> mir::DestructuredConst<'tcx> {
932+
) -> Option<mir::DestructuredConst<'tcx>> {
931933
desc { "destructure constant" }
932934
remap_env_constness
933935
}

compiler/rustc_middle/src/ty/print/pretty.rs

+12-4
Original file line numberDiff line numberDiff line change
@@ -1459,10 +1459,18 @@ pub trait PrettyPrinter<'tcx>:
14591459
// FIXME(eddyb) for `--emit=mir`/`-Z dump-mir`, we should provide the
14601460
// correct `ty::ParamEnv` to allow printing *all* constant values.
14611461
(_, ty::Array(..) | ty::Tuple(..) | ty::Adt(..)) if !ty.has_param_types_or_consts() => {
1462-
let contents =
1463-
self.tcx().destructure_const(ty::ParamEnv::reveal_all().and(
1464-
self.tcx().mk_const(ty::ConstS { val: ty::ConstKind::Value(ct), ty }),
1465-
));
1462+
let Some(contents) = self.tcx().try_destructure_const(
1463+
ty::ParamEnv::reveal_all()
1464+
.and(self.tcx().mk_const(ty::ConstS { val: ty::ConstKind::Value(ct), ty })),
1465+
) else {
1466+
// Fall back to debug pretty printing for invalid constants.
1467+
p!(write("{:?}", ct));
1468+
if print_ty {
1469+
p!(": ", print(ty));
1470+
}
1471+
return Ok(self);
1472+
};
1473+
14661474
let fields = contents.fields.iter().copied();
14671475

14681476
match *ty.kind() {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
- // MIR for `main` before ConstProp
2+
+ // MIR for `main` after ConstProp
3+
4+
fn main() -> () {
5+
let mut _0: (); // return place in scope 0 at $DIR/invalid_constant.rs:15:11: 15:11
6+
let _1: std::option::Option<()>; // in scope 0 at $DIR/invalid_constant.rs:16:5: 16:12
7+
let mut _2: std::option::Option<std::option::Option<()>>; // in scope 0 at $DIR/invalid_constant.rs:16:7: 16:11
8+
scope 1 (inlined f) { // at $DIR/invalid_constant.rs:16:5: 16:12
9+
debug x => _2; // in scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
10+
let mut _3: isize; // in scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
11+
let _4: std::option::Option<()>; // in scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
12+
scope 2 {
13+
debug y => _4; // in scope 2 at $DIR/invalid_constant.rs:16:5: 16:12
14+
}
15+
}
16+
17+
bb0: {
18+
discriminant(_2) = 0; // scope 0 at $DIR/invalid_constant.rs:16:7: 16:11
19+
- _3 = discriminant(_2); // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
20+
- switchInt(move _3) -> [0_isize: bb3, otherwise: bb2]; // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
21+
+ _3 = const 0_isize; // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
22+
+ switchInt(const 0_isize) -> [0_isize: bb3, otherwise: bb2]; // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
23+
}
24+
25+
bb1: {
26+
nop; // scope 0 at $DIR/invalid_constant.rs:15:11: 17:2
27+
return; // scope 0 at $DIR/invalid_constant.rs:17:2: 17:2
28+
}
29+
30+
bb2: {
31+
- _4 = ((_2 as Some).0: std::option::Option<()>); // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
32+
- _1 = _4; // scope 2 at $DIR/invalid_constant.rs:16:5: 16:12
33+
+ _4 = const Scalar(0x02): Option::<()>; // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
34+
+ // ty::Const
35+
+ // + ty: std::option::Option<()>
36+
+ // + val: Value(Scalar(0x02))
37+
+ // mir::Constant
38+
+ // + span: $DIR/invalid_constant.rs:16:5: 16:12
39+
+ // + literal: Const { ty: std::option::Option<()>, val: Value(Scalar(0x02)) }
40+
+ _1 = const Scalar(0x02): Option::<()>; // scope 2 at $DIR/invalid_constant.rs:16:5: 16:12
41+
+ // ty::Const
42+
+ // + ty: std::option::Option<()>
43+
+ // + val: Value(Scalar(0x02))
44+
+ // mir::Constant
45+
+ // + span: $DIR/invalid_constant.rs:16:5: 16:12
46+
+ // + literal: Const { ty: std::option::Option<()>, val: Value(Scalar(0x02)) }
47+
goto -> bb1; // scope 0 at $DIR/invalid_constant.rs:10:20: 10:21
48+
}
49+
50+
bb3: {
51+
discriminant(_1) = 0; // scope 1 at $DIR/invalid_constant.rs:16:5: 16:12
52+
goto -> bb1; // scope 0 at $DIR/invalid_constant.rs:9:17: 9:21
53+
}
54+
}
55+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
// Verify that we can pretty print invalid constant introduced
2+
// by constant propagation. Regression test for issue #93688.
3+
//
4+
// compile-flags: -Copt-level=0 -Zinline-mir
5+
6+
#[inline(always)]
7+
pub fn f(x: Option<Option<()>>) -> Option<()> {
8+
match x {
9+
None => None,
10+
Some(y) => y,
11+
}
12+
}
13+
14+
// EMIT_MIR invalid_constant.main.ConstProp.diff
15+
fn main() {
16+
f(None);
17+
}

0 commit comments

Comments
 (0)