checker: check for enum value duplicates/overflow (#5659)

pull/5662/head
div72 2020-07-04 18:28:01 +03:00 committed by GitHub
parent b018381f10
commit be04de98bb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 74 additions and 13 deletions

View File

@ -12,6 +12,8 @@ import v.errors
const ( const (
max_nr_errors = 300 max_nr_errors = 300
enum_min = int(0x80000000)
enum_max = 0x7FFFFFFF
) )
pub struct Checker { pub struct Checker {
@ -740,7 +742,8 @@ fn (mut c Checker) check_map_and_filter(is_map bool, elem_typ table.Type, call_e
ast.AnonFn { ast.AnonFn {
if arg_expr.decl.args.len > 1 { if arg_expr.decl.args.len > 1 {
c.error('function needs exactly 1 argument', call_expr.pos) c.error('function needs exactly 1 argument', call_expr.pos)
} else if is_map && (arg_expr.decl.return_type != elem_typ || arg_expr.decl.args[0].typ != elem_typ) { } else if is_map && (arg_expr.decl.return_type != elem_typ ||
arg_expr.decl.args[0].typ != elem_typ) {
c.error('type mismatch, should use `fn(a $elem_sym.name) $elem_sym.name {...}`', c.error('type mismatch, should use `fn(a $elem_sym.name) $elem_sym.name {...}`',
call_expr.pos) call_expr.pos)
} else if !is_map && (arg_expr.decl.return_type != table.bool_type || } else if !is_map && (arg_expr.decl.return_type != table.bool_type ||
@ -1131,7 +1134,6 @@ pub fn (mut c Checker) call_fn(mut call_expr ast.CallExpr) table.Type {
} }
if call_arg.is_mut { if call_arg.is_mut {
c.fail_if_immutable(call_arg.expr) c.fail_if_immutable(call_arg.expr)
if !arg.is_mut { if !arg.is_mut {
mut words := 'mutable' mut words := 'mutable'
mut tok := 'mut' mut tok := 'mut'
@ -1427,6 +1429,7 @@ pub fn (mut c Checker) return_stmt(mut return_stmt ast.Return) {
pub fn (mut c Checker) enum_decl(decl ast.EnumDecl) { pub fn (mut c Checker) enum_decl(decl ast.EnumDecl) {
c.check_valid_pascal_case(decl.name, 'enum name', decl.pos) c.check_valid_pascal_case(decl.name, 'enum name', decl.pos)
mut seen := []int{}
for i, field in decl.fields { for i, field in decl.fields {
if util.contains_capital(field.name) { if util.contains_capital(field.name) {
c.error('field name `$field.name` cannot contain uppercase letters, use snake_case instead', c.error('field name `$field.name` cannot contain uppercase letters, use snake_case instead',
@ -1438,8 +1441,16 @@ pub fn (mut c Checker) enum_decl(decl ast.EnumDecl) {
} }
} }
if field.has_expr { if field.has_expr {
match field.expr { match field.expr as field_expr {
ast.IntegerLiteral {} ast.IntegerLiteral {
val := field_expr.val.i64()
if val < enum_min || val > enum_max {
c.error('enum value `$val` overflows int', field_expr.pos)
} else if int(val) in seen {
c.error('enum value `$val` already exists', field_expr.pos)
}
seen << int(val)
}
ast.PrefixExpr {} ast.PrefixExpr {}
else { else {
if field.expr is ast.Ident { if field.expr is ast.Ident {
@ -1455,6 +1466,16 @@ pub fn (mut c Checker) enum_decl(decl ast.EnumDecl) {
c.error('default value for enum has to be an integer', pos) c.error('default value for enum has to be an integer', pos)
} }
} }
} else {
if seen.len > 0 {
last := seen[seen.len - 1]
if last == enum_max {
c.error('enum value overflows', field.pos)
}
seen << last + 1
} else {
seen << 0
}
} }
} }
} }
@ -2953,7 +2974,6 @@ fn (mut c Checker) fn_decl(node ast.FnDecl) {
} }
} }
} }
if node.language == .v && node.is_method && node.name == 'str' { if node.language == .v && node.is_method && node.name == 'str' {
if node.return_type != table.string_type { if node.return_type != table.string_type {
c.error('.str() methods should return `string`', node.pos) c.error('.str() methods should return `string`', node.pos)
@ -2962,7 +2982,6 @@ fn (mut c Checker) fn_decl(node ast.FnDecl) {
c.error('.str() methods should have 0 arguments', node.pos) c.error('.str() methods should have 0 arguments', node.pos)
} }
} }
c.expected_type = table.void_type c.expected_type = table.void_type
c.cur_fn = &node c.cur_fn = &node
c.stmts(node.stmts) c.stmts(node.stmts)

View File

@ -0,0 +1,6 @@
vlib/v/checker/tests/enum_field_overflow.v:4:2: error: enum value overflows
2 | red
3 | green = 2147483647
4 | blue
| ~~~~
5 | }

View File

@ -0,0 +1,5 @@
enum Color {
red
green = 2147483647
blue
}

View File

@ -0,0 +1,13 @@
vlib/v/checker/tests/enum_field_value_duplicate_err.v:3:10: error: enum value `0` already exists
1 | enum Color {
2 | red
3 | green = 0
| ^
4 | blue = 1
5 | alpha = 1
vlib/v/checker/tests/enum_field_value_duplicate_err.v:5:10: error: enum value `1` already exists
3 | green = 0
4 | blue = 1
5 | alpha = 1
| ^
6 | }

View File

@ -0,0 +1,6 @@
enum Color {
red
green = 0
blue = 1
alpha = 1
}

View File

@ -0,0 +1,7 @@
vlib/v/checker/tests/enum_field_value_overflow.v:3:10: error: enum value `2147483648` overflows int
1 | enum Color {
2 | red
3 | green = 2147483648
| ~~~~~~~~~~
4 | blue
5 | }

View File

@ -0,0 +1,5 @@
enum Color {
red
green = 2147483648
blue
}