Generate grammar from precedence DAG

This commit is contained in:
Suzanne Soy 2020-08-24 02:40:36 +00:00
parent 1cf8cb6bd8
commit 6d5d0dd00b
5 changed files with 119 additions and 19 deletions

101
MixFix.cs
View File

@ -20,6 +20,14 @@ public class Foo {
} }
public static partial class MixFix { public static partial class MixFix {
public partial class Grammar {
public static Grammar Sequence(params Grammar[] xs)
=> Grammar.Sequence(xs.ToImmutableList());
public static Grammar Or(params Grammar[] xs)
=> Grammar.Or(xs.ToImmutableList());
}
public partial class Operator { public partial class Operator {
private string CustomToString() private string CustomToString()
=> $"Operator(\"{precedenceGroup}\", {fixity}, {parts.Select(x => x.Match(Hole: h => h.Select(g => g.ToString()).JoinWith("|"), Name: n => $"\"{n}\"")).JoinWith(", ")})"; => $"Operator(\"{precedenceGroup}\", {fixity}, {parts.Select(x => x.Match(Hole: h => h.Select(g => g.ToString()).JoinWith("|"), Name: n => $"\"{n}\"")).JoinWith(", ")})";
@ -48,6 +56,12 @@ public static partial class MixFix {
get => parts.Last().Bind(lastPart => lastPart.AsHole); get => parts.Last().Bind(lastPart => lastPart.AsHole);
} }
// TODO: does this need any caching?
public IEnumerable<Part> internalParts {
get => parts.SkipWhile(part => part.IsHole)
.SkipLastWhile(part => part.IsHole);
}
private static Func<Fixity> error = private static Func<Fixity> error =
() => throw new Exception("Internal error: unexpected fixity."); () => throw new Exception("Internal error: unexpected fixity.");
@ -110,6 +124,16 @@ public static partial class MixFix {
public Option<ImmutableHashSet<string>> rightmostHole { public Option<ImmutableHashSet<string>> rightmostHole {
get => allOperators.First(@operator => @operator.rightmostHole); get => allOperators.First(@operator => @operator.rightmostHole);
} }
// TODO: cache this for improved complexity
public ImmutableHashSet<string> leftmostHole_ {
get => leftmostHole.Else(ImmutableHashSet<string>.Empty);
}
// TODO: cache this for improved complexity
public ImmutableHashSet<string> rightmostHole_ {
get => rightmostHole.Else(ImmutableHashSet<string>.Empty);
}
} }
public static DAGNode EmptyDAGNode = new DAGNode( public static DAGNode EmptyDAGNode = new DAGNode(
@ -145,7 +169,7 @@ public static partial class MixFix {
existing.IfSome(existingHole => existing.IfSome(existingHole =>
@new.IfSome(newHole => { @new.IfSome(newHole => {
if (! newHole.SetEquals(existingHole)) { if (! newHole.SetEquals(existingHole)) {
throw new ParserExtensionException($"Cannot extend parser with operator {@operator}, its {name} hole ({newHole.ToString()}) must either be empty or else use the same precedence groups as the existing operators in {@operator.precedenceGroup}, i.e. {existingHole}."); throw new ParserExtensionException($"Cannot extend parser with operator {@operator}, its {name} hole ({newHole.ToString()}) must either be absent or else use the same precedence groups as the existing operators in {@operator.precedenceGroup}, i.e. {existingHole}.");
} }
return unit; return unit;
}) })
@ -184,7 +208,7 @@ public static partial class MixFix {
} }
public static void CheckSingleUseOfNames(PrecedenceDAG precedenceDAG, Operator @operator) { public static void CheckSingleUseOfNames(PrecedenceDAG precedenceDAG, Operator @operator) {
// TODO: check that each name part isn't used elsewhere in a way that could cause ambiguity (use in a different namespace which is bracketed is okay, etc.). Probably something to do with paths reachable from the root. // TODO: check that each name part isn't used elsewhere in a way that adding this operator would could cause ambiguity (use in a different namespace which is bracketed is okay, etc.). Probably something to do with paths reachable from the root.
} }
public static void CheckNotAlias(PrecedenceDAG precedenceDAG, Operator @operator) { public static void CheckNotAlias(PrecedenceDAG precedenceDAG, Operator @operator) {
@ -193,6 +217,10 @@ public static partial class MixFix {
} }
} }
public static void CheckAcyclic(PrecedenceDAG precedenceDAG, Operator @operator) {
// TODO: check that the DAG stays acyclic after adding this operator
}
public static PrecedenceDAG With(this PrecedenceDAG precedenceDAG, Operator @operator) { public static PrecedenceDAG With(this PrecedenceDAG precedenceDAG, Operator @operator) {
// This is where all the checks are done to ensure that the // This is where all the checks are done to ensure that the
// resulting grammar is well-formed and assuredly unambiguous. // resulting grammar is well-formed and assuredly unambiguous.
@ -204,6 +232,7 @@ public static partial class MixFix {
CheckConsecutiveHoles(precedenceDAG, @operator); CheckConsecutiveHoles(precedenceDAG, @operator);
CheckSingleUseOfNames(precedenceDAG, @operator); CheckSingleUseOfNames(precedenceDAG, @operator);
CheckNotAlias(precedenceDAG, @operator); CheckNotAlias(precedenceDAG, @operator);
CheckAcyclic(precedenceDAG, @operator);
return precedenceDAG.lens[@operator.precedenceGroup].Add(@operator); return precedenceDAG.lens[@operator.precedenceGroup].Add(@operator);
} }
@ -214,23 +243,63 @@ public static partial class MixFix {
associativity: associativity, associativity: associativity,
parts: parts.ToImmutableList())); parts: parts.ToImmutableList()));
public static Grammar OperatorToGrammar(Operator @operator) { public static Grammar HoleToGrammar(PrecedenceDAG precedenceDAG, Hole precedenceGroups)
//@operator. => Grammar.Or(
precedenceGroups.Select(precedenceGroup =>
DAGNodeToGrammar(precedenceDAG, precedenceDAG[precedenceGroup])));
public static Grammar PartToGrammar(PrecedenceDAG precedenceDAG, Part part)
=> part.Match(
Name: name => Grammar.Terminal(name),
Hole: precedenceGroups => HoleToGrammar(precedenceDAG, precedenceGroups));
public static Grammar OperatorToGrammar(PrecedenceDAG precedenceDAG, Operator @operator)
=> Grammar.Sequence(
@operator.internalParts.Select(
part => PartToGrammar(precedenceDAG, part)));
public static Grammar OperatorsToGrammar(PrecedenceDAG precedenceDAG, IEnumerable<Operator> operators)
=> Grammar.Or(
operators.Select(@operator =>
OperatorToGrammar(precedenceDAG, @operator)));
public static Grammar DAGNodeToGrammar(PrecedenceDAG precedenceDAG, DAGNode node) {
return Grammar.Or(ImmutableList<Grammar>(
OperatorsToGrammar(precedenceDAG, node.closed),
// successor_left nonassoc successor_right
Grammar.Sequence(
HoleToGrammar(precedenceDAG, node.leftmostHole_),
OperatorsToGrammar(precedenceDAG, node.infixNonAssociative),
HoleToGrammar(precedenceDAG, node.rightmostHole_)),
// (prefix | successor_left leftassoc)+ successor_right
// TODO: post-processsing of the leftassoc list.
Grammar.Sequence(
Grammar.RepeatOnePlus(
Grammar.Or(
OperatorsToGrammar(precedenceDAG, node.prefix),
Grammar.Sequence(
HoleToGrammar(precedenceDAG, node.leftmostHole_),
OperatorsToGrammar(precedenceDAG, node.infixRightAssociative)))),
HoleToGrammar(precedenceDAG, node.rightmostHole_)),
// successor_left (posftix | leftassoc successor_right)+
// TODO: post-processsing of the leftassoc list.
Grammar.Sequence(
HoleToGrammar(precedenceDAG, node.leftmostHole_),
Grammar.RepeatOnePlus(
Grammar.Or(
OperatorsToGrammar(precedenceDAG, node.postfix),
Grammar.Sequence(
OperatorsToGrammar(precedenceDAG, node.infixLeftAssociative),
HoleToGrammar(precedenceDAG, node.rightmostHole_)))))
));
throw new NotImplementedException(); throw new NotImplementedException();
} }
public static Grammar DAGNodeToGrammar(DAGNode node) { public static void DAGToGrammar(PrecedenceDAG precedenceDAG) {
// return Grammar.Or(ImmutableList<Grammar>(
// node.closed
// closed,
// succl nonassoc succr,
// (prefix | succl rightassoc)+ succr,
// succl (suffix | succr rightassoc)+
// ));
throw new NotImplementedException();
}
public static void DAGToGrammar(DAGNode precedenceDAG) {
} }

