/* * Helpers for dealing with type specifiers. * Copyright © 2011 Nick Bowler * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include "cdecl.h" #include "typemap.h" /* * We can represent type specifiers as a bitmap, which gives us a finite * list of acceptable bitmap values according to the C standard. However, * the "long" specifier is allowed to occur more than once, but only at most * 2 times. Treat it as a special case, assigning an unused bit to represent * the second long. */ #define CDECL_TYPE_LLONG 31 static unsigned long add_typespec(unsigned long map, struct cdecl_declspec *s) { if (s->type >= CDECL_TYPE_LLONG) { fprintf(stderr, "invalid type specifier\n"); return -1; } if (s->type == CDECL_TYPE_LONG) { if (map & (1ul<type)) { fprintf(stderr, "duplicate type specifier\n"); return -1; } return map | (1ul<type); } unsigned long cdecl__build_typemap(struct cdecl_declspec *s) { unsigned long map = 0; for (struct cdecl_declspec *c = s; c; c = c->next) { if (cdecl_spec_kind(c) != CDECL_SPEC_TYPE) continue; map = add_typespec(map, c); if (map == -1) return -1; } switch (map) { case 0: fprintf(stderr, "no type specifiers\n"); return -1; # include "validtypes.h" return map; default: fprintf(stderr, "conflicting type specifiers\n"); return -1; } } const char *cdecl__explain_typemap(unsigned long map) { switch (map) { # include "typenames.h" default: fprintf(stderr, "invalid type specifiers\n"); return NULL; } }