mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-23 07:14:28 +00:00
Check inferred integer literals for overflows, closes #4220
This commit is contained in:
parent
be09626148
commit
20627c7430
@ -529,7 +529,7 @@ msgid ""
|
|||||||
"The basic example below illustrates this.\n"
|
"The basic example below illustrates this.\n"
|
||||||
"~~~\n"
|
"~~~\n"
|
||||||
"# fn make_a_sandwich() {};\n"
|
"# fn make_a_sandwich() {};\n"
|
||||||
"fn fib(n: uint) -> uint {\n"
|
"fn fib(n: u64) -> u64 {\n"
|
||||||
" // lengthy computation returning an uint\n"
|
" // lengthy computation returning an uint\n"
|
||||||
" 12586269025\n"
|
" 12586269025\n"
|
||||||
"}\n"
|
"}\n"
|
||||||
|
@ -529,7 +529,7 @@ msgid ""
|
|||||||
"The basic example below illustrates this.\n"
|
"The basic example below illustrates this.\n"
|
||||||
"~~~\n"
|
"~~~\n"
|
||||||
"# fn make_a_sandwich() {};\n"
|
"# fn make_a_sandwich() {};\n"
|
||||||
"fn fib(n: uint) -> uint {\n"
|
"fn fib(n: u64) -> u64 {\n"
|
||||||
" // lengthy computation returning an uint\n"
|
" // lengthy computation returning an uint\n"
|
||||||
" 12586269025\n"
|
" 12586269025\n"
|
||||||
"}\n"
|
"}\n"
|
||||||
|
@ -273,7 +273,7 @@ later.
|
|||||||
The basic example below illustrates this.
|
The basic example below illustrates this.
|
||||||
~~~
|
~~~
|
||||||
# fn make_a_sandwich() {};
|
# fn make_a_sandwich() {};
|
||||||
fn fib(n: uint) -> uint {
|
fn fib(n: u64) -> u64 {
|
||||||
// lengthy computation returning an uint
|
// lengthy computation returning an uint
|
||||||
12586269025
|
12586269025
|
||||||
}
|
}
|
||||||
|
@ -72,6 +72,7 @@ pub enum lint {
|
|||||||
non_uppercase_statics,
|
non_uppercase_statics,
|
||||||
non_uppercase_pattern_statics,
|
non_uppercase_pattern_statics,
|
||||||
type_limits,
|
type_limits,
|
||||||
|
type_overflow,
|
||||||
unused_unsafe,
|
unused_unsafe,
|
||||||
|
|
||||||
managed_heap_memory,
|
managed_heap_memory,
|
||||||
@ -219,6 +220,14 @@ static lint_table: &'static [(&'static str, LintSpec)] = &[
|
|||||||
default: warn
|
default: warn
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
("type_overflow",
|
||||||
|
LintSpec {
|
||||||
|
lint: type_overflow,
|
||||||
|
desc: "literal out of range for its type",
|
||||||
|
default: warn
|
||||||
|
}),
|
||||||
|
|
||||||
|
|
||||||
("unused_unsafe",
|
("unused_unsafe",
|
||||||
LintSpec {
|
LintSpec {
|
||||||
lint: unused_unsafe,
|
lint: unused_unsafe,
|
||||||
@ -321,6 +330,9 @@ struct Context {
|
|||||||
// levels, this stack keeps track of the previous lint levels of whatever
|
// levels, this stack keeps track of the previous lint levels of whatever
|
||||||
// was modified.
|
// was modified.
|
||||||
lint_stack: ~[(lint, level, LintSource)],
|
lint_stack: ~[(lint, level, LintSource)],
|
||||||
|
|
||||||
|
// id of the last visited negated expression
|
||||||
|
negated_expr_id: ast::NodeId
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Context {
|
impl Context {
|
||||||
@ -507,7 +519,48 @@ fn check_type_limits(cx: &Context, e: &ast::Expr) {
|
|||||||
cx.span_lint(type_limits, e.span,
|
cx.span_lint(type_limits, e.span,
|
||||||
"comparison is useless due to type limits");
|
"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");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
_ => ()
|
||||||
|
};
|
||||||
|
},
|
||||||
_ => ()
|
_ => ()
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -1078,11 +1131,25 @@ impl Visitor<()> for Context {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn visit_expr(&mut self, e: @ast::Expr, _: ()) {
|
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_while_true_expr(self, e);
|
||||||
check_stability(self, e);
|
check_stability(self, e);
|
||||||
check_unused_unsafe(self, e);
|
check_unused_unsafe(self, e);
|
||||||
check_unnecessary_allocation(self, e);
|
check_unnecessary_allocation(self, e);
|
||||||
check_heap_expr(self, e);
|
check_heap_expr(self, e);
|
||||||
|
|
||||||
check_type_limits(self, e);
|
check_type_limits(self, e);
|
||||||
|
|
||||||
visit::walk_expr(self, e, ());
|
visit::walk_expr(self, e, ());
|
||||||
@ -1139,6 +1206,7 @@ pub fn check_crate(tcx: ty::ctxt, crate: &ast::Crate) {
|
|||||||
cur: SmallIntMap::new(),
|
cur: SmallIntMap::new(),
|
||||||
tcx: tcx,
|
tcx: tcx,
|
||||||
lint_stack: ~[],
|
lint_stack: ~[],
|
||||||
|
negated_expr_id: -1
|
||||||
};
|
};
|
||||||
|
|
||||||
// Install default lint levels, followed by the command line levels, and
|
// Install default lint levels, followed by the command line levels, and
|
||||||
|
@ -24,11 +24,13 @@ fn bar() -> i8 {
|
|||||||
|
|
||||||
fn baz() -> bool {
|
fn baz() -> bool {
|
||||||
128 > bar() //~ ERROR comparison is useless due to type limits
|
128 > bar() //~ ERROR comparison is useless due to type limits
|
||||||
|
//~^ WARNING literal out of range for its type
|
||||||
}
|
}
|
||||||
|
|
||||||
fn qux() {
|
fn qux() {
|
||||||
let mut i = 1i8;
|
let mut i = 1i8;
|
||||||
while 200 != i { //~ ERROR comparison is useless due to type limits
|
while 200 != i { //~ ERROR comparison is useless due to type limits
|
||||||
|
//~^ WARNING literal out of range for its type
|
||||||
i += 1;
|
i += 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
50
src/test/compile-fail/lint-type-overflow.rs
Normal file
50
src/test/compile-fail/lint-type-overflow.rs
Normal file
@ -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
|
||||||
|
}
|
@ -8,6 +8,6 @@
|
|||||||
// option. This file may not be copied, modified, or distributed
|
// option. This file may not be copied, modified, or distributed
|
||||||
// except according to those terms.
|
// 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
|
||||||
|
Loading…
Reference in New Issue
Block a user