Skip to content

Commit fd0f292

Browse files
committed
Auto merge of #47261 - estebank:immutable-arg, r=petrochenkov
Assignment to immutable argument: diagnostic tweak Re #46659.
2 parents b762c2d + 90bc98c commit fd0f292

File tree

6 files changed

+78
-14
lines changed

6 files changed

+78
-14
lines changed

src/librustc_borrowck/borrowck/mod.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -772,11 +772,12 @@ impl<'a, 'tcx> BorrowckCtxt<'a, 'tcx> {
772772
&move_data::Assignment) {
773773
let mut err = self.cannot_reassign_immutable(span,
774774
&self.loan_path_to_string(lp),
775+
false,
775776
Origin::Ast);
776777
err.span_label(span, "cannot assign twice to immutable variable");
777778
if span != assign.span {
778779
err.span_label(assign.span, format!("first assignment to `{}`",
779-
self.loan_path_to_string(lp)));
780+
self.loan_path_to_string(lp)));
780781
}
781782
err.emit();
782783
}

src/librustc_mir/borrow_check/error_reporting.rs

+27-7
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010

1111
use syntax_pos::Span;
1212
use rustc::middle::region::ScopeTree;
13-
use rustc::mir::{BorrowKind, Field, Local, Location, Operand};
13+
use rustc::mir::{BorrowKind, Field, Local, LocalKind, Location, Operand};
1414
use rustc::mir::{Place, ProjectionElem, Rvalue, Statement, StatementKind};
1515
use rustc::ty::{self, RegionKind};
1616
use rustc_data_structures::indexed_vec::Idx;
@@ -568,19 +568,39 @@ impl<'cx, 'gcx, 'tcx> MirBorrowckCtxt<'cx, 'gcx, 'tcx> {
568568
(place, span): (&Place<'tcx>, Span),
569569
assigned_span: Span,
570570
) {
571+
let is_arg = if let Place::Local(local) = place {
572+
if let LocalKind::Arg = self.mir.local_kind(*local) {
573+
true
574+
} else {
575+
false
576+
}
577+
} else {
578+
false
579+
};
580+
571581
let mut err = self.tcx.cannot_reassign_immutable(
572582
span,
573583
&self.describe_place(place).unwrap_or("_".to_owned()),
584+
is_arg,
574585
Origin::Mir,
575586
);
576-
err.span_label(span, "cannot assign twice to immutable variable");
587+
let msg = if is_arg {
588+
"cannot assign to immutable argument"
589+
} else {
590+
"cannot assign twice to immutable variable"
591+
};
577592
if span != assigned_span {
578-
let value_msg = match self.describe_place(place) {
579-
Some(name) => format!("`{}`", name),
580-
None => "value".to_owned(),
581-
};
582-
err.span_label(assigned_span, format!("first assignment to {}", value_msg));
593+
if is_arg {
594+
err.span_label(assigned_span, "argument not declared as `mut`");
595+
} else {
596+
let value_msg = match self.describe_place(place) {
597+
Some(name) => format!("`{}`", name),
598+
None => "value".to_owned(),
599+
};
600+
err.span_label(assigned_span, format!("first assignment to {}", value_msg));
601+
}
583602
}
603+
err.span_label(span, msg);
584604
err.emit();
585605
}
586606
}

src/librustc_mir/util/borrowck_errors.rs

+8-3
Original file line numberDiff line numberDiff line change
@@ -269,12 +269,17 @@ pub trait BorrowckErrors {
269269
self.cancel_if_wrong_origin(err, o)
270270
}
271271

272-
fn cannot_reassign_immutable(&self, span: Span, desc: &str, o: Origin)
272+
fn cannot_reassign_immutable(&self, span: Span, desc: &str, is_arg: bool, o: Origin)
273273
-> DiagnosticBuilder
274274
{
275+
let msg = if is_arg {
276+
"to immutable argument"
277+
} else {
278+
"twice to immutable variable"
279+
};
275280
let err = struct_span_err!(self, span, E0384,
276-
"cannot assign twice to immutable variable `{}`{OGN}",
277-
desc, OGN=o);
281+
"cannot assign {} `{}`{OGN}",
282+
msg, desc, OGN=o);
278283

279284
self.cancel_if_wrong_origin(err, o)
280285
}

src/test/compile-fail/issue-45199.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -31,11 +31,11 @@ fn test_call() {
3131
}
3232

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

4141
fn main() {}

src/test/ui/borrowck/immutable-arg.rs

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
//compile-flags: -Z emit-end-regions -Z borrowck=compare
12+
13+
fn foo(_x: u32) {
14+
_x = 4;
15+
//~^ ERROR cannot assign to immutable argument `_x` (Mir)
16+
//~^^ ERROR cannot assign twice to immutable variable `_x` (Ast)
17+
}
18+
19+
fn main() {}
20+
+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
error[E0384]: cannot assign twice to immutable variable `_x` (Ast)
2+
--> $DIR/immutable-arg.rs:14:5
3+
|
4+
13 | fn foo(_x: u32) {
5+
| -- first assignment to `_x`
6+
14 | _x = 4;
7+
| ^^^^^^ cannot assign twice to immutable variable
8+
9+
error[E0384]: cannot assign to immutable argument `_x` (Mir)
10+
--> $DIR/immutable-arg.rs:14:5
11+
|
12+
13 | fn foo(_x: u32) {
13+
| -- argument not declared as `mut`
14+
14 | _x = 4;
15+
| ^^^^^^ cannot assign to immutable argument
16+
17+
error: aborting due to 2 previous errors
18+

0 commit comments

Comments
 (0)