summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNadrieril2019-03-24 01:06:40 +0100
committerNadrieril2019-03-24 01:06:40 +0100
commit7d30b044a2c8c2cef8143b9e0ac763024c50026c (patch)
treecf6aa5f9af31edf77ac49a27083f53af2b6d2a30
parent062fc44a93a18ee432e51db852290ab5849f4dd9 (diff)
Refactor printer
Avoids stupid stack overflows when adding variants, gets precedences right, and updates to latest grammar changes
-rw-r--r--dhall_core/src/core.rs32
-rw-r--r--dhall_core/src/label.rs4
-rw-r--r--dhall_core/src/parser.rs4
-rw-r--r--dhall_core/src/printer.rs435
-rw-r--r--dhall_generator/src/lib.rs6
5 files changed, 249 insertions, 232 deletions
diff --git a/dhall_core/src/core.rs b/dhall_core/src/core.rs
index 502a9bc..0ea8c83 100644
--- a/dhall_core/src/core.rs
+++ b/dhall_core/src/core.rs
@@ -102,32 +102,34 @@ pub enum Const {
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct V(pub Label, pub usize);
-#[derive(Debug, Copy, Clone, PartialEq, Eq)]
+// Definition order must match precedence order for
+// pretty-printing to work correctly
+#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord)]
pub enum BinOp {
- /// x && y`
- BoolAnd,
+ /// x ? y
+ ImportAlt,
/// x || y`
BoolOr,
- /// x == y`
- BoolEQ,
- /// x != y`
- BoolNE,
/// x + y`
NaturalPlus,
- /// x * y`
- NaturalTimes,
/// x ++ y`
TextAppend,
+ /// x # y
+ ListAppend,
+ /// x && y`
+ BoolAnd,
/// x ∧ y`
Combine,
- /// x //\\ y
- CombineTypes,
- /// x ? y
- ImportAlt,
/// x // y
Prefer,
- /// x # y
- ListAppend,
+ /// x //\\ y
+ CombineTypes,
+ /// x * y`
+ NaturalTimes,
+ /// x == y`
+ BoolEQ,
+ /// x != y`
+ BoolNE,
}
/// Built-ins
diff --git a/dhall_core/src/label.rs b/dhall_core/src/label.rs
index 8b371af..9dc2816 100644
--- a/dhall_core/src/label.rs
+++ b/dhall_core/src/label.rs
@@ -18,8 +18,8 @@ impl<'a> From<&'a str> for Label {
}
}
-impl From<Label> for String {
- fn from(x: Label) -> String {
+impl From<&Label> for String {
+ fn from(x: &Label) -> String {
x.0.as_ref().to_owned()
}
}
diff --git a/dhall_core/src/parser.rs b/dhall_core/src/parser.rs
index a62f861..3f53c49 100644
--- a/dhall_core/src/parser.rs
+++ b/dhall_core/src/parser.rs
@@ -685,7 +685,7 @@ make_parser! {
rule!(identifier<ParsedExpr> as expression; children!(
[label(l), natural_literal(idx)] => {
- let name = String::from(l.clone());
+ let name = String::from(&l);
match Builtin::parse(name.as_str()) {
Some(b) => bx(Expr::Builtin(b)),
None => match name.as_str() {
@@ -698,7 +698,7 @@ make_parser! {
}
},
[label(l)] => {
- let name = String::from(l.clone());
+ let name = String::from(&l);
match Builtin::parse(name.as_str()) {
Some(b) => bx(Expr::Builtin(b)),
None => match name.as_str() {
diff --git a/dhall_core/src/printer.rs b/dhall_core/src/printer.rs
index d93336e..aa9f707 100644
--- a/dhall_core/src/printer.rs
+++ b/dhall_core/src/printer.rs
@@ -2,269 +2,212 @@ use crate::*;
use itertools::Itertools;
use std::fmt::{self, Display};
-// There used to be a one-to-one correspondence between the formatters in this section
-// and the grammar. Each formatter is named after the
-// corresponding grammar rule and the relationship between formatters exactly matches
-// the relationship between grammar rules. This leads to the nice emergent property
-// of automatically getting all the parentheses and precedences right.
-//
-// WARNING: This approach has one major disadvantage: you can get an infinite loop if
-// you add a new constructor to the syntax tree without adding a matching
-// case the corresponding builder.
-
impl<S, A: Display> Display for Expr<S, A> {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- // buildExprA
- use crate::Expr::*;
- match self {
- &Annot(ref a, ref b) => {
- a.fmt_b(f)?;
- write!(f, " : ")?;
- b.fmt(f)
- }
- &Note(_, ref b) => b.fmt(f),
- a => a.fmt_b(f),
- }
+ self.fmt_phase(f, PrintPhase::Base)
}
}
+// There is a one-to-one correspondence between the formatter and the grammar. Each phase is
+// named after a corresponding grammar group, and the structure of the formatter reflects
+// the relationship between the corresponding grammar rules. This leads to the nice property
+// of automatically getting all the parentheses and precedences right.
+#[derive(Debug, Copy, Clone, Ord, PartialOrd, Eq, PartialEq)]
+enum PrintPhase {
+ Base,
+ Operator,
+ BinOp(core::BinOp),
+ App,
+ Import,
+ Primitive,
+ Paren,
+}
+
impl<S, A: Display> Expr<S, A> {
- fn fmt_b(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
+ fn fmt_phase(
+ &self,
+ f: &mut fmt::Formatter,
+ phase: PrintPhase,
+ ) -> Result<(), fmt::Error> {
use crate::Expr::*;
+ use PrintPhase::*;
match self {
- &Lam(ref a, ref b, ref c) => {
+ _ if phase == Paren => {
+ f.write_str("(")?;
+ self.fmt_phase(f, Base)?;
+ f.write_str(")")?;
+ }
+
+ Lam(a, b, c) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "λ({} : ", a)?;
b.fmt(f)?;
write!(f, ") → ")?;
- c.fmt_b(f)
+ c.fmt(f)?;
}
- &BoolIf(ref a, ref b, ref c) => {
+ BoolIf(a, b, c) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "if ")?;
a.fmt(f)?;
write!(f, " then ")?;
- b.fmt_b(f)?;
+ b.fmt(f)?;
write!(f, " else ")?;
- c.fmt_c(f)
+ c.fmt(f)?;
+ }
+ Pi(a, b, c) if &String::from(a) == "_" => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
+ b.fmt_phase(f, Operator)?;
+ write!(f, " → ")?;
+ c.fmt(f)?;
}
- // TODO: wait for decision on label types
- // &Pi("_", ref b, ref c) => {
- // b.fmt_c(f)?;
- // write!(f, " → ")?;
- // c.fmt_b(f)
- // }
- &Pi(ref a, ref b, ref c) => {
+ Pi(a, b, c) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "∀({} : ", a)?;
b.fmt(f)?;
write!(f, ") → ")?;
- c.fmt_b(f)
- }
- &Let(ref a, None, ref c, ref d) => {
- write!(f, "let {} = ", a)?;
c.fmt(f)?;
- write!(f, " in ")?;
- d.fmt_b(f)
}
- &Let(ref a, Some(ref b), ref c, ref d) => {
- write!(f, "let {} : ", a)?;
- b.fmt(f)?;
+ Let(a, b, c, d) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
+ write!(f, "let {}", a)?;
+ if let Some(b) = b {
+ write!(f, " : ")?;
+ b.fmt(f)?;
+ }
write!(f, " = ")?;
c.fmt(f)?;
write!(f, " in ")?;
- d.fmt_b(f)
+ d.fmt(f)?;
}
- &EmptyListLit(ref t) => {
+ EmptyListLit(t) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "[] : List ")?;
- t.fmt_e(f)
+ t.fmt_phase(f, Import)?;
}
- &NEListLit(ref es) => {
- fmt_list("[", ", ", "]", es, f, |e, f| e.fmt(f))
+ NEListLit(es) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
+ fmt_list("[", ", ", "]", es, f, |e, f| e.fmt(f))?;
}
- &EmptyOptionalLit(ref t) => {
+ EmptyOptionalLit(t) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "None ")?;
- t.fmt_e(f)?;
- Ok(())
+ t.fmt_phase(f, Import)?;
}
- &NEOptionalLit(ref e) => {
+ NEOptionalLit(e) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "Some ")?;
- e.fmt_e(f)?;
- Ok(())
+ e.fmt_phase(f, Import)?;
}
- &Merge(ref a, ref b, ref c) => {
+ Merge(a, b, c) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
write!(f, "merge ")?;
- a.fmt_e(f)?;
+ a.fmt_phase(f, Import)?;
write!(f, " ")?;
- b.fmt_e(f)?;
- match c {
- Some(c) => {
- write!(f, " : ")?;
- c.fmt_d(f)?
- }
- None => {}
+ b.fmt_phase(f, Import)?;
+ if let Some(c) = c {
+ write!(f, " : ")?;
+ c.fmt_phase(f, PrintPhase::App)?;
}
- Ok(())
}
- &Note(_, ref b) => b.fmt_b(f),
- a => a.fmt_c(f),
- }
- }
-
- fn fmt_c(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- use crate::BinOp::*;
- use crate::Expr::*;
- match self {
- // FIXME precedence
- &BinOp(op, ref a, ref b) => {
- write!(f, "(")?;
- a.fmt_d(f)?;
- write!(
- f,
- " {} ",
- match op {
- BoolOr => "||",
- TextAppend => "++",
- NaturalPlus => "+",
- BoolAnd => "&&",
- Combine => "/\\",
- NaturalTimes => "*",
- BoolEQ => "==",
- BoolNE => "!=",
- CombineTypes => "//\\\\",
- ImportAlt => "?",
- Prefer => "//",
- ListAppend => "#",
- }
- )?;
- b.fmt_c(f)?;
- write!(f, ")")
+ Annot(a, b) => {
+ if phase > Base {
+ return self.fmt_phase(f, Paren);
+ }
+ a.fmt_phase(f, Operator)?;
+ write!(f, " : ")?;
+ b.fmt(f)?;
}
- &Note(_, ref b) => b.fmt_c(f),
- a => a.fmt_d(f),
- }
- }
-
- fn fmt_d(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- use crate::Expr::*;
- match self {
- &App(ref a, ref args) => {
- a.fmt_d(f)?;
+ Expr::BinOp(op, a, b) => {
+ // Precedence is magically handled by the ordering of BinOps.
+ if phase >= PrintPhase::BinOp(*op) {
+ return self.fmt_phase(f, Paren);
+ }
+ a.fmt_phase(f, PrintPhase::BinOp(*op))?;
+ write!(f, " {} ", op)?;
+ b.fmt_phase(f, PrintPhase::BinOp(*op))?;
+ }
+ Expr::App(a, args) => {
+ if phase > PrintPhase::App {
+ return self.fmt_phase(f, Paren);
+ }
+ a.fmt_phase(f, Import)?;
for x in args {
f.write_str(" ")?;
- x.fmt_e(f)?;
+ x.fmt_phase(f, Import)?;
}
- Ok(())
- }
- &Note(_, ref b) => b.fmt_d(f),
- a => a.fmt_e(f),
- }
- }
-
- fn fmt_e(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- use crate::Expr::*;
- match self {
- &Field(ref a, ref b) => {
- a.fmt_e(f)?;
- write!(f, ".{}", b)
- }
- &Projection(ref e, ref ls) => {
- e.fmt_e(f)?;
- write!(f, ".")?;
- fmt_list("{ ", ", ", " }", ls, f, |l, f| write!(f, "{}", l))
- }
- &Note(_, ref b) => b.fmt_e(f),
- a => a.fmt_f(f),
- }
- }
-
- fn fmt_f(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- use crate::Expr::*;
- match &self {
- &Var(a) => a.fmt(f),
- &Const(k) => k.fmt(f),
- &Builtin(v) => v.fmt(f),
- &BoolLit(true) => f.write_str("True"),
- &BoolLit(false) => f.write_str("False"),
- &NaturalLit(a) => a.fmt(f),
- &IntegerLit(a) if *a >= 0 => {
- f.write_str("+")?;
- a.fmt(f)
}
- &IntegerLit(a) => a.fmt(f),
- &DoubleLit(a) => {
- let a = f64::from(*a);
- if a == std::f64::INFINITY {
- f.write_str("Infinity")
- } else if a == std::f64::NEG_INFINITY {
- f.write_str("-Infinity")
- } else if a.is_nan() {
- f.write_str("NaN")
- } else {
- let s = format!("{}", a);
- if s.contains("e") || s.contains(".") {
- f.write_str(&s)
- } else {
- write!(f, "{}.0", s)
- }
+ Field(a, b) => {
+ if phase > Import {
+ return self.fmt_phase(f, Paren);
}
+ a.fmt_phase(f, Primitive)?;
+ write!(f, ".{}", b)?;
}
- &TextLit(ref a) => {
- f.write_str("\"")?;
- for x in a.iter() {
- match x {
- InterpolatedTextContents::Text(a) => {
- for c in a.chars() {
- match c {
- '\\' => f.write_str("\\\\"),
- '"' => f.write_str("\\\""),
- '$' => f.write_str("\\$"),
- '\u{0008}' => f.write_str("\\b"),
- '\u{000C}' => f.write_str("\\f"),
- '\n' => f.write_str("\\n"),
- '\r' => f.write_str("\\r"),
- '\t' => f.write_str("\\t"),
- c => write!(f, "{}", c),
- }?;
- }
- }
- InterpolatedTextContents::Expr(e) => {
- f.write_str("${ ")?;
- e.fmt(f)?;
- f.write_str(" }")?;
- }
- }
+ Projection(e, ls) => {
+ if phase > Import {
+ return self.fmt_phase(f, Paren);
}
- f.write_str("\"")?;
- Ok(())
- }
- &RecordType(ref a) if a.is_empty() => f.write_str("{}"),
- &RecordType(ref a) => {
- fmt_list("{ ", ", ", " }", a, f, |(k, t), f| {
- write!(f, "{} : {}", k, t)
- })
- }
- &RecordLit(ref a) if a.is_empty() => f.write_str("{=}"),
- &RecordLit(ref a) => {
- fmt_list("{ ", ", ", " }", a, f, |(k, v), f| {
- write!(f, "{} = {}", k, v)
- })
+ e.fmt_phase(f, Primitive)?;
+ write!(f, ".")?;
+ fmt_list("{ ", ", ", " }", ls, f, |l, f| write!(f, "{}", l))?;
}
- &UnionType(ref a) => {
- fmt_list("< ", " | ", " >", a, f, |(k, v), f| {
- write!(f, "{} : {}", k, v)
- })
+ Var(a) => a.fmt(f)?,
+ Const(k) => k.fmt(f)?,
+ Builtin(v) => v.fmt(f)?,
+ BoolLit(true) => f.write_str("True")?,
+ BoolLit(false) => f.write_str("False")?,
+ NaturalLit(a) => a.fmt(f)?,
+ IntegerLit(a) if *a >= 0 => {
+ f.write_str("+")?;
+ a.fmt(f)?;
}
- &UnionLit(ref a, ref b, ref c) => {
+ IntegerLit(a) => a.fmt(f)?,
+ DoubleLit(a) => a.fmt(f)?,
+ TextLit(a) => a.fmt(f)?,
+ RecordType(a) if a.is_empty() => f.write_str("{}")?,
+ RecordType(a) => fmt_list("{ ", ", ", " }", a, f, |(k, t), f| {
+ write!(f, "{} : {}", k, t)
+ })?,
+ RecordLit(a) if a.is_empty() => f.write_str("{=}")?,
+ RecordLit(a) => fmt_list("{ ", ", ", " }", a, f, |(k, v), f| {
+ write!(f, "{} = {}", k, v)
+ })?,
+ UnionType(a) => fmt_list("< ", " | ", " >", a, f, |(k, v), f| {
+ write!(f, "{} : {}", k, v)
+ })?,
+ UnionLit(a, b, c) => {
f.write_str("< ")?;
write!(f, "{} = {}", a, b)?;
for (k, v) in c {
f.write_str(" | ")?;
write!(f, "{} : {}", k, v)?;
}
- f.write_str(" >")
+ f.write_str(" >")?
}
- &Embed(ref a) => a.fmt(f),
- &Note(_, ref b) => b.fmt_f(f),
- a => write!(f, "({})", a),
+ Embed(a) => a.fmt(f)?,
+ Note(_, b) => b.fmt_phase(f, phase)?,
}
+ Ok(())
}
}
@@ -290,15 +233,87 @@ where
f.write_str(close)
}
+impl<S, A: Display> Display for InterpolatedText<S, A> {
+ fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
+ f.write_str("\"")?;
+ for x in self.iter() {
+ match x {
+ InterpolatedTextContents::Text(a) => {
+ for c in a.chars() {
+ match c {
+ '\\' => f.write_str("\\\\"),
+ '"' => f.write_str("\\\""),
+ '$' => f.write_str("\\$"),
+ '\u{0008}' => f.write_str("\\b"),
+ '\u{000C}' => f.write_str("\\f"),
+ '\n' => f.write_str("\\n"),
+ '\r' => f.write_str("\\r"),
+ '\t' => f.write_str("\\t"),
+ c => write!(f, "{}", c),
+ }?;
+ }
+ }
+ InterpolatedTextContents::Expr(e) => {
+ f.write_str("${ ")?;
+ e.fmt(f)?;
+ f.write_str(" }")?;
+ }
+ }
+ }
+ f.write_str("\"")?;
+ Ok(())
+ }
+}
+
impl Display for Const {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
<Self as fmt::Debug>::fmt(self, f)
}
}
+impl Display for BinOp {
+ fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
+ use crate::BinOp::*;
+ f.write_str(match self {
+ BoolOr => "||",
+ TextAppend => "++",
+ NaturalPlus => "+",
+ BoolAnd => "&&",
+ Combine => "/\\",
+ NaturalTimes => "*",
+ BoolEQ => "==",
+ BoolNE => "!=",
+ CombineTypes => "//\\\\",
+ ImportAlt => "?",
+ Prefer => "//",
+ ListAppend => "#",
+ })
+ }
+}
+
+impl Display for NaiveDouble {
+ fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
+ let v = f64::from(*self);
+ if v == std::f64::INFINITY {
+ f.write_str("Infinity")
+ } else if v == std::f64::NEG_INFINITY {
+ f.write_str("-Infinity")
+ } else if v.is_nan() {
+ f.write_str("NaN")
+ } else {
+ let s = format!("{}", v);
+ if s.contains("e") || s.contains(".") {
+ f.write_str(&s)
+ } else {
+ write!(f, "{}.0", s)
+ }
+ }
+ }
+}
+
impl Display for Label {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- let s = String::from(self.clone());
+ let s = String::from(self);
let is_keyword = |s| match s {
"let" | "in" | "if" | "then" | "else" => true,
_ => false,
@@ -414,7 +429,7 @@ impl Display for Scheme {
impl Display for V {
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
- let V(ref x, ref n) = *self;
+ let V(x, n) = self;
x.fmt(f)?;
if *n != 0 {
write!(f, "@{}", n)?;
diff --git a/dhall_generator/src/lib.rs b/dhall_generator/src/lib.rs
index 328bc3e..ee9af5e 100644
--- a/dhall_generator/src/lib.rs
+++ b/dhall_generator/src/lib.rs
@@ -102,12 +102,12 @@ fn dhall_to_tokenstream_bx(
match ctx.lookup(&s, *n) {
// Non-free variable; interpolates as itself
Some(()) => {
- let s: String = s.clone().into();
+ let s: String = s.into();
quote! { bx(Var(V(#s.into(), #n))) }
}
// Free variable; interpolates as a rust variable
None => {
- let s: String = s.clone().into();
+ let s: String = s.into();
// TODO: insert appropriate shifts ?
let v: TokenStream = s.parse().unwrap();
quote! { {
@@ -134,7 +134,7 @@ fn binop_to_tokenstream(b: BinOp) -> TokenStream {
}
fn label_to_tokenstream(l: &Label) -> TokenStream {
- let l = String::from(l.clone());
+ let l = String::from(l);
quote! { #l.into() }
}