shithub: scc

ref: 675adafd7148e7fdcc413759648e2bb45d795109
dir: /decl.c/

View raw version
#include <assert.h>
#include <stddef.h>
#include <string.h>

#include "sizes.h"
#include "cc.h"
#include "tokens.h"
#include "syntax.h"
#include "symbol.h"

char parser_out_home;

static struct symbol *cursym;
static void declarator(struct ctype *tp);

static void
newiden(struct ctype *tp)
{
	register unsigned char yyns, ns;

	ns = tp->c_typedef ? NS_TYPEDEF : NS_IDEN;
	yyns = yyval.sym->ns;

	if (yyns == NS_ANY) {     /* First appearence of the symbol */
		yyval.sym->ns = ns;
		cursym = yyval.sym;
		return;
	} else if (ns == yyns) {  /* Duplicated symbol */
		if (yyval.sym->ctx == curctx)
			error("redeclaration of '%s'", yytext);
	}
	cursym = lookup(yytext, ns);
}

static void
dirdcl(register struct ctype *tp)
{
	if (accept('(')) {
		declarator(tp);
		expect(')');
	} else if (yytoken == IDEN) {
		newiden(tp);
		next();
	} else {
		error("expected '(' or identifier before of '%s'", yytext);
	}

	for (;;) {
		if (accept('(')) {
			next();
			pushtype(FTN);
			if (accept(')'))
				; /* TODO: k&r function */
			else
				/* TODO: prototyped function */;
			continue;
		} else if (accept('[')) {
			unsigned len;

			if (accept(']')) {
				len = 0;
			} else {
				expect(CONSTANT);
				len = yyval.sym->val;
				expect(']');
			}
			pushtype(len);
			pushtype(ARY);
			continue;
		} else {
			return;
		}
	}
}

struct ctype *
spec(void)
{
	unsigned char sign;
	register struct ctype *tp = NULL;

	for (sign = 0; ; next()) {
		switch (yytoken) {
		case TYPEDEF:  case EXTERN: case STATIC: case AUTO:
		case REGISTER: case CONST:  case VOLATILE:
			tp = storage(tp, yytoken);
			break;
		case UNSIGNED: case SIGNED:
			if (sign == yytoken)
				error("duplicated '%s'", yytext);
			if (sign)
				error("both 'signed' and 'unsigned' in declaration specifiers");
			if (!tp)
				tp = newctype();
			switch (tp->type) {
			case FLOAT: case DOUBLE: case LDOUBLE: case BOOL:
				goto invalid_sign;
			}
			if ((sign = yytoken) == UNSIGNED)
				tp->c_unsigned = 1;
			break;
		case FLOAT: case DOUBLE: case BOOL:
			if (sign)
				goto invalid_sign;
		case VOID: case CHAR: case SHORT: case INT: case LONG:
			tp = btype(tp, yytoken);
			break;
		case STRUCT:    /* TODO */
		case UNION:	/* TODO */
		case ENUM:	/* TODO */
		case IDEN:
			if (!tp || !tp->type) {
				struct symbol *sym;
				unsigned char tok = ahead();

				sym = (yyval.sym->ns == NS_TYPEDEF) ?
					yyval.sym : find(yytext, NS_TYPEDEF);
				if (sym && tok != ';' && tok != ',') {
					(tp = sym->ctype)->refcnt++;
					next();
				}
			}
		default:
			if (!tp || tp->type)
				return tp;
			warning_error(options.implicit,
			              "type defaults to 'int' in declaration");
			tp->type = INT;
			return tp;
		}
	}
invalid_sign:
	error("invalid sign modifier");
}

static void
declarator(struct ctype *tp)
{
	unsigned char qlf[NR_DECLARATORS];
	register unsigned char *bp, *lim;

	lim = &qlf[NR_DECLARATORS];
	for (bp = qlf; yytoken == '*' && bp != lim; ) {
		next();
		*bp++ = PTR;
		while (bp != lim) {
			switch (yytoken) {
			case CONST: case VOLATILE: case RESTRICT:
				*bp++ = yytoken;
				next();
				break;
			default:
				goto continue_outer;
			}
		}
	continue_outer: ;
	}
	if (bp == lim)
		error("Too much type declarators");

	dirdcl(tp);

	for (lim = bp - 1, bp = qlf; bp < lim; ++bp)
		pushtype(*bp);
}

static struct node *
initializer(register struct ctype *tp)
{
	register struct node *np;

	if (accept('{')) {
		np = nodecomp();
		addstmt(np, initializer(tp));
		while (accept(',')) {
			if (accept('}'))
				return np;
			addstmt(np, initializer(tp));
		}
		expect('}');
	} else {
		np = expr();
	}
	return np;
}

static struct node *
listdcl(struct ctype *base)
{
	struct node *lp = nodecomp();

	do {
		struct node *sp, *np;
		register struct ctype *tp;

		declarator(base);
		tp = decl_type(base);
		(cursym->ctype = tp)->refcnt++;
		sp = nodesym(cursym);
		if (tp->type == FTN && yytoken == '{') {
			np  = node2(ODEF, sp, function(cursym));
			return addstmt(lp, np);
		}
		np = node2(ODEF, sp, accept('=') ? initializer(tp) : NULL);
		lp = addstmt(lp, np);
	} while (accept(','));
	expect(';');

	return lp;
}

struct node *
decl(void)
{
	register struct ctype *tp;
	register struct node *np = NULL;

	while (accept(';'))
		/* nothing */;
	if (!(tp = spec())) {
		if (curctx != CTX_OUTER)
			goto end;
		warning("data definition has no type or storage class");
	}
	if (accept(';')) {
		warning_error(options.useless,
			      "useless type name in empty declaration");
		delctype(tp);
	} else {
		np = listdcl(tp);
	}

end:	return np;
}

void
type_name()
{

}