View File

@ -4,7 +4,8 @@ public static class ParserGenerator {
public static void Main() { public static void Main() {
Generate( Generate(
"MixFixGenerated.cs", "MixFixGenerated.cs",
"using System.Collections.Immutable;\n" "using System.Collections.Generic;\n"
+ "using System.Collections.Immutable;\n"
+ "using S = Lexer.S;\n" + "using S = Lexer.S;\n"
+ "using PrecedenceGroupName = System.String;", + "using PrecedenceGroupName = System.String;",
"public static partial class MixFix {", "public static partial class MixFix {",
@ -12,8 +13,10 @@ public static class ParserGenerator {
"MixFix.", "MixFix.",
Types( Types(
Variant("Grammar", Variant("Grammar",
Case("ImmutableList<Grammar>", "Or"), Case("Grammar", "RepeatOnePlus"),
Case("ImmutableList<Grammar>", "Sequence")), Case("IEnumerable<Grammar>", "Or"),
Case("IEnumerable<Grammar>", "Sequence"),
Case("S", "Terminal")),
Variant("Fixity", Variant("Fixity",
Case("Closed"), Case("Closed"),

View File

@ -210,4 +210,29 @@ public static class Collection {
public static bool SetEquals<T>(this ImmutableHashSet<T> a, ImmutableHashSet<T> b) public static bool SetEquals<T>(this ImmutableHashSet<T> a, ImmutableHashSet<T> b)
=> a.All(x => b.Contains(x)) && b.All(x => a.Contains(x)); => a.All(x => b.Contains(x)) && b.All(x => a.Contains(x));
public static (Option<T> firstElement, ImmutableQueue<T> rest) Dequeue<T>(this ImmutableQueue<T> q) {
if (q.IsEmpty) {
return (Option.None<T>(), q);
} else {
T firstElement;
var rest = q.Dequeue(out firstElement);
return (firstElement.Some(), rest);
}
}
public static IEnumerable<T> SkipLastWhile<T>(this IEnumerable<T> e, Func<T, bool> predicate) {
var pending = ImmutableQueue<T>.Empty;
foreach (var x in e) {
if (predicate(x)) {
pending = pending.Enqueue(x);
} else {
while (!pending.IsEmpty) {
yield return pending.Peek();
pending = pending.Dequeue();
}
yield return x;
}
}
}
} }

View File

@ -67,6 +67,9 @@ namespace Immutable {
public static T Else<T>(this Option<T> o, Func<T> none) public static T Else<T>(this Option<T> o, Func<T> none)
=> o.Match_(some => some, none); => o.Match_(some => some, none);
public static T Else<T>(this Option<T> o, T none)
=> o.Match_(some => some, () => none);
public static Option<T> Else<T>(this Option<T> o, Func<Option<T>> none) public static Option<T> Else<T>(this Option<T> o, Func<Option<T>> none)
=> o.Match_(value => value.Some(), none); => o.Match_(value => value.Some(), none);