Skip to content

Commit

Permalink
Auto merge of #47261 - estebank:immutable-arg, r=petrochenkov
Browse files Browse the repository at this point in the history
Assignment to immutable argument: diagnostic tweak

Re #46659.
  • Loading branch information
bors committed Jan 14, 2018
2 parents b762c2d + 90bc98c commit fd0f292
Show file tree
Hide file tree
Showing 6 changed files with 78 additions and 14 deletions.
3 changes: 2 additions & 1 deletion src/librustc_borrowck/borrowck/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -772,11 +772,12 @@ impl<'a, 'tcx> BorrowckCtxt<'a, 'tcx> {
&move_data::Assignment) {
let mut err = self.cannot_reassign_immutable(span,
&self.loan_path_to_string(lp),
false,
Origin::Ast);
err.span_label(span, "cannot assign twice to immutable variable");
if span != assign.span {
err.span_label(assign.span, format!("first assignment to `{}`",
self.loan_path_to_string(lp)));
self.loan_path_to_string(lp)));
}
err.emit();
}
Expand Down
34 changes: 27 additions & 7 deletions src/librustc_mir/borrow_check/error_reporting.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@

use syntax_pos::Span;
use rustc::middle::region::ScopeTree;
use rustc::mir::{BorrowKind, Field, Local, Location, Operand};
use rustc::mir::{BorrowKind, Field, Local, LocalKind, Location, Operand};
use rustc::mir::{Place, ProjectionElem, Rvalue, Statement, StatementKind};
use rustc::ty::{self, RegionKind};
use rustc_data_structures::indexed_vec::Idx;
Expand Down Expand Up @@ -568,19 +568,39 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
(place, span): (&Place<'tcx>, Span),
assigned_span: Span,
) {
let is_arg = if let Place::Local(local) = place {
if let LocalKind::Arg = self.mir.local_kind(*local) {
true
} else {
false
}
} else {
false
};

let mut err = self.tcx.cannot_reassign_immutable(
span,
&self.describe_place(place).unwrap_or("_".to_owned()),
is_arg,
Origin::Mir,
);
err.span_label(span, "cannot assign twice to immutable variable");
let msg = if is_arg {
"cannot assign to immutable argument"
} else {
"cannot assign twice to immutable variable"
};
if span != assigned_span {
let value_msg = match self.describe_place(place) {
Some(name) => format!("`{}`", name),
None => "value".to_owned(),
};
err.span_label(assigned_span, format!("first assignment to {}", value_msg));
if is_arg {
err.span_label(assigned_span, "argument not declared as `mut`");
} else {
let value_msg = match self.describe_place(place) {
Some(name) => format!("`{}`", name),
None => "value".to_owned(),
};
err.span_label(assigned_span, format!("first assignment to {}", value_msg));
}
}
err.span_label(span, msg);
err.emit();
}
}
Expand Down
11 changes: 8 additions & 3 deletions src/librustc_mir/util/borrowck_errors.rs
Original file line number Diff line number Diff line change
Expand Up @@ -269,12 +269,17 @@ pub trait BorrowckErrors {
self.cancel_if_wrong_origin(err, o)
}

fn cannot_reassign_immutable(&self, span: Span, desc: &str, o: Origin)
fn cannot_reassign_immutable(&self, span: Span, desc: &str, is_arg: bool, o: Origin)
-> DiagnosticBuilder
{
let msg = if is_arg {
"to immutable argument"
} else {
"twice to immutable variable"
};
let err = struct_span_err!(self, span, E0384,
"cannot assign twice to immutable variable `{}`{OGN}",
desc, OGN=o);
"cannot assign {} `{}`{OGN}",
msg, desc, OGN=o);

self.cancel_if_wrong_origin(err, o)
}
Expand Down
6 changes: 3 additions & 3 deletions src/test/compile-fail/issue-45199.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,11 +31,11 @@ fn test_call() {
}

fn test_args(b: Box<i32>) { //[ast]~ NOTE first assignment
//[mir]~^ NOTE first assignment
//[mir]~^ NOTE argument not declared as `mut`
b = Box::new(2); //[ast]~ ERROR cannot assign twice to immutable variable
//[mir]~^ ERROR cannot assign twice to immutable variable `b`
//[mir]~^ ERROR cannot assign to immutable argument `b`
//[ast]~| NOTE cannot assign twice to immutable
//[mir]~| NOTE cannot assign twice to immutable
//[mir]~| NOTE cannot assign to immutable argument
}

fn main() {}
20 changes: 20 additions & 0 deletions src/test/ui/borrowck/immutable-arg.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

//compile-flags: -Z emit-end-regions -Z borrowck=compare

fn foo(_x: u32) {
_x = 4;
//~^ ERROR cannot assign to immutable argument `_x` (Mir)
//~^^ ERROR cannot assign twice to immutable variable `_x` (Ast)
}

fn main() {}

18 changes: 18 additions & 0 deletions src/test/ui/borrowck/immutable-arg.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
error[E0384]: cannot assign twice to immutable variable `_x` (Ast)
--> $DIR/immutable-arg.rs:14:5
|
13 | fn foo(_x: u32) {
| -- first assignment to `_x`
14 | _x = 4;
| ^^^^^^ cannot assign twice to immutable variable

error[E0384]: cannot assign to immutable argument `_x` (Mir)
--> $DIR/immutable-arg.rs:14:5
|
13 | fn foo(_x: u32) {
| -- argument not declared as `mut`
14 | _x = 4;
| ^^^^^^ cannot assign to immutable argument

error: aborting due to 2 previous errors

0 comments on commit fd0f292

Please sign in to comment.