annotate parser/Parser.d @ 22:e331e4e816e4

now handling structs to some extend
author johnsen@johnsen-laptop
date Fri, 18 Apr 2008 23:45:45 +0200
parents 0fb2d13dce37
children 69464d465284
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
21
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
10 import misc.Error;
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
11
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
12 import tango.io.Stdout,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
13 Integer = tango.text.convert.Integer;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
14
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
15 class Parser
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
16 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
17
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
18 public:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
19 Decl[] parse(Lexer lexer)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
20 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
21 this.lexer = lexer;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
22
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
23
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
24 Decl[] declarations;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
25
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
26 while(lexer.peek.type != Tok.EOF)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
27 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
28 declarations ~= parseDecl;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
29 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
30
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
31 return declarations;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
32 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
33
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
34 Decl parseDecl()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
35 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
36 Token t = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
37
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
38 switch(t.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
39 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
40 case Tok.Byte, Tok.Ubyte,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
41 Tok.Short, Tok.Ushort,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
42 Tok.Int, Tok.Uint,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
43 Tok.Long, Tok.Ulong,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
44 Tok.Float, Tok.Double,
12
6282db07115f Added some ekstra tests, and allowed bool as a type
Anders Halager <halager@gmail.com>
parents: 11
diff changeset
45 Tok.Bool,
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
46 Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
47 Identifier type = new Identifier(t);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
48
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
49 Token iden = lexer.next;
21
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
50
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
51 switch(iden.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
52 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
53 case Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
54 Identifier identifier = new Identifier(iden);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
55 Token p = lexer.peek();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
56 switch(p.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
57 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
58 case Tok.OpenParentheses:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
59 return parseFunc(type, identifier);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
60 case Tok.Seperator:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
61 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
62 return new VarDecl(type, identifier, null);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
63 case Tok.Assign:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
64 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
65 auto exp = parseExpression();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
66 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
67 return new VarDecl(type, identifier, exp);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
68 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
69 char[] c = t.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
70 error("Unexpexted token "~c~" at line "~Integer.toString(__LINE__));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
71 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
72 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
73 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
74 char[] c = t.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
75 error("Unexpexted token "~c~" at line "~Integer.toString(__LINE__));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
76 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
77 break;
22
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
78 case Tok.Struct:
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
79 Token iden = lexer.next;
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
80 switch(iden.type)
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
81 {
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
82 case Tok.Identifier:
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
83 Identifier identifier = new Identifier(iden);
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
84 return new StructDecl (identifier, parseStruct());
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
85 default:
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
86 throw new Error("Expected struct identifier, but got "~iden.getType,
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
87 iden.location);
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
88 }
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
89 case Tok.EOF:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
90 return null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
91 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
92 char[] c = t.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
93 error("Unexpexted token "~c~" at line "~Integer.toString(__LINE__));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
94 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
95 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
96
22
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
97 VarDecl[] parseStruct()
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
98 {
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
99 VarDecl[] varDecls;
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
100 require(Tok.OpenBrace);
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
101 while(lexer.peek.type != Tok.CloseBrace)
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
102 {
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
103 varDecls ~= cast(VarDecl)parseDecl;
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
104 }
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
105
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
106 require(Tok.CloseBrace);
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
107 return varDecls;
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
108 }
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
109
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
110 Stmt parseStatement()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
111 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
112 Token t = lexer.peek;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
113
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
114 switch(t.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
115 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
116 case Tok.Return:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
117 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
118 auto ret = new ReturnStmt();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
119 ret.exp = parseExpression();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
120 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
121 return ret;
5
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
122
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
123 case Tok.If:
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
124 lexer.next;
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
125 require(Tok.OpenParentheses);
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
126 auto condition = parseExpression();
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
127 require(Tok.CloseParentheses);
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
128
11
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
129 auto then_body = parseBlockOrSingleStmt();
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
130
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
131 Stmt[] else_body;
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
132 if (lexer.peek.type == Tok.Else)
5
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
133 {
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
134 lexer.next;
11
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
135 else_body = parseBlockOrSingleStmt();
5
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
136 }
11
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
137
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
138 return new IfStmt(condition, then_body, else_body);
5
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
139
11
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
140 case Tok.While:
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
141 lexer.next;
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
142 require(Tok.OpenParentheses);
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
143 auto condition = parseExpression();
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
144 require(Tok.CloseParentheses);
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
145 return new WhileStmt(condition, parseBlockOrSingleStmt());
5
2c5a8f4c254a Added very simple if support.
Anders Halager <halager@gmail.com>
parents: 1
diff changeset
146
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
147 case Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
148 Token n = lexer.peek(1);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
149 switch(n.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
150 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
151 case Tok.Assign:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
152 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
153 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
154 auto stmt = new ExpStmt(new AssignExp(new Identifier(t), parseExpression()));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
155 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
156 return stmt;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
157 break;
22
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
158 case Tok.Identifier:
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
159 auto decl = new DeclStmt(parseDecl());
e331e4e816e4 now handling structs to some extend
johnsen@johnsen-laptop
parents: 21
diff changeset
160 return decl;
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
161
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
162 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
163 auto e = new ExpStmt(parseExpression());
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
164 require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
165 return e;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
166
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
167 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
168 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
169
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
170 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
171 auto decl = new DeclStmt(parseDecl());
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
172 //require(Tok.Seperator);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
173 return decl;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
174 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
175 return new Stmt();
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 FuncDecl parseFunc(Identifier type, Identifier identifier)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
179 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
180 VarDecl[] funcArgs = parseFuncArgs();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
181
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
182 lexer.next; // Remove the "{"
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
183
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
184 Stmt[] statements;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
185
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
186 while(lexer.peek.type != Tok.CloseBrace)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
187 statements ~= parseStatement();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
188
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
189 lexer.next; // Remove "}"
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
190
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
191 return new FuncDecl(type, identifier, funcArgs, statements);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
192 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
193
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
194 VarDecl[] parseFuncArgs()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
195 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
196 lexer.next; // Remove the "(" token.
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
197
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
198 VarDecl[] funcArgs;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
199
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
200 while(lexer.peek.type != Tok.CloseParentheses)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
201 {
21
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
202 auto t = parseType;
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
203 auto i = parseIdentifier;
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
204 funcArgs ~= new VarDecl(t, i);
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
205
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
206 if(lexer.peek.type == Tok.Comma)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
207 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
208 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
209
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
210 lexer.next; // Remove the ")"
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
211
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
212 return funcArgs;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
213 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
214
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
215 Identifier parseIdentifier()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
216 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
217 Token identifier = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
218
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
219 switch(identifier.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
220 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
221 case Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
222 return new Identifier(identifier);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
223 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
224 default:
21
0fb2d13dce37 Now working with gdc also (gdc use reverse paremeter validating on function calls)
johnsen@johnsen-laptop
parents: 12
diff changeset
225 throw new Error("Unexpexted token in Identifier parsing. Got "~identifier.getType, identifier.location);
1
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
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
229 Identifier parseType()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
230 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
231 Token type = lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
232
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
233 switch(type.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
234 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
235 case Tok.Byte, Tok.Ubyte,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
236 Tok.Short, Tok.Ushort,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
237 Tok.Int, Tok.Uint,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
238 Tok.Long, Tok.Ulong,
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
239 Tok.Float, Tok.Double,
12
6282db07115f Added some ekstra tests, and allowed bool as a type
Anders Halager <halager@gmail.com>
parents: 11
diff changeset
240 Tok.Bool,
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
241 Tok.Identifier:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
242 return new Identifier(type);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
243 break;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
244 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
245 char[] c = type.getType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
246 error("Unexpexted token in Type parsing. Got "~c);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
247 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
248 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
249
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
250 // -- Expression parsing -- //
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
251 private:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
252 Exp parseExpression(int p = 0)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
253 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
254 auto exp = P();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
255 Token next = lexer.peek();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
256 BinOp* op = null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
257 while ((op = binary(next.type)) != null && op.prec >= p)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
258 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
259 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
260 int q = op.leftAssoc? 1 + op.prec : op.prec;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
261 auto exp2 = parseExpression(q);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
262 exp = new BinaryExp(op.operator, exp, exp2);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
263 next = lexer.peek();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
264 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
265
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
266 return exp;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
267 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
268
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
269 Exp P()
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
270 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
271 Token next = lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
272 if (auto op = unary(next.type))
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
273 return new NegateExp(parseExpression(op.prec));
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
274 else if (next.type == Tok.OpenParentheses)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
275 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
276 auto e = parseExpression(0);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
277 require(Tok.CloseParentheses);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
278 return e;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
279 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
280 else if (next.type == Tok.Identifier)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
281 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
282 switch(lexer.peek.type)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
283 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
284 case Tok.OpenParentheses:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
285 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
286 Exp[] args;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
287 while(lexer.peek.type != Tok.CloseParentheses)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
288 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
289 if(lexer.peek.type == Tok.Comma)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
290 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
291 lexer.next;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
292 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
293 args ~= parseExpression();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
294 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
295
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
296 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
297 return new CallExp(new Identifier(next), args);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
298
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
299 default:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
300 return new Identifier(next);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
301 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
302 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
303 else if (next.type == Tok.Integer)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
304 return new IntegerLit(next);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
305
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
306 Stdout.formatln("{}", next.getType);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
307 assert(0, "Should not happen");
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
308 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
309
11
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
310 private Stmt[] parseBlockOrSingleStmt()
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
311 {
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
312 Stmt[] stmts;
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
313 if (lexer.peek.type == Tok.OpenBrace)
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
314 {
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
315 lexer.next;
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
316 while(lexer.peek.type != Tok.CloseBrace)
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
317 stmts ~= parseStatement();
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
318 lexer.next;
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
319 }
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
320 else
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
321 stmts ~= parseStatement();
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
322
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
323 return stmts;
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
324 }
642c6a998fd9 Support for while statements and fixed scope for if
Anders Halager <halager@gmail.com>
parents: 10
diff changeset
325
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
326 struct UnOp
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
327 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
328 Tok tokenType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
329 int prec;
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 static UnOp[] _unary = [{Tok.Sub, 4}];
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
333 UnOp* unary(Tok t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
334 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
335 foreach (ref op; _unary)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
336 if (op.tokenType == t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
337 return &op;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
338 return null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
339 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
340
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
341 struct BinOp
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
342 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
343 Tok tokenType;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
344 int prec;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
345 bool leftAssoc;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
346 BinaryExp.Operator operator;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
347 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
348
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
349 static BinOp[] _binary =
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
350 [
10
2f493057cf17 Some support for the rest of the boolean operators
Anders Halager <halager@gmail.com>
parents: 7
diff changeset
351 {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
352 {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
353 {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
354 {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
355 {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
356 {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
357
7
2ce5209f1954 Starting to work on bool support, for now == works
Anders Halager <halager@gmail.com>
parents: 5
diff changeset
358 {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
359 {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
360
7
2ce5209f1954 Starting to work on bool support, for now == works
Anders Halager <halager@gmail.com>
parents: 5
diff changeset
361 {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
362 {Tok.Div, 5, true, BinaryExp.Operator.Div}
1
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
363 ];
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
364 BinOp* binary(Tok t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
365 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
366 foreach (ref op; _binary)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
367 if (op.tokenType == t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
368 return &op;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
369 return null;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
370 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
371
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
372 private:
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
373
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
374 void require(Tok t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
375 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
376 if (lexer.peek().type != t)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
377 error("Unexpexted token: Got '"~lexer.peek.getType~"' Expected '"~typeToString[t]~"'");
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
378 lexer.next();
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
379 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
380
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
381 void error(char[] errMsg)
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
382 {
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
383 throw new Exception("Parser error: " ~errMsg);
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
384 }
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
385
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
386 Lexer lexer;
2168f4cb73f1 First push
johnsen@johnsen-desktop
parents:
diff changeset
387 }