Skip to content

Commit

Permalink
Check inferred integer literals for overflows, closes #4220
Browse files Browse the repository at this point in the history
  • Loading branch information
fhahn committed Nov 12, 2013
1 parent be09626 commit 20627c7
Show file tree
Hide file tree
Showing 7 changed files with 126 additions and 6 deletions.
2 changes: 1 addition & 1 deletion doc/po/ja/tutorial-tasks.md.po
Original file line number Diff line number Diff line change
Expand Up @@ -529,7 +529,7 @@ msgid ""
"The basic example below illustrates this.\n"
"~~~\n"
"# fn make_a_sandwich() {};\n"
"fn fib(n: uint) -> uint {\n"
"fn fib(n: u64) -> u64 {\n"
" // lengthy computation returning an uint\n"
" 12586269025\n"
"}\n"
Expand Down
2 changes: 1 addition & 1 deletion doc/po/tutorial-tasks.md.pot
Original file line number Diff line number Diff line change
Expand Up @@ -529,7 +529,7 @@ msgid ""
"The basic example below illustrates this.\n"
"~~~\n"
"# fn make_a_sandwich() {};\n"
"fn fib(n: uint) -> uint {\n"
"fn fib(n: u64) -> u64 {\n"
" // lengthy computation returning an uint\n"
" 12586269025\n"
"}\n"
Expand Down
2 changes: 1 addition & 1 deletion doc/tutorial-tasks.md
Original file line number Diff line number Diff line change
Expand Up @@ -273,7 +273,7 @@ later.
The basic example below illustrates this.
~~~
# fn make_a_sandwich() {};
fn fib(n: uint) -> uint {
fn fib(n: u64) -> u64 {
// lengthy computation returning an uint
12586269025
}
Expand Down
70 changes: 69 additions & 1 deletion src/librustc/middle/lint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ pub enum lint {
non_uppercase_statics,
non_uppercase_pattern_statics,
type_limits,
type_overflow,
unused_unsafe,

managed_heap_memory,
Expand Down Expand Up @@ -219,6 +220,14 @@ static lint_table: &'static [(&'static str, LintSpec)] = &[
default: warn
}),

("type_overflow",
LintSpec {
lint: type_overflow,
desc: "literal out of range for its type",
default: warn
}),


("unused_unsafe",
LintSpec {
lint: unused_unsafe,
Expand Down Expand Up @@ -321,6 +330,9 @@ struct Context {
// levels, this stack keeps track of the previous lint levels of whatever
// was modified.
lint_stack: ~[(lint, level, LintSource)],

// id of the last visited negated expression
negated_expr_id: ast::NodeId
}

impl Context {
Expand Down Expand Up @@ -507,7 +519,48 @@ fn check_type_limits(cx: &Context, e: &ast::Expr) {
cx.span_lint(type_limits, e.span,
"comparison is useless due to type limits");
}
}
},
ast::ExprLit(lit) => {
match ty::get(ty::expr_ty(cx.tcx, e)).sty {
ty::ty_int(t) => {
let int_type = if t == ast::ty_i {
cx.tcx.sess.targ_cfg.int_type
} else { t };
let (min, max) = int_ty_range(int_type);
let mut lit_val: i64 = match lit.node {
ast::lit_int(v, _) => v,
ast::lit_uint(v, _) => v as i64,
ast::lit_int_unsuffixed(v) => v,
_ => fail!()
};
if cx.negated_expr_id == e.id {
lit_val *= -1;
}
if lit_val < min || lit_val > max {
cx.span_lint(type_overflow, e.span,
"literal out of range for its type");
}
},
ty::ty_uint(t) => {
let uint_type = if t == ast::ty_u {
cx.tcx.sess.targ_cfg.uint_type
} else { t };
let (min, max) = uint_ty_range(uint_type);
let lit_val: u64 = match lit.node {
ast::lit_int(v, _) => v as u64,
ast::lit_uint(v, _) => v,
ast::lit_int_unsuffixed(v) => v as u64,
_ => fail!()
};
if lit_val < min || lit_val > max {
cx.span_lint(type_overflow, e.span,
"literal out of range for its type");
}
},

_ => ()
};
},
_ => ()
};

Expand Down Expand Up @@ -1078,11 +1131,25 @@ impl Visitor<()> for Context {
}

