Skip to content

Commit

Permalink
Change to accept syms without parens and to declare multiple names
Browse files Browse the repository at this point in the history
Syntax of names is changed such that in simple declaration and binding contexts
the parentheses around a symbol can be omitted.  For example, previously one
would write:

    {(+), (-)} = Int
    (+): int -> int -> int

Now one can also write:

    {+, -} = Int
    + : int -> int -> int

Parentheses are still required when declaring symbol named functions:

    type (|) a b = ;; ...
    (+) x y = ;; ...

Of course, the infix notation can still also be used:

    type a | n = ;; ...
    x + y = ;; ...

Syntax of declarations is also changed such that multiple names can be declared
with the same type.  Previously one would write:

    (+): int -> int -> int
    (-): int -> int -> int

Now one can also write:

    + - : int -> int -> int

To make this possible, only implicit type parameters are allowed before the
colon.  Previously one could write:

    id a: a -> a

Now one must write:

    id: (a : type) -> a -> a

Implicit type parameters are still allowed before the colon:

    id 'a: a -> a

As explicit type parameters preceding the colon are a relic from 1ML without
type inference this should not cause major inconvenience.

Syntax of paths (namelists) is also changed to allow symbols without
parentheses.
  • Loading branch information
polytypic committed Jul 28, 2020
1 parent 34c712c commit 942793c
Show file tree
Hide file tree
Showing 13 changed files with 186 additions and 147 deletions.
2 changes: 1 addition & 1 deletion basis/index.1ml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ List = import "./list"

type ORD = {
type t
(<=) : t -> t ~> Bool.t
<= : t -> t ~> Bool.t
}

