Keep annotations in Parser

This commit is contained in:
Suzanne Soy 2020-08-31 18:38:52 +00:00
parent 89c9908957
commit 592a758fb1
9 changed files with 91 additions and 84 deletions

View File

@ -12,6 +12,10 @@ public static class AstGenerator {
Variant("Expr", Variant("Expr",
Case("int", "Int"), Case("int", "Int"),
Case("string", "String")), Case("string", "String")),
Variant("ParserResult",
Case("MixFix.Annotation", "Annotated"),
Case("Lexer.Lexeme", "Terminal"),
Case("IEnumerable<ParserResult>", "Productions")),
Variant("AstNode", Variant("AstNode",
Case("Expr", "Terminal"), Case("Expr", "Terminal"),
Case("IEnumerable<AstNode>", "Operator")))); Case("IEnumerable<AstNode>", "Operator"))));

View File

@ -7,14 +7,14 @@ using static MixFix.Associativity;
public static class DefaultGrammar { public static class DefaultGrammar {
public static PrecedenceDAG DefaultPrecedenceDAG public static PrecedenceDAG DefaultPrecedenceDAG
= EmptyPrecedenceDAG = EmptyPrecedenceDAG
/*.WithOperator("bool", NonAssociative, "equality|terminal", S.And, "equality|terminal") .WithOperator("bool", NonAssociative, "equality|terminal", S.And, "equality|terminal")
.WithOperator("equality", NonAssociative, "int|terminal|additive|multiplicative", S.Eq, "int|terminal|additive|multiplicative") .WithOperator("equality", NonAssociative, "int|terminal|additive|multiplicative", S.Eq, "int|terminal|additive|multiplicative")
.WithOperator("int", NonAssociative, S.Int) .WithOperator("int", NonAssociative, S.Int)
.WithOperator("additive", LeftAssociative, "int|terminal|multiplicative", S.Plus, "int|terminal|multiplicative") .WithOperator("additive", LeftAssociative, "int|terminal|multiplicative", S.Plus, "int|terminal|multiplicative")
.WithOperator("multiplicative", LeftAssociative, "int|terminal", S.Times, "int|terminal")*/ .WithOperator("multiplicative", LeftAssociative, "int|terminal", S.Times, "int|terminal")
.WithOperator("terminal", NonAssociative, S.Ident) .WithOperator("terminal", NonAssociative, S.Ident)
// This is the root set of operators // This is the root set of operators
.WithOperator("program", NonAssociative, .WithOperator("program", NonAssociative,
// "bool" // TODO: this needs aliases // "bool" // TODO: this needs aliases
"terminal", S.And, "terminal");// TODO: re-add equality| "equality|terminal", S.And, "equality|terminal");
} }

View File

@ -232,11 +232,12 @@ public static partial class Lexer {
IImmutableEnumerator<Lexeme> lx IImmutableEnumerator<Lexeme> lx
) )
=> lx.FirstAndRest().Match<Tuple<Lexeme, IImmutableEnumerator<Lexeme>>, IImmutableEnumerator<Lexeme>>( => lx.FirstAndRest().Match<Tuple<Lexeme, IImmutableEnumerator<Lexeme>>, IImmutableEnumerator<Lexeme>>(
Some: hdtl => Some: hdtl => {
// skip the initial empty whitespace var rest = hdtl.Item2.Lazy(DiscardWhitespace.Eq);
hdtl.Item1.state.Equals(S.Space) return hdtl.Item1.state.Equals(S.Space)
? hdtl.Item2 ? rest
: hdtl.Item1.ImSingleton().Concat(hdtl.Item2.Lazy(DiscardWhitespace.Eq)), : hdtl.Item1.ImSingleton().Concat(rest);
},
None: Empty<Lexeme>()); None: Empty<Lexeme>());
} }

View File

