v/vlib/compiler/enum.v

109 lines
2.4 KiB
V
Raw Normal View History

// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
// Use of this source code is governed by an MIT license
// that can be found in the LICENSE file.
module compiler
fn (p mut Parser) enum_decl(no_name bool) {
is_pub := p.tok == .key_pub
if is_pub {
p.next()
2019-12-19 20:52:27 +01:00
p.fspace()
2019-12-05 12:09:33 +01:00
}
p.check(.key_enum)
2019-11-10 01:08:53 +01:00
p.fspace()
2019-12-19 22:29:37 +01:00
mut enum_name := p.check_name()
is_c := enum_name == 'C' && p.tok == .dot
if is_c {
p.check(.dot)
enum_name = p.check_name()
}
// Specify full type name
if !p.builtin_mod && p.mod != 'main' {
enum_name = p.prepend_mod(enum_name)
}
// Skip empty enums
if !no_name && !p.first_pass() {
p.cgen.typedefs << 'typedef int $enum_name;'
}
2019-11-10 01:08:53 +01:00
p.fspace()
p.check(.lcbr)
mut val := 0
mut fields := []string
for p.tok == .name {
field := p.check_name()
2019-10-24 13:25:03 +02:00
if contains_capital(field) {
p.warn('enum values cannot contain uppercase letters, use snake_case instead')
}
fields << field
name := '${mod_gen_name(p.mod)}__${enum_name}_$field'
if p.tok == .assign {
2019-12-20 00:35:07 +01:00
p.fspace()
mut enum_assign_tidx := p.cur_tok_index()
2019-12-21 08:35:29 +01:00
next := p.peek()
if next in [.number, .minus] {
p.next()
2019-12-20 00:35:07 +01:00
p.fspace()
2019-12-21 08:35:29 +01:00
is_neg := p.tok == .minus
if is_neg {
p.next()
}
val = p.lit.int()
2019-12-21 08:35:29 +01:00
if is_neg {
val = -val
}
p.next()
2019-12-19 22:29:37 +01:00
}
else {
p.next()
enum_assign_tidx = p.cur_tok_index()
p.error_with_token_index('only numbers are allowed in enum initializations', enum_assign_tidx)
2019-12-05 12:09:33 +01:00
}
}
if p.pass == .main {
p.cgen.consts << '#define $name $val'
}
if p.tok == .comma {
p.next()
}
2019-12-20 00:35:07 +01:00
p.fgen_nl()
val++
}
is_flag := p.attr == 'flag'
if is_flag && fields.len > 32 {
p.error('when an enum is used as bit field, it must have a max of 32 fields')
}
2019-12-19 22:29:37 +01:00
mut T := Type{
name: enum_name
mod: p.mod
parent: 'int'
2019-11-06 04:26:04 +01:00
cat: .enum_
enum_vals: fields.clone()
is_public: is_pub
is_flag: is_flag
}
if is_flag && !p.first_pass() {
p.gen_enum_flag_methods(mut T)
}
p.table.register_type(T)
p.check(.rcbr)
2019-12-20 00:35:07 +01:00
p.fgen_nl()
p.fgen_nl()
}
fn (p mut Parser) check_enum_member_access() {
T := p.find_type(p.expected_type)
if T.cat == .enum_ {
p.check(.dot)
val := p.check_name()
// Make sure this enum value exists
if !T.has_enum_val(val) {
p.error('enum `$T.name` does not have value `$val`')
}
p.gen(mod_gen_name(T.mod) + '__' + p.expected_type + '_' + val)
2019-12-19 22:29:37 +01:00
}
else {
p.error('`$T.name` is not an enum')
}
}