type SET = {
Expand Down
2 changes: 1 addition & 1 deletion examples/fc.1ml
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ GADTs = {
AssociatedTypes = {
type EQ = {
type t
(==): t -> t -> bool
== : t -> t -> bool
}

type COLLECTS = {
Expand Down
2 changes: 1 addition & 1 deletion examples/leibniz.1mls
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ type t a b
type a ~ b = t a b

id 'a: a ~ a ;; Reflexivity
(<<) 'a 'b 'c: b ~ c -> a ~ b -> a ~ c ;; Transitivity
<< 'a 'b 'c: b ~ c -> a ~ b -> a ~ c ;; Transitivity
invert 'a 'b: a ~ b -> b ~ a ;; Symmetry

to 'a 'b: a ~ b -> a -> b
Expand Down
10 changes: 5 additions & 5 deletions examples/type-level.1ml
Original file line number Diff line number Diff line change
Expand Up @@ -28,8 +28,8 @@ TypeLevel: {

not: t -> t

(&&&): t -> t -> t
(|||): t -> t -> t
&&& : t -> t -> t
||| : t -> t -> t

equals: t -> t -> t
}
Expand All @@ -56,15 +56,15 @@ TypeLevel: {

succ: t -> t

(+): t -> t -> t
(*): t -> t -> t
+ : t -> t -> t
* : t -> t -> t
}

List: {
type t = type -> (type _ _ _) -> type

nil: t
(::): type -> t -> t
:: : type -> t -> t

map: (type _ _) -> t -> t

Expand Down
4 changes: 2 additions & 2 deletions lexer.mll
Original file line number Diff line number Diff line change
Expand Up @@ -253,7 +253,7 @@ let escape = ['n''t''\\''\'''\"']
let character = [^'"''\\''\n'] | '\\'escape

let num = digit+
let name = (letter | '_') (letter | digit | '_' | tick)*
let word = (letter | '_') (letter | digit | '_' | tick)*
let text = '"'character*'"'
let char = '\''character '\''

Expand Down Expand Up @@ -298,7 +298,7 @@ rule token = parse
| "}" { RBRACE }
| "," { COMMA }
| ";" { SEMI }
| name as s { NAME s }
| word as s { WORD s }
| symbol* as s { SYM s }
| num as s { NUM (convert_num s) }
| char as s { CHAR (convert_char s) }
Expand Down
10 changes: 5 additions & 5 deletions paper.1ml
Original file line number Diff line number Diff line change
Expand Up @@ -60,9 +60,9 @@ type EQ = {
type MAP = {
type key
type map a
empty (a : type) : map a
add (a : type) : key -> a -> map a ~> map a
lookup (a : type) : key -> map a ~> opt a
empty : (a : type) -> map a
add : (a : type) -> key -> a -> map a ~> map a
lookup : (a : type) -> key -> map a ~> opt a
}

Map (Key : EQ) :> MAP with (type key = Key.t) = {
Expand Down Expand Up @@ -101,8 +101,8 @@ entries (c : type) (C : COLL c) (xs : c) : list (C.key, C.val) =
List.map (C.keys xs) (fun (k : C.key) => (k, caseopt (C.lookup xs k) bot id))

type MONAD (m : type -> type) = {
return (a : type) : a ~> m a
bind (a : type) (b : type) : m a -> (a ~> m b) ~> m b
return : (a : type) -> a ~> m a
bind : (a : type) -> (b : type) -> m a -> (a ~> m b) ~> m b
}
map (a : type) (b : type) (m : type -> type) (M : MONAD m) (f : a ~> b) (mx : m a) =
M.bind a b mx (fun (x : a) => M.return b (f x)) ;; : m b
Expand Down
163 changes: 111 additions & 52 deletions parser.mly
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ let parse_error s = raise (Source.Error (Source.nowhere_region, s))

%token EOF

%token<string> NAME
%token<string> WORD
%token<string> SYM
%token<string> TEXT
%token<char> CHAR
Expand All @@ -54,36 +54,80 @@ let parse_error s = raise (Source.Error (Source.nowhere_region, s))

%%

label :
| name
{ $1 }
| sym
{ $1 }
| NUM
{ index($1)@@at() }
word :
| WORD
{ $1@@at() }
;

hole :
| HOLE
{ "_"@@at() }
;

sym :
| SYM
{ $1@@at() }
;
name :
| NAME
{ $1@@at() }
psym :
| LPAR sym RPAR
{ $2 }
;
namelist :
| name
{ $1::[] }
| name DOT namelist
{ $1::$3 }

pname :
| word
{ $1 }
| psym
{ $1 }
;

name :
| word
{ $1 }
| psym
{ $1 }
| sym
{ $1 }
;

phead :
| word
{ $1 }
| psym
{ $1 }
| hole
{ $1 }
;

head :
| word
{ $1 }
| psym
{ $1 }
| hole
{ $1 }
| sym
{ $1 }
;

names :
| name
{ [$1] }
| name names
{ $1::$2 }
;

label :
| name
{ $1 }
| HOLE
{ "_"@@at() }
| NUM
{ index($1)@@at() }
;

path :
| label
{ $1::[] }
| label DOT path
{ $1::$3 }
;

infixtyp :
Expand All @@ -95,20 +139,20 @@ infixtyp :
typdec :
| infixtyp
{ $1 }
| TYPE name typparamlist
| TYPE pname typparamlist
{ ($2, $3) }
;
typpat :
| infixtyp
{ $1 }
| TYPE head typparamlist
| TYPE phead typparamlist
{ ($2, $3) }
;

implparam :
| TICK LPAR head COLON TYPE RPAR
{ (headB($3)@@ati 3, TypT@@ati 5, Impl@@ati 1)@@at() }
| TICK LPAR TYPE head RPAR
| TICK LPAR TYPE phead RPAR
{ (headB($4)@@ati 4, TypT@@ati 3, Impl@@ati 1)@@at() }
| TICK head
{ (headB($2)@@ati 2, TypT@@ati 1, Impl@@ati 1)@@at() }
Expand Down Expand Up @@ -185,9 +229,9 @@ attyp :
withtyp :
| infpathexp
{ pathT($1)@@at() }
| withtyp WITH LPAR namelist typparamlist EQUAL exp RPAR
| withtyp WITH LPAR path typparamlist EQUAL exp RPAR
{ WithT($1, $4, funE($5, $7)@@span[ati 5; ati 7])@@at() }
| withtyp WITH LPAR TYPE namelist typparamlist EQUAL typ RPAR
| withtyp WITH LPAR TYPE path typparamlist EQUAL typ RPAR
{ WithT($1, $5, funE($6, typE($8)@@ati 8)@@span[ati 6; ati 8])@@at() }
;
typ :
Expand Down Expand Up @@ -228,13 +272,22 @@ opttypdef :
;

atdec :
| name typparamlist COLON typ
{ VarD($1, funT($2, $4, Pure@@ati 2)@@span[ati 2; ati 4])@@at() }
| name typparamlist EQUAL exp
{ VarD($1, funT($2, EqT($4)@@ati 4, Pure@@ati 3)@@span[ati 2; ati 4])
@@at() }
| name
{ VarD($1, funT([], EqT(VarE($1)@@ati 1)@@ati 1, Pure@@ati 1)@@ati 1)@@at() }
| names implparamlist COLON typ
{ let r = span[ati 2; ati 4] in
let v = var "t" @@r in
let t = pathT(VarE(v)@@r)@@r in
letD(VarB(v, typE(funT($2, $4, Pure@@ati 2)@@r)@@r)@@r,
($1 |> List.map (fun n -> VarD(n, t)@@at()) |> seqDs)@@at())@@at() }
| names implparamlist EQUAL exp
{ let r = span[ati 2; ati 4] in
let v = var "t" @@r in
let t = pathT(VarE(v)@@r)@@r in
letD(VarB(v, typE(funT($2, EqT($4)@@ati 4, Pure@@ati 3)@@r)@@r)@@r,
($1 |> List.map (fun n -> VarD(n, t)@@at()) |> seqDs)@@at())@@at() }
| names
{ ($1
|> List.map (fun n -> VarD(n, EqT(VarE(n)@@n.at)@@n.at)@@n.at)
|> seqDs)@@at() }
| typdec optannot opttypdef
{ VarD(fst $1,
funT(snd $1,
Expand Down Expand Up @@ -272,7 +325,7 @@ dotpathexp :
{ DotE($1, $3)@@at() }
;
atpathexp :
| name
| pname
{ VarE($1)@@at() }
| HOLE
{ typE(HoleT@@at())@@at() }
Expand Down Expand Up @@ -307,10 +360,8 @@ dotexp :
{ DotE($1, $3)@@at() }
;
atexp :
| name
{ VarE($1)@@at() }
| HOLE
{ typE(HoleT@@at())@@at() }
| atpathexp
{ $1 }
| PRIMITIVE TEXT
{ match Prim.fun_of_string $2 with
| Some f -> PrimE(Prim.FunV f)@@at()
Expand Down Expand Up @@ -395,7 +446,7 @@ explist :
;

funbind :
| head param paramlist
| phead param paramlist
{ ($1, $2::$3) }
| oneexplparam sym oneexplparam
{ ($2, $1 @ $3) }
Expand Down Expand Up @@ -437,7 +488,7 @@ bindanns_opt :
atbind :
| funbind bindanns_opt EQUAL exp
{ let (h, ps) = $1 in VarB(h, funE(ps, $2($4))@@at())@@at() }
| atpat bindanns_opt EQUAL exp
| bindpat bindanns_opt EQUAL exp
{ patB($1, $2($4))@@at() }
| name
{ VarB($1, VarE($1.it@@at())@@at())@@at() }
Expand Down Expand Up @@ -469,8 +520,15 @@ bind :
{ letB($2, $4)@@at() }
;

bindpat :
| atpat
{ $1 }
| sym
{ headP $1@@at() }
;

atpat :
| head
| phead
{ headP $1@@at() }
| LBRACE decon RBRACE
{ strP($2, at())@@at() }
Expand Down Expand Up @@ -515,20 +573,21 @@ patlist :
;

atdecon :
| name optannot EQUAL pat
{ ($1, opt annotP($4, $2)@@span[ati 2; ati 4])@@at() }
| name optannot
{ ($1, opt annotP(varP($1)@@ati 1, $2)@@at())@@at() }
| name typparam typparamlist COLON typ
{ ($1, annotP(varP($1)@@$1.at,
funT($2::$3, $5, Pure@@at())@@at())@@at())@@at() }
| names optannot EQUAL pat
{ let p = opt annotP($4, $2)@@span[ati 2; ati 4] in
$1 |> List.map (fun n -> (n, p)@@at()) }
| names optannot
{ $1 |> List.map (fun n -> (n, opt annotP(varP(n)@@n.at, $2)@@at())@@at()) }
| names implparam implparamlist COLON typ
{ let t = funT($2::$3, $5, Pure@@at())@@at() in
$1 |> List.map (fun n -> (n, annotP(varP(n)@@n.at, t)@@at())@@at()) }
| typdec optannot
{ let (n, tps) = $1 in
(n,
annotP(varP(n)@@n.at,
funT(tps,
Lib.Option.value $2 ~default:(TypT@@at()),
Pure@@at())@@at())@@at())@@at() }
[(n,
annotP(varP(n)@@n.at,
funT(tps,
Lib.Option.value $2 ~default:(TypT@@at()),
Pure@@at())@@at())@@at())@@at()] }
/*
| LPAR decon RPAR
{ $2 }
Expand All @@ -538,9 +597,9 @@ decon :
|
{ [] }
| atdecon
{ [$1] }
{ $1 }
| atdecon COMMA decon
{ $1::$3 }
{ $1@$3 }
;

prog :
Expand Down
10 changes: 5 additions & 5 deletions prelude/index.1ml
Original file line number Diff line number Diff line change
Expand Up @@ -58,17 +58,17 @@ List = {
...rec {type t _} => {type t x = Opt.t (x, t x)}
nil = Opt.none
hd :: tl = Opt.some (hd, tl)
case {nil, (::)} = Opt.case {
case {nil, ::} = Opt.case {
none = nil
some (hd, tl) = hd :: tl
}
}

{either, inl, inr, plus} = Alt
{false, not, true} = Bool
{(<<), (<|), (>>), (|>), bot, const, curry, flip, id, uncurry} = Fun
{(%), (*), (+), (-), (/), (<), (<=), (<>), (==), (>), (>=)} = Int
{(::), nil} = List
{<<, <|, >>, |>, bot, const, curry, flip, id, uncurry} = Fun
{%, *, +, -, /, <, <=, <>, ==, >, >=} = Int
{::, nil} = List
{none, some} = Opt
{cross, fork, fst, snd} = Pair
{(++), print} = Text
{++, print} = Text
Loading

0 comments on commit 942793c

Please sign in to comment.