@ -155,7 +155,8 @@ public static partial class MixFix {
Or: l => l.All(g => g.IsEmpty), Or: l => l.All(g => g.IsEmpty),
Sequence: l => l.All(g => g.IsEmpty), Sequence: l => l.All(g => g.IsEmpty),
RepeatOnePlus: g => g.IsEmpty, RepeatOnePlus: g => g.IsEmpty,
Terminal: t => false Terminal: t => false,
Annotated: a => a.Item2.IsEmpty
); );
} }
@ -164,7 +165,8 @@ public static partial class MixFix {
Or: l => l.All(g => g.IsImpossible), Or: l => l.All(g => g.IsImpossible),
Sequence: l => l.Any(g => g.IsImpossible), Sequence: l => l.Any(g => g.IsImpossible),
RepeatOnePlus: g => g.IsImpossible, RepeatOnePlus: g => g.IsImpossible,
Terminal: t => false Terminal: t => false,
Annotated: a => a.Item2.IsImpossible
); );
} }
@ -182,7 +184,8 @@ public static partial class MixFix {
? "EmptySequence" ? "EmptySequence"
: Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(", ")), : Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(", ")),
RepeatOnePlus: g => $"{g.Str()}+", RepeatOnePlus: g => $"{g.Str()}+",
Terminal: t => t.Str() Terminal: t => t.Str(),
Annotated: a => $"Annotated({a.Item1.Str()}, {a.Item2.Str()})"
); );
} }
@ -414,16 +417,27 @@ public static partial class MixFix {
Hole: precedenceGroups => precedenceGroups.ToGrammar1()); Hole: precedenceGroups => precedenceGroups.ToGrammar1());
public static Grammar1 ToGrammar1(this Operator @operator) public static Grammar1 ToGrammar1(this Operator @operator)
=> Grammar1.Sequence( => Grammar1.Annotated(
@operator.internalParts.Select(part => part.ToGrammar1())); (Annotation.Operator(@operator),
Grammar1.Sequence(
@operator.internalParts.Select(
part => part.ToGrammar1()))));
public static Grammar1 ToGrammar1(this IEnumerable<Operator> operators) public static Grammar1 ToGrammar1(this IEnumerable<Operator> operators)
=> Grammar1.Or( => Grammar1.Or(
operators.Select(@operator => @operator.ToGrammar1())); operators.Select(@operator => @operator.ToGrammar1()));
public static Grammar1 ToGrammar1(this DAGNode node) { public static Grammar1 ToGrammar1(this DAGNode node) {
var lsucc = node.leftmostHole_.ToGrammar1(); Func<Associativity, Grammar1, Grammar1> SamePrecedence = (a, g)
var rsucc = node.rightmostHole_.ToGrammar1(); => Grammar1.Annotated((Annotation.SamePrecedence(a), g));
Func<Grammar1, Grammar1> R = g => SamePrecedence(Associativity.RightAssociative, g);
Func<Grammar1, Grammar1> L = g => SamePrecedence(Associativity.LeftAssociative, g);
Func<Grammar1, Grammar1> N = g => SamePrecedence(Associativity.NonAssociative, g);
Func<Grammar1, Grammar1> H = g => Grammar1.Annotated((Annotation.Hole, g));
var Impossible = Grammar1.Impossible;
var lsucc = H(node.leftmostHole_.ToGrammar1());
var rsucc = H(node.rightmostHole_.ToGrammar1());
var closed = node.closed.ToGrammar1(); var closed = node.closed.ToGrammar1();
var nonAssoc = node.infixNonAssociative.ToGrammar1(); var nonAssoc = node.infixNonAssociative.ToGrammar1();
var prefix = node.prefix.ToGrammar1(); var prefix = node.prefix.ToGrammar1();
@ -431,25 +445,13 @@ public static partial class MixFix {
var infixl = node.infixLeftAssociative.ToGrammar1(); var infixl = node.infixLeftAssociative.ToGrammar1();
var infixr = node.infixRightAssociative.ToGrammar1(); var infixr = node.infixRightAssociative.ToGrammar1();
// TODO: BUG: only include these parts if there are
// any operators with that fixity, with the code below
// they are empty.
Func<string, Grammar1, Grammar1> Annotated = (s, g) => Grammar1.Annotated((s, g));
return return
Annotated("closed", closed ? closed : Grammar1.Impossible) // TODO: we can normally remove the ?: checks, as the constructors for grammars
| Annotated("nonAssoc", (nonAssoc ? (lsucc, nonAssoc, rsucc) : Grammar1.Impossible)) // now coalesce Impossible cases in the correct way.
// TODO: post-processsing of the rightassoc list. (closed ? N(closed) : Impossible)
| Annotated("prefix||infixr", ((prefix || infixr) ? ((prefix || (lsucc, infixr))["+"], rsucc) : Grammar1.Impossible)) | (nonAssoc ? N( (lsucc, nonAssoc, rsucc) ) : Impossible)
// TODO: post-processsing of the leftassoc list. | ((prefix || infixr) ? R( ((prefix | (lsucc, infixr))["+"], rsucc) ) : Impossible)
| Annotated("postfix||infixl", ((postfix || infixl) ? (lsucc, (postfix || (infixl, rsucc))["+"]) : Grammar1.Impossible)); | ((postfix || infixl) ? L( (lsucc, (postfix || (infixl, rsucc))["+"]) ) : Impossible);
} }
public static EquatableDictionary<string, Grammar1> ToGrammar1(this PrecedenceDAG precedenceDAG) public static EquatableDictionary<string, Grammar1> ToGrammar1(this PrecedenceDAG precedenceDAG)
@ -469,7 +471,7 @@ public static partial class MixFix {
} }
return recur(labeled[r], labeled); return recur(labeled[r], labeled);
}, },
Annotated: a => recur(a.Item2, labeled), Annotated: a => Grammar2.Annotated((a.Item1, recur(a.Item2, labeled))),
Terminal: t => Grammar2.Terminal(t), Terminal: t => Grammar2.Terminal(t),
Sequence: l => Grammar2.Sequence(l.Select(g => recur(g, labeled))), Sequence: l => Grammar2.Sequence(l.Select(g => recur(g, labeled))),
Or: l => Grammar2.Or(l.Select(g => recur(g, labeled))), Or: l => Grammar2.Or(l.Select(g => recur(g, labeled))),

