-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCheckVisitor.java
123 lines (115 loc) · 2.92 KB
/
CheckVisitor.java
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
123
package edu.lsu.cct.piraha;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class CheckVisitor extends Visitor {
Map<String,Boolean> patterns;
CheckVisitor child = null;
List<Boolean> canBeZero = new ArrayList<Boolean>();
String checking;
boolean retry = true;
public Map<String, Boolean> defaults;
public CheckVisitor() {
patterns = new HashMap<String,Boolean>();
}
private CheckVisitor(Map<String,Boolean> patterns) {
this.patterns = patterns;
}
Boolean getDefault(String key) {
if(defaults != null && defaults.containsKey(key)) {
return defaults.get(key);
}
return Boolean.FALSE;
}
@Override
public Visitor startVisit(Pattern p) {
if(p instanceof Multi) {
CheckVisitor cv = new CheckVisitor(patterns);
child = cv;
cv.checking = checking;
return cv;
} else if(p instanceof Or) {
CheckVisitor cv = new CheckVisitor(patterns);
child = cv;
cv.checking = checking;
return cv;
} else if(p instanceof Seq) {
CheckVisitor cv = new CheckVisitor(patterns);
child = cv;
cv.checking = checking;
return cv;
} else if(p instanceof Lookup) {
CheckVisitor cv = new CheckVisitor(patterns);
child = cv;
Lookup ll = (Lookup)p;
cv.checking = ll.lookup;
return cv;
}
return this;
}
private boolean andZero() {
for(Boolean b : child.canBeZero) {
if(!b)
return false;
}
return true;
}
private boolean orZero() {
for(Boolean b : child.canBeZero) {
if(b)
return true;
}
return false;
}
@Override
public void finishVisit(Pattern p) {
if(p instanceof Multi) {
Multi m = (Multi)p;
if(m.max > 1 && andZero()) {
System.out.println(child.canBeZero);
System.out.println(patterns);
//p.visit(new DebugVisitor());
throw new ParseException(checking+
": cannot have zero length pattern in quantifier: "+p.decompile());
}
if(m.min==0)
canBeZero.add(Boolean.TRUE);
else
canBeZero.add(andZero());
} else if(p instanceof Nothing) {
canBeZero.add(Boolean.TRUE);
} else if(p instanceof LookAhead) {
canBeZero.add(Boolean.TRUE);
} else if(p instanceof NegLookAhead) {
canBeZero.add(Boolean.TRUE);
} else if(p instanceof End) {
canBeZero.add(Boolean.TRUE);
} else if(p instanceof Start) {
canBeZero.add(Boolean.TRUE);
} else if(p instanceof Seq) {
//p.visit(new DebugVisitor());
canBeZero.add(andZero());
} else if(p instanceof Or) {
canBeZero.add(orZero());
} else if(p instanceof Lookup) {
Lookup l = (Lookup)p;
if(patterns.containsKey(l.lookup)) {
canBeZero.add(patterns.get(l.lookup));
} else {
Boolean defaultValue = getDefault(l.lookup);
patterns.put(l.lookup,defaultValue);
l.pattern.visit(child);
boolean res = andZero();
if(res) {
patterns.put(l.lookup,Boolean.TRUE);
}
if(defaultValue ^ res)
retry = true;
canBeZero.add(res);
}
} else {
canBeZero.add(Boolean.FALSE);
}
}
}