fn visit_expr(&mut self, e: @ast::Expr, _: ()) {
match e.node {
ast::ExprUnary(_, ast::UnNeg, expr) => {
// propagate negation, if the negation itself isn't negated
if self.negated_expr_id != e.id {
self.negated_expr_id = expr.id;
}
},
ast::ExprParen(expr) => if self.negated_expr_id == e.id {
self.negated_expr_id = expr.id
},
_ => ()
};

check_while_true_expr(self, e);
check_stability(self, e);
check_unused_unsafe(self, e);
check_unnecessary_allocation(self, e);
check_heap_expr(self, e);

check_type_limits(self, e);

visit::walk_expr(self, e, ());
Expand Down Expand Up @@ -1139,6 +1206,7 @@ pub fn check_crate(tcx: ty::ctxt, crate: &ast::Crate) {
cur: SmallIntMap::new(),
tcx: tcx,
lint_stack: ~[],
negated_expr_id: -1
};

// Install default lint levels, followed by the command line levels, and
Expand Down
2 changes: 2 additions & 0 deletions src/test/compile-fail/lint-type-limits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,11 +24,13 @@ fn bar() -> i8 {

fn baz() -> bool {
128 > bar() //~ ERROR comparison is useless due to type limits
//~^ WARNING literal out of range for its type
}

fn qux() {
let mut i = 1i8;
while 200 != i { //~ ERROR comparison is useless due to type limits
//~^ WARNING literal out of range for its type
i += 1;
}
}
50 changes: 50 additions & 0 deletions src/test/compile-fail/lint-type-overflow.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
// Copyright 2013 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.
//

#[deny(type_overflow)];

fn test(x: i8) {
println!("x {}", x);
}

#[allow(unused_variable)]
fn main() {
let x1: u8 = 255; // should be OK
let x1: u8 = 256; //~ error: literal out of range for its type

let x1 = 255_u8; // should be OK
let x1 = 256_u8; //~ error: literal out of range for its type

let x2: i8 = -128; // should be OK
let x1: i8 = 128; //~ error: literal out of range for its type
let x2: i8 = --128; //~ error: literal out of range for its type

let x3: i8 = -129; //~ error: literal out of range for its type
let x3: i8 = -(129); //~ error: literal out of range for its type
let x3: i8 = -{129}; //~ error: literal out of range for its type

test(1000); //~ error: literal out of range for its type

let x = 128_i8; //~ error: literal out of range for its type
let x = 127_i8;
let x = -128_i8;
let x = -(128_i8);
let x = -129_i8; //~ error: literal out of range for its type

let x: i32 = 2147483647; // should be OK
let x = 2147483647_i32; // should be OK
let x: i32 = 2147483648; //~ error: literal out of range for its type
let x = 2147483648_i32; //~ error: literal out of range for its type
let x: i32 = -2147483648; // should be OK
let x = -2147483648_i32; // should be OK
let x: i32 = -2147483649; //~ error: literal out of range for its type
let x = -2147483649_i32; //~ error: literal out of range for its type
}
4 changes: 2 additions & 2 deletions src/test/compile-fail/oversized-literal.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,6 @@
// option. This file may not be copied, modified, or distributed
// except according to those terms.

// error-pattern:literal out of range
// compile-flags: -D type-overflow

fn main() { info!("{}", 300u8); }
fn main() { info!("{}", 300u8); } //~ error: literal out of range for its type

5 comments on commit 20627c7

@bors
Copy link
Contributor

@bors bors commented on 20627c7 Nov 14, 2013

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

saw approval from alexcrichton
at fhahn@20627c7

@bors
Copy link
Contributor

@bors bors commented on 20627c7 Nov 14, 2013

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

merging fhahn/rust/check-inferred-ints = 20627c7 into auto

@bors
Copy link
Contributor

@bors bors commented on 20627c7 Nov 14, 2013

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fhahn/rust/check-inferred-ints = 20627c7 merged ok, testing candidate = ade310c

@bors
Copy link
Contributor

@bors bors commented on 20627c7 Nov 14, 2013

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@bors
Copy link
Contributor

@bors bors commented on 20627c7 Nov 14, 2013

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fast-forwarding master to auto = ade310c

Please sign in to comment.