Skip to content

Commit 121499a

Browse files
committed
Don't promote to 'static the result of dereferences.
1 parent 6cf081c commit 121499a

File tree

2 files changed

+46
-15
lines changed

2 files changed

+46
-15
lines changed

src/librustc_mir/transform/qualify_consts.rs

+34-2
Original file line numberDiff line numberDiff line change
@@ -491,9 +491,10 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
491491
this.add(Qualif::STATIC);
492492
}
493493

494+
this.add(Qualif::NOT_CONST);
495+
494496
let base_ty = proj.base.ty(this.mir, this.tcx).to_ty(this.tcx);
495497
if let ty::TyRawPtr(_) = base_ty.sty {
496-
this.add(Qualif::NOT_CONST);
497498
if this.mode != Mode::Fn {
498499
struct_span_err!(this.tcx.sess,
499500
this.span, E0396,
@@ -570,7 +571,38 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
570571

571572
fn visit_rvalue(&mut self, rvalue: &Rvalue<'tcx>, location: Location) {
572573
// Recurse through operands and places.
573-
self.super_rvalue(rvalue, location);
574+
if let Rvalue::Ref(region, kind, ref place) = *rvalue {
575+
let mut is_reborrow = false;
576+
if let Place::Projection(ref proj) = *place {
577+
if let ProjectionElem::Deref = proj.elem {
578+
let base_ty = proj.base.ty(self.mir, self.tcx).to_ty(self.tcx);
579+
if let ty::TyRef(..) = base_ty.sty {
580+
is_reborrow = true;
581+
}
582+
}
583+
}
584+
585+
if is_reborrow {
586+
self.nest(|this| {
587+
this.super_place(place, PlaceContext::Borrow {
588+
region,
589+
kind
590+
}, location);
591+
if !this.try_consume() {
592+
return;
593+
}
594+
595+
if this.qualif.intersects(Qualif::STATIC_REF) {
596+
this.qualif = this.qualif - Qualif::STATIC_REF;
597+
this.add(Qualif::STATIC);
598+
}
599+
});
600+
} else {
601+
self.super_rvalue(rvalue, location);
602+
}
603+
} else {
604+
self.super_rvalue(rvalue, location);
605+
}
574606

575607
match *rvalue {
576608
Rvalue::Use(_) |

src/librustc_passes/consts.rs

+12-13
Original file line numberDiff line numberDiff line change
@@ -352,14 +352,9 @@ fn check_expr<'a, 'tcx>(v: &mut CheckCrateVisitor<'a, 'tcx>, e: &hir::Expr, node
352352
hir::ExprBox(_) => {
353353
v.promotable = false;
354354
}
355-
hir::ExprUnary(op, ref inner) => {
356-
match v.tables.node_id_to_type(inner.hir_id).sty {
357-
ty::TyRawPtr(_) => {
358-
assert!(op == hir::UnDeref);
359-
360-
v.promotable = false;
361-
}
362-
_ => {}
355+
hir::ExprUnary(op, _) => {
356+
if op == hir::UnDeref {
357+
v.promotable = false;
363358
}
364359
}
365360
hir::ExprBinary(op, ref lhs, _) => {
@@ -548,7 +543,8 @@ fn check_expr<'a, 'tcx>(v: &mut CheckCrateVisitor<'a, 'tcx>, e: &hir::Expr, node
548543
fn check_adjustments<'a, 'tcx>(v: &mut CheckCrateVisitor<'a, 'tcx>, e: &hir::Expr) {
549544
use rustc::ty::adjustment::*;
550545

551-
for adjustment in v.tables.expr_adjustments(e) {
546+
let mut adjustments = v.tables.expr_adjustments(e).iter().peekable();
547+
while let Some(adjustment) = adjustments.next() {
552548
match adjustment.kind {
553549
Adjust::NeverToAny |
554550
Adjust::ReifyFnPointer |
@@ -558,11 +554,14 @@ fn check_adjustments<'a, 'tcx>(v: &mut CheckCrateVisitor<'a, 'tcx>, e: &hir::Exp
558554
Adjust::Borrow(_) |
559555
Adjust::Unsize => {}
560556

561-
Adjust::Deref(ref overloaded) => {
562-
if overloaded.is_some() {
563-
v.promotable = false;
564-
break;
557+
Adjust::Deref(_) => {
558+
if let Some(next_adjustment) = adjustments.peek() {
559+
if let Adjust::Borrow(_) = next_adjustment.kind {
560+
continue;
561+
}
565562
}
563+
v.promotable = false;
564+
break;
566565
}
567566
}
568567
}

0 commit comments

Comments
 (0)