annotate parser/Parser.d @ 10:2f493057cf17

Some support for the rest of the boolean operators
author Anders Halager <halager@gmail.com>
date Fri, 18 Apr 2008 13:01:11 +0200
parents 2ce5209f1954
children 642c6a998fd9
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
1 module parser.Parser;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
2
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
3 import lexer.Lexer,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
4 lexer.Token;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
5
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
6 import ast.Exp,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
7 ast.Stmt,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
8 ast.Decl;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
9
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
10 import tango.io.Stdout,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
11 Integer = tango.text.convert.Integer;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
12
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
13 class Parser
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
14 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
15
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
16 public:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
17 Decl[] parse(Lexer lexer)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
18 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
19 this.lexer = lexer;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
20
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
21
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
22 Decl[] declarations;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
23
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
24 while(lexer.peek.type != Tok.EOF)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
25 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
26 declarations ~= parseDecl;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
27 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
28
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
29 return declarations;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
30 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
31
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
32 Decl parseDecl()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
33 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
34 Token t = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
35
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
36 switch(t.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
37 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
38 case Tok.Byte, Tok.Ubyte,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
39 Tok.Short, Tok.Ushort,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
40 Tok.Int, Tok.Uint,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
41 Tok.Long, Tok.Ulong,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
42 Tok.Float, Tok.Double,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
43 Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
44 Identifier type = new Identifier(t);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
45
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
46 Token iden = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
47 switch(iden.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
48 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
49 case Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
50 Identifier identifier = new Identifier(iden);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
51 Token p = lexer.peek();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
52 switch(p.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
53 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
54 case Tok.OpenParentheses:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
55 return parseFunc(type, identifier);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
56 case Tok.Seperator:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
57 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
58 return new VarDecl(type, identifier, null);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
59 case Tok.Assign:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
60 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
61 auto exp = parseExpression();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
62 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
63 return new VarDecl(type, identifier, exp);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
64 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
65 char[] c = t.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
66 error("Unexpexted token "~c~" at line "~Integer.toString(__LINE__));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
67 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
68 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
69 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
70 char[] c = t.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
71 error("Unexpexted token "~c~" at line "~Integer.toString(__LINE__));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
72 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
73 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
74
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
75 case Tok.EOF:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
76 return null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
77 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
78 char[] c = t.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
79 error("Unexpexted token "~c~" at line "~Integer.toString(__LINE__));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
80 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
81 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
82
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
83 Stmt parseStatement()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
84 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
85 Token t = lexer.peek;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
86
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
87 switch(t.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
88 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
89 case Tok.Return:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
90 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
91 auto ret = new ReturnStmt();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
92 ret.exp = parseExpression();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
93 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
94 return ret;
5
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
95
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
96 case Tok.If:
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
97 lexer.next;
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
98 require(Tok.OpenParentheses);
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
99 auto condition = parseExpression();
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
100 require(Tok.CloseParentheses);
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
101
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
102 Stmt[] stmts;
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
103 if (lexer.peek.type == Tok.OpenBrace)
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
104 {
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
105 lexer.next;
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
106 while(lexer.peek.type != Tok.CloseBrace)
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
107 stmts ~= parseStatement();
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
108 lexer.next;
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
109 }
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
110 else
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
111 stmts ~= parseStatement();
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
112
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
113 return new IfStmt(condition, stmts);
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
114
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
115 case Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
116 Token n = lexer.peek(1);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
117 switch(n.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
118 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
119 case Tok.Assign:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
120 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
121 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
122 auto stmt = new ExpStmt(new AssignExp(new Identifier(t), parseExpression()));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
123 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
124 return stmt;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
125 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
126
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
127 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
128 auto e = new ExpStmt(parseExpression());
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
129 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
130 return e;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
131
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
132 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
133 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
134
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
135 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
136 auto decl = new DeclStmt(parseDecl());
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
137 //require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
138 return decl;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
139 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
140 return new Stmt();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
141 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
142
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
143 FuncDecl parseFunc(Identifier type, Identifier identifier)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
144 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
145 VarDecl[] funcArgs = parseFuncArgs();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
146
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
147 lexer.next; // Remove the "{"
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
148
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
149 Stmt[] statements;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
150
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
151 while(lexer.peek.type != Tok.CloseBrace)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
152 statements ~= parseStatement();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
153
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
154 lexer.next; // Remove "}"
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
155
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
156 return new FuncDecl(type, identifier, funcArgs, statements);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
157 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
158
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
159 VarDecl[] parseFuncArgs()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
160 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
161 lexer.next; // Remove the "(" token.
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
162
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
163 VarDecl[] funcArgs;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
164
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
165 while(lexer.peek.type != Tok.CloseParentheses)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
166 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
167 funcArgs ~= new VarDecl(parseType, parseIdentifier);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
168
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
169 if(lexer.peek.type == Tok.Comma)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
170 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
171 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
172
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
173 lexer.next; // Remove the ")"
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
174
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
175 return funcArgs;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
176 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
177
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
178 Identifier parseIdentifier()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
179 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
180 Token identifier = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
181
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
182 switch(identifier.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
183 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
184 case Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
185 return new Identifier(identifier);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
186 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
187 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
188 error("Unexpexted token in Identifier parsing");
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
189 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
190 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
191
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
192 Identifier parseType()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
193 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
194 Token type = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
195
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
196 switch(type.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
197 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
198 case Tok.Byte, Tok.Ubyte,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
199 Tok.Short, Tok.Ushort,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
200 Tok.Int, Tok.Uint,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
201 Tok.Long, Tok.Ulong,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
202 Tok.Float, Tok.Double,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
203 Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
204 return new Identifier(type);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
205 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
206 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
207 char[] c = type.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
208 error("Unexpexted token in Type parsing. Got "~c);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
209 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
210 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
211
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
212 // -- Expression parsing -- //
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
213 private:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
214 Exp parseExpression(int p = 0)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
215 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
216 auto exp = P();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
217 Token next = lexer.peek();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
218 BinOp* op = null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
219 while ((op = binary(next.type)) != null && op.prec >= p)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
220 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
221 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
222 int q = op.leftAssoc? 1 + op.prec : op.prec;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
223 auto exp2 = parseExpression(q);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
224 exp = new BinaryExp(op.operator, exp, exp2);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
225 next = lexer.peek();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
226 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
227
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
228 return exp;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
229 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
230
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
231 Exp P()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
232 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
233 Token next = lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
234 if (auto op = unary(next.type))
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
235 return new NegateExp(parseExpression(op.prec));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
236 else if (next.type == Tok.OpenParentheses)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
237 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
238 auto e = parseExpression(0);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
239 require(Tok.CloseParentheses);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
240 return e;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
241 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
242 else if (next.type == Tok.Identifier)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
243 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
244 switch(lexer.peek.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
245 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
246 case Tok.OpenParentheses:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
247 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
248 Exp[] args;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
249 while(lexer.peek.type != Tok.CloseParentheses)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
250 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
251 if(lexer.peek.type == Tok.Comma)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
252 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
253 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
254 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
255 args ~= parseExpression();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
256 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
257
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
258 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
259 return new CallExp(new Identifier(next), args);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
260
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
261 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
262 return new Identifier(next);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
263 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
264 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
265 else if (next.type == Tok.Integer)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
266 return new IntegerLit(next);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
267
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
268 Stdout.formatln("{}", next.getType);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
269 assert(0, "Should not happen");
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
270 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
271
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
272 struct UnOp
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
273 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
274 Tok tokenType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
275 int prec;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
276 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
277
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
278 static UnOp[] _unary = [{Tok.Sub, 4}];
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
279 UnOp* unary(Tok t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
280 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
281 foreach (ref op; _unary)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
282 if (op.tokenType == t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
283 return &op;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
284 return null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
285 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
286
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
287 struct BinOp
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
288 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
289 Tok tokenType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
290 int prec;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
291 bool leftAssoc;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
292 BinaryExp.Operator operator;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
293 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
294
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
295 static BinOp[] _binary =
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
296 [
10
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
297 {Tok.Eq, 2, true, BinaryExp.Operator.Eq},
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
298 {Tok.Ne, 2, true, BinaryExp.Operator.Ne},
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
299 {Tok.Lt, 2, true, BinaryExp.Operator.Lt},
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
300 {Tok.Le, 2, true, BinaryExp.Operator.Le},
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
301 {Tok.Gt, 2, true, BinaryExp.Operator.Gt},
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
302 {Tok.Ge, 2, true, BinaryExp.Operator.Ge},
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
303
7
2ce5209f1954 Starting to work on bool support, for now == works
Anders Halager <halager@gmail.com>
parents: 5
diff changeset
304 {Tok.Add, 3, true, BinaryExp.Operator.Add},
2ce5209f1954 Starting to work on bool support, for now == works
Anders Halager <halager@gmail.com>
parents: 5
diff changeset
305 {Tok.Sub, 3, true, BinaryExp.Operator.Sub},
10
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
306
7
2ce5209f1954 Starting to work on bool support, for now == works
Anders Halager <halager@gmail.com>
parents: 5
diff changeset
307 {Tok.Mul, 5, true, BinaryExp.Operator.Mul},
2ce5209f1954 Starting to work on bool support, for now == works
Anders Halager <halager@gmail.com>
parents: 5
diff changeset
308 {Tok.Div, 5, true, BinaryExp.Operator.Div}
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
309 ];
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
310 BinOp* binary(Tok t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
311 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
312 foreach (ref op; _binary)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
313 if (op.tokenType == t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
314 return &op;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
315 return null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
316 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
317
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
318 private:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
319
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
320 void require(Tok t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
321 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
322 if (lexer.peek().type != t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
323 error("Unexpexted token: Got '"~lexer.peek.getType~"' Expected '"~typeToString[t]~"'");
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
324 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
325 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
326
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
327 void error(char[] errMsg)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
328 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
329 throw new Exception("Parser error: " ~errMsg);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
330 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
331
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
332 Lexer lexer;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
333 }