View File

@ -20,14 +20,20 @@ public static class ParserGenerator {
Case("IEnumerable<Grammar1>", "Or"), Case("IEnumerable<Grammar1>", "Or"),
Case("IEnumerable<Grammar1>", "Sequence"), Case("IEnumerable<Grammar1>", "Sequence"),
Case("S", "Terminal"), Case("S", "Terminal"),
Case("ValueTuple<string, Grammar1>", "Annotated"), Case("ValueTuple<Annotation, Grammar1>", "Annotated"),
Case("string", "Rule")), Case("string", "Rule")),
Variant("Annotation",
Case("MixFix.Operator", "Operator"),
Case("Associativity", "SamePrecedence"),
Case("Hole")),
Variant("Grammar2", Variant("Grammar2",
Case("Grammar2", "RepeatOnePlus"), Case("Grammar2", "RepeatOnePlus"),
Case("IEnumerable<Grammar2>", "Or"), Case("IEnumerable<Grammar2>", "Or"),
Case("IEnumerable<Grammar2>", "Sequence"), Case("IEnumerable<Grammar2>", "Sequence"),
Case("S", "Terminal")), Case("S", "Terminal"),
Case("ValueTuple<Annotation, Grammar2>", "Annotated")),
Variant("Fixity", Variant("Fixity",
Case("Closed"), Case("Closed"),

View File

@ -11,61 +11,44 @@ using Grammar2 = MixFix.Grammar2;
using static Global; using static Global;
public static partial class Parser { public static partial class Parser {
public static Option<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>> Parse3( public static Option<ValueTuple<IImmutableEnumerator<Lexeme>, ParserResult>> Parse3(
Func<IImmutableEnumerator<Lexeme>, Func<IImmutableEnumerator<Lexeme>,
Grammar2, Grammar2,
//Option<IImmutableEnumerator<Lexeme>> //Option<IImmutableEnumerator<Lexeme>>
Option<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>> Option<ValueTuple<IImmutableEnumerator<Lexeme>, ParserResult>>
> >
Parse3, Parse3,
IImmutableEnumerator<Lexeme> tokens, IImmutableEnumerator<Lexeme> tokens,
Grammar2 grammar Grammar2 grammar
) => )
tokens => grammar.Match(
.FirstAndRest() RepeatOnePlus: g =>
.Match( tokens.FoldMapWhileSome(restI => Parse3(restI, g))
None: () => .If((restN, nodes) => nodes.Count() > 1)
//throw new Exception("EOF, what to do?"), .IfSome((restN, nodes) => (restN, ParserResult.Productions(nodes))),
None<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>>(), // TODO: to check for ambiguous parses, we can use
Some: firstRest => { // .Single(…) instead of .First(…).
var first = firstRest.Item1; Or: l =>
var rest = firstRest.Item2; l.First(g => Parse3(tokens, g)),
Log(first.lexeme); Sequence: l =>
Log(grammar.ToString()); l.BindFoldMap(tokens, (restI, g) => Parse3(restI, g))
Log(grammar.Match( .IfSome((restN, nodes) =>
RepeatOnePlus: _ => "RepeatOnePlus", Log($"{nodes.Count()}/{l.Count()}", () =>
Or: _ => "Or", (restN, ParserResult.Productions(nodes)))),
Sequence: _ => "Sequence", Terminal: t =>
Terminal: t => "Terminal:"+t.ToString())); // TODO: move the FirstAndRest here!
return grammar.Match( tokens
RepeatOnePlus: g => .FirstAndRest()
rest.FoldMapWhileSome(restI => Parse3(restI, g)) // When EOF is reached, the parser can't accept this derivation.
.If<IImmutableEnumerator<Lexeme>, IEnumerable<AstNode>>((restN, nodes) => nodes.Count() > 1) .If((first, rest) => first.state.Equals(t))
.IfSome((restN, nodes) => (restN, AstNode.Operator(nodes))), .IfSome((first, rest) => (rest, ParserResult.Terminal(first))),
// TODO: to check for ambiguous parses, we can use Annotated: a =>
// .Single(…) instead of .First(…). // TODO: use the annotation to give some shape to these lists
Or: l => Parse3(tokens, a.Item2));
l.First(g => Parse3(rest, g)), // TODO: at the top-level, check that the lexemes
Sequence: l => { // are empty if the parser won't accept anything else.
return l.BindFoldMap(rest, (restI, g) => Parse3(restI, g))
.IfSome((restN, nodes) => {
Log($"{nodes.Count()}/{l.Count()}");
return (restN, AstNode.Operator(nodes));
});
},
Terminal: t =>
first.state.Equals(t)
? (rest,
AstNode.Terminal(/* TODO: */ Expr.String(rest.ToString())))
.Some()
: None<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>>()
);
// TODO: at the top-level, check that the lexemes
// are empty if the parser won't accept anything else.
}
);
public static Option<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>> Parse2(string source) { public static Option<ValueTuple<IImmutableEnumerator<Lexeme>, ParserResult>> Parse2(string source) {
Grammar2 grammar = Grammar2 grammar =
DefaultGrammar.DefaultPrecedenceDAG.ToGrammar2(); DefaultGrammar.DefaultPrecedenceDAG.ToGrammar2();
//Log(grammar.Str()); //Log(grammar.Str());
@ -73,7 +56,7 @@ public static partial class Parser {
var P = Func.YMemoize< var P = Func.YMemoize<
IImmutableEnumerator<Lexeme>, IImmutableEnumerator<Lexeme>,
Grammar2, Grammar2,
Option<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>>>( Option<ValueTuple<IImmutableEnumerator<Lexeme>, ParserResult>>>(
Parse3 Parse3
); );

View File

@ -28,6 +28,8 @@ public static class Global {
public static T To<T>(this T x) => x; public static T To<T>(this T x) => x;
public static Option<T> If<T>(this bool cond, T x) => x.If(_ => cond);
public static A FoldWhileSome<A>(A init, Func<A, Option<A>> f) public static A FoldWhileSome<A>(A init, Func<A, Option<A>> f)
=> Collection.FoldWhileSome(init, f); => Collection.FoldWhileSome(init, f);

View File

@ -76,6 +76,12 @@ namespace Immutable {
public static Option<ValueTuple<T1, T2>> If<T1, T2>(this ValueTuple<T1, T2> value, Func<T1, T2, bool> predicate) public static Option<ValueTuple<T1, T2>> If<T1, T2>(this ValueTuple<T1, T2> value, Func<T1, T2, bool> predicate)
=> predicate(value.Item1, value.Item2) ? value.Some() : Option.None<ValueTuple<T1, T2>>(); => predicate(value.Item1, value.Item2) ? value.Some() : Option.None<ValueTuple<T1, T2>>();
public static Option<Tuple<T1, T2>> If<T1, T2>(this Tuple<T1, T2> value, Func<T1, T2, bool> predicate)
=> predicate(value.Item1, value.Item2) ? value.Some() : Option.None<Tuple<T1, T2>>();
public static Option<Tuple<T1, T2>> If<T1, T2>(this Option<Tuple<T1, T2>> value, Func<T1, T2, bool> predicate)
=> value.Bind(val => val.If(predicate));
public static Option<U> IfSome<T1, T2, U>(this Option<Tuple<T1, T2>> o, Func<T1, T2, U> some) public static Option<U> IfSome<T1, T2, U>(this Option<Tuple<T1, T2>> o, Func<T1, T2, U> some)
=> o.Map(o1o2 => some(o1o2.Item1, o1o2.Item2)); => o.Map(o1o2 => some(o1o2.Item1, o1o2.Item2));

View File

@ -32,6 +32,9 @@ public static class ToStringImplementations {
public static string Str<T>(this IEnumerable<Ast.AstNode> e) public static string Str<T>(this IEnumerable<Ast.AstNode> e)
=> $"IEnumerab({e.Select(x => x.Str<Ast.AstNode>()).JoinWith(", ")})"; => $"IEnumerab({e.Select(x => x.Str<Ast.AstNode>()).JoinWith(", ")})";
public static string Str<T>(this IEnumerable<Ast.ParserResult> e)
=> $"IEnumerab({e.Select(x => x.Str<Ast.ParserResult>()).JoinWith(", ")})";
public static string Str<Grammar>(this ImmutableDictionary<string,Grammar> h) public static string Str<Grammar>(this ImmutableDictionary<string,Grammar> h)
=> $"ImmutableDictionary(\n{h.Select(x => $" {x.Key.Str<string>()}:{x.Value.Str<Grammar>()}").JoinWith(",\n")}\n)"; => $"ImmutableDictionary(\n{h.Select(x => $" {x.Key.Str<string>()}:{x.Value.Str<Grammar>()}").JoinWith(",\n")}\n)";