Skip to content

Commit ddc2545

Browse files
committed
Check types of statics in MIR typeck
1 parent cd9f5ff commit ddc2545

File tree

5 files changed

+97
-2
lines changed

5 files changed

+97
-2
lines changed

src/librustc_mir/borrow_check/type_check/mod.rs

+16-2
Original file line numberDiff line numberDiff line change
@@ -309,6 +309,7 @@ impl<'a, 'b, 'tcx> Visitor<'tcx> for TypeVerifier<'a, 'b, 'tcx> {
309309
);
310310
}
311311
} else {
312+
let tcx = self.tcx();
312313
if let ty::ConstKind::Unevaluated(def_id, substs, promoted) = constant.literal.val {
313314
if let Some(promoted) = promoted {
314315
let check_err = |verifier: &mut TypeVerifier<'a, 'b, 'tcx>,
@@ -358,10 +359,23 @@ impl<'a, 'b, 'tcx> Visitor<'tcx> for TypeVerifier<'a, 'b, 'tcx> {
358359
);
359360
}
360361
}
362+
} else if let Some(static_def_id) = constant.check_static_ptr(tcx) {
363+
let unnormalized_ty = tcx.type_of(static_def_id);
364+
let locations = location.to_locations();
365+
let normalized_ty = self.cx.normalize(unnormalized_ty, locations);
366+
let literal_ty = constant.literal.ty.builtin_deref(true).unwrap().ty;
367+
368+
if let Err(terr) = self.cx.eq_types(
369+
normalized_ty,
370+
literal_ty,
371+
locations,
372+
ConstraintCategory::Boring,
373+
) {
374+
span_mirbug!(self, constant, "bad static type {:?} ({:?})", constant, terr);
375+
}
361376
}
362-
if let ty::FnDef(def_id, substs) = constant.literal.ty.kind {
363-
let tcx = self.tcx();
364377

378+
if let ty::FnDef(def_id, substs) = constant.literal.ty.kind {
365379
let instantiated_predicates = tcx.predicates_of(def_id).instantiate(tcx, substs);
366380
self.cx.normalize_and_prove_instantiated_predicates(
367381
instantiated_predicates,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
// Check that borrowck ensures that `static mut` items have the expected type.
2+
3+
static FOO: u8 = 42;
4+
static mut BAR: &'static u8 = &FOO;
5+
static mut BAR_ELIDED: &u8 = &FOO;
6+
7+
fn main() {
8+
unsafe {
9+
println!("{} {}", BAR, BAR_ELIDED);
10+
set_bar();
11+
set_bar_elided();
12+
println!("{} {}", BAR, BAR_ELIDED);
13+
}
14+
}
15+
16+
fn set_bar() {
17+
let n = 42;
18+
unsafe {
19+
BAR = &n;
20+
//~^ ERROR does not live long enough
21+
}
22+
}
23+
24+
fn set_bar_elided() {
25+
let n = 42;
26+
unsafe {
27+
BAR_ELIDED = &n;
28+
//~^ ERROR does not live long enough
29+
}
30+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
error[E0597]: `n` does not live long enough
2+
--> $DIR/issue-69114-static-mut-ty.rs:19:15
3+
|
4+
LL | BAR = &n;
5+
| ------^^
6+
| | |
7+
| | borrowed value does not live long enough
8+
| assignment requires that `n` is borrowed for `'static`
9+
...
10+
LL | }
11+
| - `n` dropped here while still borrowed
12+
13+
error[E0597]: `n` does not live long enough
14+
--> $DIR/issue-69114-static-mut-ty.rs:27:22
15+
|
16+
LL | BAR_ELIDED = &n;
17+
| -------------^^
18+
| | |
19+
| | borrowed value does not live long enough
20+
| assignment requires that `n` is borrowed for `'static`
21+
...
22+
LL | }
23+
| - `n` dropped here while still borrowed
24+
25+
error: aborting due to 2 previous errors
26+
27+
For more information about this error, try `rustc --explain E0597`.
+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
// Check that borrowck ensures that `static` items have the expected type.
2+
3+
static FOO: &'static (dyn Fn(&'static u8) + Send + Sync) = &drop;
4+
5+
fn main() {
6+
let n = 42;
7+
FOO(&n);
8+
//~^ ERROR does not live long enough
9+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
error[E0597]: `n` does not live long enough
2+
--> $DIR/issue-69114-static-ty.rs:7:9
3+
|
4+
LL | FOO(&n);
5+
| ----^^-
6+
| | |
7+
| | borrowed value does not live long enough
8+
| argument requires that `n` is borrowed for `'static`
9+
LL |
10+
LL | }
11+
| - `n` dropped here while still borrowed
12+
13+
error: aborting due to previous error
14+
15+
For more information about this error, try `rustc --explain E0597`.

0 commit comments

Comments
 (0)