From 18ad54b003608f3060d4baf2fef00a8a2b6f0a7b Mon Sep 17 00:00:00 2001
From: Michael Forney <mforney@mforney.org>
Date: Sat, 16 Feb 2019 03:39:03 -0800
Subject: Allow labels with same name as typedef

---
 pp.c                    |  3 ++-
 stmt.c                  | 31 ++++++++++++++++++++-----------
 tests/label-typedef.c   |  4 ++++
 tests/label-typedef.qbe |  7 +++++++
 4 files changed, 33 insertions(+), 12 deletions(-)
 create mode 100644 tests/label-typedef.c
 create mode 100644 tests/label-typedef.qbe

diff --git a/pp.c b/pp.c
index 41b67ed..af36686 100644
--- a/pp.c
+++ b/pp.c
@@ -112,7 +112,8 @@ next(void)
 bool
 peek(int kind)
 {
-	nextinto(&pending);
+	if (!pending.kind)
+		nextinto(&pending);
 	if (pending.kind != kind)
 		return false;
 	pending.kind = TNONE;
diff --git a/stmt.c b/stmt.c
index 8eced24..a5192c5 100644
--- a/stmt.c
+++ b/stmt.c
@@ -14,6 +14,23 @@
 #include "token.h"
 #include "type.h"
 
+static bool
+gotolabel(struct function *f)
+{
+	char *name;
+	struct gotolabel *g;
+
+	if (tok.kind != TIDENT)
+		return false;
+	name = tok.lit;
+	if (!peek(TCOLON))
+		return false;
+	g = funcgoto(f, name);
+	g->defined = true;
+	funclabel(f, g->label);
+	return true;
+}
+
 /* 6.8 Statements and blocks */
 void
 stmt(struct function *f, struct scope *s)
@@ -25,6 +42,8 @@ stmt(struct function *f, struct scope *s)
 	struct switchcases swtch = {0};
 	uint64_t i;
 
+	while (gotolabel(f))
+		;
 	switch (tok.kind) {
 	/* 6.8.1 Labeled statements */
 	case TCASE:
@@ -55,7 +74,7 @@ stmt(struct function *f, struct scope *s)
 		next();
 		s = mkscope(s);
 		while (tok.kind != TRBRACE) {
-			if (!decl(s, f))
+			if (gotolabel(f) || !decl(s, f))
 				stmt(f, s);
 		}
 		s = delscope(s);
@@ -66,16 +85,6 @@ stmt(struct function *f, struct scope *s)
 	case TSEMICOLON:
 		next();
 		break;
-	case TIDENT:
-		name = tok.lit;
-		if (peek(TCOLON)) {
-			struct gotolabel *g = funcgoto(f, name);
-			g->defined = true;
-			funclabel(f, g->label);
-			stmt(f, s);
-			break;
-		}
-		/* fallthrough */
 	default:
 		e = expr(s);
 		v = funcexpr(f, e);
diff --git a/tests/label-typedef.c b/tests/label-typedef.c
new file mode 100644
index 0000000..f3744e3
--- /dev/null
+++ b/tests/label-typedef.c
@@ -0,0 +1,4 @@
+typedef int x;
+void f(void) {
+x:;
+}
diff --git a/tests/label-typedef.qbe b/tests/label-typedef.qbe
new file mode 100644
index 0000000..b11f621
--- /dev/null
+++ b/tests/label-typedef.qbe
@@ -0,0 +1,7 @@
+export
+function $f() {
+@start.1
+@body.2
+@x.3
+	ret
+}
-- 
cgit v1.2.3