-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathParser.cs
executable file
·122 lines (121 loc) · 3.89 KB
/
Parser.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace FinalPA
{
// Dsl ::= stats sat notSat
// Stats ::= var "= {" vals "}" stats | ε
// Vals ::= val seqVals | ε
// SeqVals ::= "," val seqVals | ε
// Sat ::= "{" cons "}" | ε
// NotSat ::= "!{" cons "}" | ε
// Cons ::= pair seqCons | ε
// SeqCons ::= "," pair seqCons | ε
// Pair ::= "(" val "," val ")"
// Var ::= identifier
// Val ::= string
using TokenStream = System.Collections.Generic.IEnumerator<Token>;
class Parser
{
private DSL myDsl;
private TokenStream ts;
public Parser(TokenStream ts) { this.myDsl = new DSL(); this.ts = ts; }
// Dsl ::= stats sat notSat
void Dsl(TokenStream ts)
{
Stats(ts); Sat(); NotSat();
}
// Stats ::= var "= {" vals "}" stats |
void Stats(TokenStream ts)
{
if (ts.Current.kind != Kind.VARIABLE) return;
Var(); Expect(Kind.ASSIGN); Expect(Kind.LEFT_BRACE); Vals(); Expect(Kind.RIGHT_BRACE); Stats(ts);
}
// Vals ::= val seqVals | ε
void Vals()
{
Val();
SeqVals();
return;
}
// SeqVals ::= "," val seqVals | ε
void SeqVals()
{
if (ts.Current.kind != Kind.COMMA) return;
ts.MoveNext();
Val();
SeqVals();
}
// Sat ::= "{" cons "}" | ε
void Sat()
{
if (ts.Current.kind != Kind.LEFT_BRACE) return;
ts.MoveNext();
Cons(myDsl.satisfy);
Expect(Kind.RIGHT_BRACE);
}
// NotSat ::= "!{" cons "}" | ε
void NotSat()
{
if (ts.Current.kind != Kind.EXCLAMATION) return;
ts.MoveNext();
Expect(Kind.LEFT_BRACE);
Cons(myDsl.notSatisfy);
Expect(Kind.RIGHT_BRACE);
}
// Cons ::= pair seqCons | ε
void Cons(Relation r)
{
Pair(r); SeqCons(r);
}
// SeqCons ::= "," pair seqCons | ε
void SeqCons(Relation r)
{
if (ts.Current.kind != Kind.COMMA) return;
ts.MoveNext();
Pair(r);
SeqCons(r);
}
// Pair ::= "(" val "," val ")"
void Pair(Relation r)
{
if (ts.Current.kind != Kind.LEFT_PARENT) return;
Expect(Kind.LEFT_PARENT);
string x = ts.Current.value;
Expect(Kind.VALUE);
Expect(Kind.COMMA);
string y = ts.Current.value;
Expect(Kind.VALUE);
Expect(Kind.RIGHT_PARENT);
r.AddPair(x, y);
}
// Var ::= identifier
void Var()
{
if (ts.Current.kind != Kind.VARIABLE) throw new ApplicationException("Expected " + Kind.VARIABLE + " instead of " + ts.Current.kind);
myDsl.statements.vars.Add(new Variable(ts.Current.value));
ts.MoveNext();
}
// Val ::= string
void Val()
{
if (ts.Current.kind != Kind.VALUE) return;
myDsl.statements.vars[myDsl.statements.vars.Count - 1].values.Add(ts.Current.value);
ts.MoveNext();
}
void Expect(Kind kind)
{
if (ts.Current.kind != kind) throw new ApplicationException("Expected " + kind + " instead of " + ts.Current.kind);
ts.MoveNext();
}
public DSL Parse()
{
ts.MoveNext();
Dsl(ts);
if (ts.Current.kind != Kind.EOF) throw new ApplicationException("Expected end of file instead of " + ts.Current.kind);
return myDsl;
}
}
}