Skip to content

Commit ed19532

Browse files
committed
Auto merge of #120960 - RalfJung:static-promoted-cycle, r=oli-obk
fix cycle error when a static and a promoted are mutually recursive This also now allows promoteds everywhere to point to 'extern static', because why not? We still check that constants cannot transitively reach 'extern static' through references. (We allow it through raw pointers.) r? `@oli-obk` Fixes #120949
2 parents b17491c + 5fa69de commit ed19532

File tree

3 files changed

+29
-24
lines changed

3 files changed

+29
-24
lines changed

compiler/rustc_const_eval/src/const_eval/eval_queries.rs

+2-11
Original file line numberDiff line numberDiff line change
@@ -356,22 +356,13 @@ pub fn const_validate_mplace<'mir, 'tcx>(
356356
let mut inner = false;
357357
while let Some((mplace, path)) = ref_tracking.todo.pop() {
358358
let mode = match ecx.tcx.static_mutability(cid.instance.def_id()) {
359-
Some(_) if cid.promoted.is_some() => {
360-
// Promoteds in statics are consts that re allowed to point to statics.
361-
CtfeValidationMode::Const {
362-
allow_immutable_unsafe_cell: false,
363-
allow_extern_static_ptrs: true,
364-
}
365-
}
359+
_ if cid.promoted.is_some() => CtfeValidationMode::Promoted,
366360
Some(mutbl) => CtfeValidationMode::Static { mutbl }, // a `static`
367361
None => {
368362
// In normal `const` (not promoted), the outermost allocation is always only copied,
369363
// so having `UnsafeCell` in there is okay despite them being in immutable memory.
370364
let allow_immutable_unsafe_cell = cid.promoted.is_none() && !inner;
371-
CtfeValidationMode::Const {
372-
allow_immutable_unsafe_cell,
373-
allow_extern_static_ptrs: false,
374-
}
365+
CtfeValidationMode::Const { allow_immutable_unsafe_cell }
375366
}
376367
};
377368
ecx.const_validate_operand(&mplace.into(), path, &mut ref_tracking, mode)?;

compiler/rustc_const_eval/src/interpret/validity.rs

+15-13
Original file line numberDiff line numberDiff line change
@@ -129,17 +129,20 @@ pub enum PathElem {
129129
pub enum CtfeValidationMode {
130130
/// Validation of a `static`
131131
Static { mutbl: Mutability },
132-
/// Validation of a `const` (including promoteds).
132+
/// Validation of a promoted.
133+
Promoted,
134+
/// Validation of a `const`.
133135
/// `allow_immutable_unsafe_cell` says whether we allow `UnsafeCell` in immutable memory (which is the
134136
/// case for the top-level allocation of a `const`, where this is fine because the allocation will be
135137
/// copied at each use site).
136-
Const { allow_immutable_unsafe_cell: bool, allow_extern_static_ptrs: bool },
138+
Const { allow_immutable_unsafe_cell: bool },
137139
}
138140

139141
impl CtfeValidationMode {
140142
fn allow_immutable_unsafe_cell(self) -> bool {
141143
match self {
142144
CtfeValidationMode::Static { .. } => false,
145+
CtfeValidationMode::Promoted { .. } => false,
143146
CtfeValidationMode::Const { allow_immutable_unsafe_cell, .. } => {
144147
allow_immutable_unsafe_cell
145148
}
@@ -149,6 +152,7 @@ impl CtfeValidationMode {
149152
fn may_contain_mutable_ref(self) -> bool {
150153
match self {
151154
CtfeValidationMode::Static { mutbl } => mutbl == Mutability::Mut,
155+
CtfeValidationMode::Promoted { .. } => false,
152156
CtfeValidationMode::Const { .. } => false,
153157
}
154158
}
@@ -476,34 +480,32 @@ impl<'rt, 'mir, 'tcx: 'mir, M: Machine<'mir, 'tcx>> ValidityVisitor<'rt, 'mir, '
476480
throw_validation_failure!(self.path, MutableRefToImmutable);
477481
}
478482
}
483+
// Mode-specific checks
479484
match self.ctfe_mode {
480-
Some(CtfeValidationMode::Static { .. }) => {
485+
Some(
486+
CtfeValidationMode::Static { .. }
487+
| CtfeValidationMode::Promoted { .. },
488+
) => {
481489
// We skip recursively checking other statics. These statics must be sound by
482490
// themselves, and the only way to get broken statics here is by using
483491
// unsafe code.
484492
// The reasons we don't check other statics is twofold. For one, in all
485493
// sound cases, the static was already validated on its own, and second, we
486494
// trigger cycle errors if we try to compute the value of the other static
487-
// and that static refers back to us.
495+
// and that static refers back to us (potentially through a promoted).
488496
// This could miss some UB, but that's fine.
489497
return Ok(());
490498
}
491-
Some(CtfeValidationMode::Const {
492-
allow_extern_static_ptrs, ..
493-
}) => {
499+
Some(CtfeValidationMode::Const { .. }) => {
494500
// For consts on the other hand we have to recursively check;
495501
// pattern matching assumes a valid value. However we better make
496502
// sure this is not mutable.
497503
if is_mut {
498504
throw_validation_failure!(self.path, ConstRefToMutable);
499505
}
506+
// We can't recursively validate `extern static`, so we better reject them.
500507
if self.ecx.tcx.is_foreign_item(did) {
501-
if !allow_extern_static_ptrs {
502-
throw_validation_failure!(self.path, ConstRefToExtern);
503-
} else {
504-
// We can't validate this...
505-
return Ok(());
506-
}
508+
throw_validation_failure!(self.path, ConstRefToExtern);
507509
}
508510
}
509511
None => {}
+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// check-pass
2+
3+
struct Value {
4+
values: &'static [&'static Value],
5+
}
6+
7+
// This `static` recursively points to itself through a promoted (the slice).
8+
static VALUE: Value = Value {
9+
values: &[&VALUE],
10+
};
11+
12+
fn main() {}

0 commit comments

Comments
 (0)