diff --git a/tool/src/org/antlr/v4/codegen/model/ElementFrequenciesVisitor.java b/tool/src/org/antlr/v4/codegen/model/ElementFrequenciesVisitor.java index 245e0e29b..d05a98fc8 100644 --- a/tool/src/org/antlr/v4/codegen/model/ElementFrequenciesVisitor.java +++ b/tool/src/org/antlr/v4/codegen/model/ElementFrequenciesVisitor.java @@ -11,7 +11,6 @@ import org.antlr.v4.misc.FrequencySet; import org.antlr.v4.misc.MutableInt; import org.antlr.v4.parse.GrammarTreeVisitor; -import org.antlr.v4.tool.ErrorManager; import org.antlr.v4.tool.Grammar; import org.antlr.v4.tool.ast.ActionAST; import org.antlr.v4.tool.ast.AltAST; @@ -97,7 +96,7 @@ protected static FrequencySet combineMin(FrequencySet a, Frequen } assert a != SENTINEL; - FrequencySet result = combineAndClip(a, b, 1); + FrequencySet result = combineAndClip(a, b, Integer.MAX_VALUE); for (Map.Entry entry : result.entrySet()) { entry.getValue().v = Math.min(a.count(entry.getKey()), b.count(entry.getKey())); } @@ -185,6 +184,31 @@ protected void exitElement(GrammarAST tree) { minFrequencies.push(combineAndClip(minFrequencies.pop(), minFrequencies.pop(), 2)); } + @Override + protected void enterBlockSet(GrammarAST tree) { + frequencies.push(new FrequencySet()); + minFrequencies.push(new FrequencySet()); + } + + @Override + protected void exitBlockSet(GrammarAST tree) { + for (Map.Entry entry : frequencies.peek().entrySet()) { + // This visitor counts a block set as a sequence of elements, not a + // sequence of alternatives of elements. Reset the count back to 1 + // for all items when leaving the set to ensure duplicate entries in + // the set are treated as a maximum of one item. + entry.getValue().v = 1; + } + + if (minFrequencies.peek().size() > 1) { + // Everything is optional + minFrequencies.peek().clear(); + } + + frequencies.push(combineAndClip(frequencies.pop(), frequencies.pop(), 2)); + minFrequencies.push(combineAndClip(minFrequencies.pop(), minFrequencies.pop(), 2)); + } + @Override protected void exitSubrule(GrammarAST tree) { if (tree.getType() == CLOSURE || tree.getType() == POSITIVE_CLOSURE) { @@ -193,7 +217,7 @@ protected void exitSubrule(GrammarAST tree) { } } - if (tree.getType() == CLOSURE) { + if (tree.getType() == CLOSURE || tree.getType() == OPTIONAL) { // Everything inside a closure is optional, so the minimum // number of occurrences for all elements is 0. minFrequencies.peek().clear(); diff --git a/tool/src/org/antlr/v4/codegen/model/RuleFunction.java b/tool/src/org/antlr/v4/codegen/model/RuleFunction.java index 1364ae092..9d902a66c 100644 --- a/tool/src/org/antlr/v4/codegen/model/RuleFunction.java +++ b/tool/src/org/antlr/v4/codegen/model/RuleFunction.java @@ -194,9 +194,10 @@ public void fillNamedActions(OutputModelFactory factory, Rule r) { */ public Set getDeclsForAllElements(List altASTs) { Set needsList = new HashSet(); - Set optional = new HashSet(); + Set nonOptional = new HashSet(); Set suppress = new HashSet(); List allRefs = new ArrayList(); + boolean firstAlt = true; for (AltAST ast : altASTs) { IntervalSet reftypes = new IntervalSet(RULE_REF, TOKEN_REF); List refs = ast.getNodesWithType(reftypes); @@ -209,13 +210,23 @@ public Set getDeclsForAllElements(List altASTs) { if (altFreq.count(refLabelName)==0) { suppress.add(refLabelName); } - if (minFreq.count(refLabelName) == 0) { - optional.add(refLabelName); - } + if ( altFreq.count(refLabelName)>1 ) { needsList.add(refLabelName); } + + if (firstAlt && minFreq.count(refLabelName) != 0) { + nonOptional.add(refLabelName); + } } + + for (String ref : nonOptional.toArray(new String[nonOptional.size()])) { + if (minFreq.count(ref) == 0) { + nonOptional.remove(ref); + } + } + + firstAlt = false; } Set decls = new LinkedHashSet(); for (GrammarAST t : allRefs) { @@ -226,7 +237,7 @@ public Set getDeclsForAllElements(List altASTs) { List d = getDeclForAltElement(t, refLabelName, needsList.contains(refLabelName), - optional.contains(refLabelName)); + !nonOptional.contains(refLabelName)); decls.addAll(d); } return decls;