diff options
author | Michael Forney <mforney@mforney.org> | 2020-06-03 02:39:09 -0700 |
---|---|---|
committer | Michael Forney <mforney@mforney.org> | 2020-06-03 02:39:09 -0700 |
commit | 6229709b8ae21d7722fef48ad8a9f2f10b900030 (patch) | |
tree | 16df9a6d31167cb85db8ec257dea2ded9693dc95 | |
parent | 0b153ef15c12a5a27f9a65000a3d7036b1829366 (diff) |
decl: Allow out-of-range enum constants when they don't change type
gcc and clang allow enum constants out of range of int, but this
means that the type of enumerator may differ inside and outside the
enum specifier.
Instead, only allow out-of-range enum constants when their types
are compatible with the final enum type.
-rw-r--r-- | cc.h | 2 | ||||
-rw-r--r-- | decl.c | 85 | ||||
-rw-r--r-- | type.c | 17 |
3 files changed, 59 insertions, 45 deletions
@@ -147,7 +147,6 @@ enum typekind { TYPECHAR, TYPESHORT, TYPEINT, - TYPEENUM, TYPELONG, TYPELLONG, TYPEFLOAT, @@ -172,6 +171,7 @@ enum typeprop { PROPAGGR = 1<<6, PROPDERIVED = 1<<7, PROPFLOAT = 1<<8, + PROPENUM = 1<<9, }; struct param { @@ -1,4 +1,5 @@ #include <assert.h> +#include <limits.h> #include <inttypes.h> #include <stdbool.h> #include <stddef.h> @@ -143,22 +144,35 @@ funcspec(enum funcspec *fs) static void structdecl(struct scope *, struct structbuilder *); +static unsigned long long +intmax(struct type *t) +{ + return -1ull >> sizeof(unsigned long long) * CHAR_BIT - t->size * 8 + t->basic.issigned; +} + +static bool +inrange(struct type *t, unsigned long long v, bool sign) +{ + if (sign && v >= 1ull << 63) + return t->basic.issigned && v >= -1ull << t->size * 8 - 1; + return v <= intmax(t); +} + static struct type * tagspec(struct scope *s) { - struct type *t; + struct type *t, *et, *ct; char *tag, *name; enum typekind kind; struct decl *d; struct expr *e; struct structbuilder b; uint64_t i; - bool large; switch (tok.kind) { case TSTRUCT: kind = TYPESTRUCT; break; case TUNION: kind = TYPEUNION; break; - case TENUM: kind = TYPEENUM; break; + case TENUM: kind = TYPEINT; break; default: fatal("internal error: unknown tag kind"); } next(); @@ -168,30 +182,28 @@ tagspec(struct scope *s) } else { tag = expect(TIDENT, "or '{' after struct/union"); t = scopegettag(s, tag, false); - if (s->parent && !t && tok.kind != TLBRACE && (kind == TYPEENUM || tok.kind != TSEMICOLON)) + if (s->parent && !t && tok.kind != TLBRACE && (kind == TYPEINT || tok.kind != TSEMICOLON)) t = scopegettag(s->parent, tag, true); } - if (t) { - if (t->kind != kind) - error(&tok.loc, "redeclaration of tag '%s' with different kind", tag); - } else { - if (kind == TYPEENUM) { - t = xmalloc(sizeof(*t)); - *t = typeuint; - t->kind = kind; + if (!t) { + t = mktype(kind, PROPOBJECT); + t->size = 0; + t->align = 0; + if (kind == TYPEINT) { + t->prop |= PROPENUM; + t->basic.issigned = false; } else { - t = mktype(kind, PROPOBJECT); if (kind == TYPESTRUCT) t->prop |= PROPAGGR; t->repr = &i64; // XXX - t->size = 0; - t->align = 0; t->structunion.tag = tag; t->structunion.members = NULL; } t->incomplete = true; if (tag) scopeputtag(s, tag, t); + } else if (t->kind != kind && (kind != TYPEINT || t->prop & PROPENUM)) { + error(&tok.loc, "redeclaration of tag '%s' with different kind", tag); } if (tok.kind != TLBRACE) return t; @@ -210,8 +222,9 @@ tagspec(struct scope *s) t->size = ALIGNUP(t->size, t->align); t->incomplete = false; break; - case TYPEENUM: - large = false; + case TYPEINT: + et = NULL; + ct = &typeint; for (i = 0; tok.kind == TIDENT; ++i) { name = tok.lit; next(); @@ -220,31 +233,31 @@ tagspec(struct scope *s) if (e->kind != EXPRCONST || !(e->type->prop & PROPINT)) error(&tok.loc, "expected integer constant expression"); i = e->constant.i; - if (e->type->basic.issigned && i >= 1ull << 63) { - if (i < -1ull << 31) - goto invalid; - t->basic.issigned = true; - } else if (i >= 1ull << 32) { - goto invalid; - } - } else if (i == 1ull << 32) { - invalid: - error(&tok.loc, "enumerator '%s' value cannot be represented as 'int' or 'unsigned int'", name); - } - d = mkdecl(DECLCONST, &typeint, QUALNONE, LINKNONE); - d->value = mkintconst(t->repr, i); - if (i >= 1ull << 31 && i < 1ull << 63) { - large = true; - d->type = &typeuint; + t->basic.issigned |= e->type->basic.issigned && i >= 1ull << 63; + if (inrange(&typeint, i, e->type->basic.issigned)) + ct = &typeint; + else if (et && !typecompatible(et, e->type)) + error(&tok.loc, "enumerator '%s' value is outside the range of 'int'", name); + else + ct = et = e->type; + } else if (i - 1 == intmax(ct)) { + /* the constant is outside the range of the type of the previous enumerator */ + error(&tok.loc, "enumerator '%s' value overflow", name); } - if (large && t->basic.issigned) - error(&tok.loc, "neither 'int' nor 'unsigned' can represent all enumerator values"); + d = mkdecl(DECLCONST, ct, QUALNONE, LINKNONE); + d->value = mkintconst(ct->repr, i); scopeputdecl(s, name, d); if (!consume(TCOMMA)) break; } expect(TRBRACE, "to close enum specifier"); - t->incomplete = false; + if (!et) + et = t->basic.issigned ? &typeint : &typeuint; + else if (t->basic.issigned && !et->basic.issigned) + error(&tok.loc, "enumerator with range outside of 'int' is not compatible with enum type"); + *t = *et; + t->prop |= PROPENUM; + break; } return t; @@ -106,7 +106,6 @@ typerank(struct type *t) case TYPEBOOL: return 1; case TYPECHAR: return 2; case TYPESHORT: return 3; - case TYPEENUM: case TYPEINT: return 4; case TYPELONG: return 5; case TYPELLONG: return 6; @@ -123,14 +122,16 @@ typecompatible(struct type *t1, struct type *t2) if (t1 == t2) return true; - if (t1->kind != t2->kind) { - /* enum types are compatible with 'int', but not with - each other (unless they are the same type) */ - return (t1->kind == TYPEENUM && t2->kind == TYPEINT || - t1->kind == TYPEINT && t2->kind == TYPEENUM) && - t1->basic.issigned == t2->basic.issigned; - } + if (t1->kind != t2->kind) + return false; switch (t1->kind) { + case TYPEINT: + case TYPELONG: + case TYPELLONG: + /* enum types are compatible with some basic integer + type, but not with other enum types */ + return (t1->prop & PROPENUM) != (t2->prop & PROPENUM) && + t1->basic.issigned == t2->basic.issigned; case TYPEPOINTER: goto derived; case TYPEARRAY: |