Skip to content

Commit c74c854

Browse files
committed
Better error message for semicolon on the last line of a function
closes #8005
1 parent caf316a commit c74c854

File tree

2 files changed

+55
-3
lines changed

2 files changed

+55
-3
lines changed

src/librustc/middle/liveness.rs

+25-3
Original file line numberDiff line numberDiff line change
@@ -439,7 +439,7 @@ fn visit_fn(v: &mut LivenessVisitor,
439439

440440
// check for various error conditions
441441
lsets.visit_block(body, ());
442-
lsets.check_ret(id, sp, fk, entry_ln);
442+
lsets.check_ret(id, sp, fk, entry_ln, body);
443443
lsets.warn_about_unused_args(decl, entry_ln);
444444
}
445445

@@ -1575,7 +1575,8 @@ impl Liveness {
15751575
id: NodeId,
15761576
sp: Span,
15771577
_fk: &FnKind,
1578-
entry_ln: LiveNode) {
1578+
entry_ln: LiveNode,
1579+
body: &Block) {
15791580
if self.live_on_entry(entry_ln, self.s.no_ret_var).is_some() {
15801581
// if no_ret_var is live, then we fall off the end of the
15811582
// function without any kind of return expression:
@@ -1588,9 +1589,30 @@ impl Liveness {
15881589
self.tcx.sess.span_err(
15891590
sp, "some control paths may return");
15901591
} else {
1592+
let ends_with_stmt = match body.expr {
1593+
None if body.stmts.len() > 0 =>
1594+
match body.stmts.last().node {
1595+
StmtSemi(e, _) => {
1596+
let t_stmt = ty::expr_ty(self.tcx, e);
1597+
ty::get(t_stmt).sty == ty::get(t_ret).sty
1598+
},
1599+
_ => false
1600+
},
1601+
_ => false
1602+
};
1603+
if ends_with_stmt {
1604+
let last_stmt = body.stmts.last();
1605+
let span_semicolon = Span {
1606+
lo: last_stmt.span.hi,
1607+
hi: last_stmt.span.hi,
1608+
expn_info: last_stmt.span.expn_info
1609+
};
1610+
self.tcx.sess.span_note(
1611+
span_semicolon, "consider removing this semicolon:");
1612+
}
15911613
self.tcx.sess.span_err(
15921614
sp, "not all control paths return a value");
1593-
}
1615+
}
15941616
}
15951617
}
15961618

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
// Copyright 2014 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+
// regression test for #8005
12+
13+
#[feature(macro_rules)];
14+
15+
macro_rules! test ( () => { fn foo() -> int { 1i; } } ) //~ ERROR not all control paths return a value
16+
//~^ NOTE consider removing this semicolon
17+
18+
fn no_return() -> int {} //~ ERROR not all control paths return a value
19+
20+
fn bar(x: u32) -> u32 { //~ ERROR not all control paths return a value
21+
x * 2; //~ NOTE consider removing this semicolon
22+
}
23+
24+
fn baz(x: u64) -> u32 { //~ ERROR not all control paths return a value
25+
x * 2;
26+
}
27+
28+
fn main() {
29+
test!();
30+
}

0 commit comments

Comments
 (0)