about summary refs log tree commit diff
path: root/src/bootstrap/parser.pry
diff options
context:
space:
mode:
authorBaitinq <[email protected]>2025-07-09 23:05:40 +0200
committerBaitinq <[email protected]>2025-07-09 23:05:40 +0200
commit94ffd157cf364529eb0415ee8054f6e8644f784f (patch)
treed5cddc390223cb34ade7ff3e6b2f7e218ffdf0cf /src/bootstrap/parser.pry
parentBootstrap: Support unary expressions (diff)
downloadpry-lang-94ffd157cf364529eb0415ee8054f6e8644f784f.tar.gz
pry-lang-94ffd157cf364529eb0415ee8054f6e8644f784f.tar.bz2
pry-lang-94ffd157cf364529eb0415ee8054f6e8644f784f.zip
Bootstrap: Support while loops
Diffstat (limited to '')
-rw-r--r--src/bootstrap/parser.pry54
1 files changed, 52 insertions, 2 deletions
diff --git a/src/bootstrap/parser.pry b/src/bootstrap/parser.pry
index cb8c269..95e654e 100644
--- a/src/bootstrap/parser.pry
+++ b/src/bootstrap/parser.pry
@@ -403,6 +403,48 @@ let parser_parse_if_statement = (p: *parser) => *Node {
 	return create_node(p, r);
 };
 
+/* WhileStatement ::= "while" Expression LBRACE Statement* RBRACE */
+let parser_parse_while_statement = (p: *parser) => *Node {
+	if parser_accept_token(p, TOKEN_WHILE) == cast(*token, null) {
+		return cast(*Node, null);
+	};
+
+	let expression = parser_parse_expression(p);
+	if expression == cast(*Node, null) {
+	    return cast(*Node, null);
+	};
+	
+	if parser_accept_token(p, TOKEN_LBRACE) == cast(*token, null) {
+		return cast(*Node, null);
+	};
+
+	let statements = cast(**Node, arena_alloc((*p).arena, sizeof(*Node) * 100));
+	let i = 0;
+	while true {
+		let n = parser_accept_parse(p, parser_parse_statement);
+		if n == cast(*Node, null) {
+			break;
+		};
+		(*(statements + cast(**Node, i))) = n;
+		i = i + 1;
+	};
+	
+	if parser_accept_token(p, TOKEN_RBRACE) == cast(*token, null) {
+		return cast(*Node, null);
+	};
+	
+	let dd = cast(*NODE_WHILE_STATEMENT_DATA, arena_alloc((*p).arena, sizeof(NODE_WHILE_STATEMENT_DATA)));
+	(*dd).condition = expression;
+	(*dd).statements = statements;
+	(*dd).statements_len = i;
+
+	let r = Node{};
+	r.type = NODE_WHILE_STATEMENT;
+	r.data = cast(*void, dd);
+
+	return create_node(p, r);
+};
+
 /* ExternDeclaration ::= "extern" IDENTIFIER EQUALS Type */
 let parser_parse_extern_declaration = (p: *parser) => *Node {
 	if parser_accept_token(p, TOKEN_EXTERN) == cast(*token, null) {
@@ -607,10 +649,10 @@ let parser_parse_equality_expression = (p: *parser) => *Node {
 
 	let typ = -1;
 	let ex = parser_accept_parse(p, (ip: *parser) => *Node {
-		if parser_accept_token(ip, TOKEN_BANG) == cast(*token, null) {
+		if parser_accept_token(ip, TOKEN_EQUALS) == cast(*token, null) {
 			return cast(*Node, null);
 		};
-		if parser_accept_token(ip, TOKEN_BANG) == cast(*token, null) {
+		if parser_accept_token(ip, TOKEN_EQUALS) == cast(*token, null) {
 			return cast(*Node, null);
 		};
 		let n = Node{};
@@ -959,6 +1001,14 @@ let parser_parse_statement = (p: *parser) => *Node {
 		};
 	    return retu;
 	};
+	
+	let retu = parser_accept_parse(p, parser_parse_while_statement);
+	if retu != cast(*Node, null) {
+		if parser_accept_token(p, TOKEN_SEMICOLON) == cast(*token, null) {
+		    return cast(*Node, null);
+		};
+	    return retu;
+	};
 
 	printf("None\n");