Remove a little weirdness from the language.
catseye
12 years ago
93 | 93 |
Grammar
|
94 | 94 |
-------
|
95 | 95 |
|
96 | |
The grammar is a little weird at points (especially the VarRef production,
|
97 | |
which includes the possibility of assignment, even though that can really
|
98 | |
only happen in a Stmt.)
|
99 | |
|
100 | 96 |
Program ::= {Defn [";"]}.
|
101 | 97 |
Defn ::= "fun" ident "(" [Arg {"," Arg}] ")" Body
|
102 | 98 |
| "struct" ident "{" {ident ":" TExpr [";"]} "}"
|
|
105 | 101 |
Body ::= "{" {VarDecl [";"]} {Stmt [";"]} "}".
|
106 | 102 |
VarDecl ::= "var" ident "=" Expr0.
|
107 | 103 |
Stmt ::= "while" Expr0 Block
|
108 | |
| "typecase" VarRef "is" TExpr0 Block
|
|
104 |
| "typecase" ident "is" TExpr0 Block
|
109 | 105 |
| "do" Expr0
|
110 | 106 |
| "return" Expr0
|
111 | 107 |
| If
|
|
121 | 117 |
| "(" Expr0 ")"
|
122 | 118 |
| "not" Expr1
|
123 | 119 |
| Literal
|
124 | |
| VarRef.
|
125 | |
VarRef ::= ident ["=" Expr0].
|
|
120 |
| ident ["=" Expr0].
|
126 | 121 |
Literal ::= strlit
|
127 | 122 |
| ["-"] intlit
|
128 | 123 |
| "true" | "false" | "null"
|
|
1084 | 1079 |
| }
|
1085 | 1080 |
= nothing there
|
1086 | 1081 |
|
1087 | |
This is a very strange case in the language. Thankfully, assignment
|
1088 | |
typechecks as void, without any automatic promotion to the union type...
|
1089 | |
|
1090 | |
| fun foo(b: integer|void) {
|
1091 | |
| var a = b;
|
1092 | |
| typecase a is integer {
|
1093 | |
| print("yes it's an integer");
|
1094 | |
| }
|
1095 | |
| typecase a = 7 as integer|void is void {
|
1096 | |
| print("yes it's void too");
|
1097 | |
| }
|
1098 | |
| }
|
1099 | |
| main = fun() {
|
1100 | |
| foo(7 as integer|void)
|
1101 | |
| }
|
1102 | |
? void not a union
|
1103 | |
|
1104 | 1082 |
### Struct Types + Union Types ###
|
1105 | 1083 |
|
1106 | 1084 |
Union types may be used to make fields of a struct "nullable", so that
|
251 | 251 |
b = self.block()
|
252 | 252 |
return AST('While', [t, b])
|
253 | 253 |
elif self.consume('typecase'):
|
254 | |
e = self.var_ref()
|
|
254 |
id = self.expect_type('identifier')
|
|
255 |
e = AST('VarRef', value=id)
|
255 | 256 |
self.expect('is')
|
256 | 257 |
te = self.texpr0()
|
257 | 258 |
b = self.block()
|
|
360 | 361 |
self.expect(')')
|
361 | 362 |
return e
|
362 | 363 |
else:
|
363 | |
return self.var_ref()
|
364 | |
|
365 | |
def var_ref(self):
|
366 | |
# Note: this production includes the possibility of assignment
|
367 | |
# therefore slightly weird things can occur in the grammar
|
368 | |
id = self.expect_type('identifier')
|
369 | |
ast = AST('VarRef', value=id)
|
370 | |
if self.consume('='):
|
371 | |
e = self.expr0()
|
372 | |
ast = AST('Assignment', [ast, e])
|
373 | |
return ast
|
|
364 |
id = self.expect_type('identifier')
|
|
365 |
ast = AST('VarRef', value=id)
|
|
366 |
if self.consume('='):
|
|
367 |
e = self.expr0()
|
|
368 |
ast = AST('Assignment', [ast, e])
|
|
369 |
return ast
|
374 | 370 |
|
375 | 371 |
def literal(self):
|
376 | 372 |
if self.on_type('string literal'):
|