{-# LANGUAGE GADTs #-}
{-# LANGUAGE LinearTypes #-}

-- |
-- Module      : Pure
-- Description : Pure functions showing the basics of linear haskell.
--
-- We have simple linear functions and simple linear data structures that
-- illustrate the basic concepts of how the type checker of GHC with linear
-- types behaves. The goal of this is to be a ridiculously simple tutorial
-- on the basics of linear types.
module Simple.Pure where

-- * Simple linear functions

------------------------------------------------------------

{-
   A linear function simply "consumes/uses" its argument exactly once.

   Giving a more precise idea of this is tricky, so first we present a
   bunch of examples. You should try to get a sense of the arithmatic of how
   many times an argument in a function is used. In other words, you should
   have some idea of a counting function in your head, that can take some
   haskell function f :: A -> B and give a natural number as to the number of
   times the argument of f is used in the body.
-}

linearIdentity :: a %1 -> a
linearIdentity :: forall a. a %1 -> a
linearIdentity a
x = a
x

{-
   DEFINITION.
   ==========
   We say the argument of a function is linear if the arrow
   that follows it is linear.

   Here, the argument is present exactly once in the body and is
   consumed exactly once.
-}

linearSwap :: (a, a) %1 -> (a, a)
linearSwap :: forall a. (a, a) %1 -> (a, a)
linearSwap (a
x, a
y) = (a
y, a
x)

{-
   Here, the argument is decomposed by the tuple data constructor into two
   pieces. Since the whole first argument is linear, the tuple constructor
   in linear haskell is linear -- i.e., notice the type of the
   constructor:

   (,) :: a %1-> b %1-> (a,b)

   Now, this does not mean that if we have some non-linear function with
   an argument (a,b) that `a` and `b` have to be consumed exactly once.

   DEFINITION.
   ==========
   A linear arrow in a data constructor merely signifies that the argument to
   the constructor that preceedes that arrow must be used linearly if the input
   to a linear function is this data type with this constructor.

   Here, since `(,) x y` was the first input to linearSwap, which is a
   linear function (meaning the first input is linear), the components `x`
   and `y` must be used linearly. Indeed, we see the `x` and `y` are each
   used exactly once.

   With a non-linear function, this is not the case; a non-linear function
   `f :: (x,y) -> B` does not have to use the `x` and `y` linearly.
   Consider the next function as an example.
-}

nonLinearSubsume :: (a, a) -> (a, a)
nonLinearSubsume :: forall a. (a, a) -> (a, a)
nonLinearSubsume (a
x, a
_) = (a
x, a
x)

{-
   This function is not linear on its argument and in fact could not have a
   linear arrow. If it did, this file would not compile. Why is this?  Well,
   the first argument would be linear, and the constructor is linear in both
   components.

   (,) :: a %1-> b %1-> (a,b)

   Again, in a linear function, this means the `a` and `b` must be used
   linearly.

   Yet, in the body of the function, `a` is used twice and `b` is used
   zero times.
-}

linearPairIdentity :: (a, a) %1 -> (a, a)
linearPairIdentity :: forall a. (a, a) %1 -> (a, a)
linearPairIdentity (a
x, a
y) = (a
x, a
y)

{-
   Here, notice that `(a,a)` is linear, and since `(,)` is linear
   on both arguments, both `a`s must be used linearly, and indeed
   they are. Each is consumed exactly once in a linear input
   to the constructor `(,) :: a %1-> b %1-> (a,b)`.

   Notice the general pattern: we consumed `(a,b)` linearly by pattern matching
   into `Constructor arg1 ... argn` and consumed the linearly bound arguments
   of the constructor linearly by giving them as arguments to some other
   constructor that is linear on the appropreate arguments.
-}

linearIdentity2 :: a %1 -> a
linearIdentity2 :: forall a. a %1 -> a
linearIdentity2 a
x = a %1 -> a
forall a. a %1 -> a
linearIdentity a
x

{-
   Of course, another way to use an input linearly (or a component of an input)
   is to pass it to a linear function.  Here, since linearIdentity is linear,
   we can be sure the term (linearIdentity x) consumes x exactly once.  Hence,
   all of linearIdentity2 consumes the input x exactly once.

   If we replaced it with the original `id`, this would fail to type check
   because `id` has the non-linear type `a -> a`.  Thus, GHC isn't sure
   that `id` uses its input exactly once.  If `id` doesn't use its input
   exactly once, then linearIdentity2 won't use its input exactly once,
   violating it's type signature.

   Now, this does not mean that merely using a linear function makes the
   use of a variable linear. For instance, both of the two functions below
   use their input exactly twice.
-}

nonLinearPair :: a -> (a, a)
nonLinearPair :: forall a. a -> (a, a)
nonLinearPair a
x = (a %1 -> a
forall a. a %1 -> a
linearIdentity a
x, a %1 -> a
forall a. a %1 -> a
linearIdentity a
x)

nonLinearPair2 :: a -> (a, a)
nonLinearPair2 :: forall a. a -> (a, a)
nonLinearPair2 a
x = (a
x, a %1 -> a
forall a. a %1 -> a
linearIdentity a
x)

{-
   The function below uses its input exactly thrice.
-}

nonLinearTriple :: a -> (a, (a, a))
nonLinearTriple :: forall a. a -> (a, (a, a))
nonLinearTriple a
x = (a %1 -> a
forall a. a %1 -> a
linearIdentity a
x, (a, a) %1 -> (a, a)
forall a. a %1 -> a
linearIdentity (a -> (a, a)
forall a. a -> (a, a)
nonLinearPair2 a
x))

{-

   With several examples in hand, we can now give a more precise way of
   constructively checking that an argument is "consumed exactly once".
   Here's a rough (good enough most of the time) definition:

   DEFINITION.
   ==========
   Let f :: A %1-> B.  Suppose that we don't have the identity, "f x = x"
   which is trivially linear. Then, the thunk (f x) is basically a tree
   composed of function applications, data constructors and case
   statements.

   We say f is linear if for any thunk (f x) ...

   I.
     If x is not a function and is not deconstructed in a case statement:

     case (a): (f x) is some function application (func t1 ... tn) or
               a constructor (Constr t1 ... tn)

       Either (i) exactly one of the t_i is x and the function func or
       constructor Constr is linear on its ith argument, or (ii) x is used
       exactly once in exactly one t_i.

     case (b): (f x) is a case statement (case s of [a_i -> t_i]),

        x is used exactly once in **each** t_i (and not at all in s)

   II.

     If x is a function, then for some argument u, (f u) is used
     exactly once.

   III.

     If x is deconstructed into (Constructor t1 ... tn) in a case statement
     then whatever pieces t_i that are bound linearly by the constuctor,
     must be consumed exactly once.

-}

regularIdentity :: a -> a
regularIdentity :: forall a. a -> a
regularIdentity a
x = a %1 -> a
forall a. a %1 -> a
linearIdentity a
x

{-
   Of course, the fact that a function is linear makes no difference in
   non-linear functions. So, non-linear functions can call linear
   functions willy-nilly and they will work as expected. (Obviously,
   the converse is false, which is kind of the point of linear types.)
   To state the obvious, linear functions are regular functions but not all
   functions are linear functions.
-}

(#.) :: (b %1 -> c) -> (a %1 -> b) -> (a %1 -> c)
b %1 -> c
g #. :: forall b c a. (b %1 -> c) -> (a %1 -> b) -> a %1 -> c
#. a %1 -> b
f = \a
a -> b %1 -> c
g (a %1 -> b
f a
a)

infixr 9 #. -- same fixity as base..

linearCompose :: (a, a) %1 -> (a, a)
linearCompose :: forall a. (a, a) %1 -> (a, a)
linearCompose = (a, a) %1 -> (a, a)
forall a. a %1 -> a
linearIdentity ((a, a) %1 -> (a, a))
-> ((a, a) %1 -> (a, a)) -> (a, a) %1 -> (a, a)
forall b c a. (b %1 -> c) -> (a %1 -> b) -> a %1 -> c
#. (a, a) %1 -> (a, a)
forall a. (a, a) %1 -> (a, a)
linearSwap

{-
   Above, we compose two linear functions and write a linear version of
   `(.)`.  Here, as before, it is critical that we are composing linear
   functions.  Notice that we cannot write a function of the following
   type:

   (##.) :: (b -> c) -> (a %1-> b) -> (a %1-> c)
-}

-- * Linear functions with user data types

------------------------------------------------------------

{-
  As we've seen, we can consume linearly bound inputs into data types if
  the constructor has a linear arrow before the input.
-}

data LinearHolder a where
  LinearHolder :: a %1 -> LinearHolder a

linearHold :: a %1 -> LinearHolder a
linearHold :: forall a. a %1 -> LinearHolder a
linearHold a
x = a -> LinearHolder a
forall a. a -> LinearHolder a
LinearHolder a
x

{-
   Note that if the constructor LinearHolder did not have the %1-> then
   linearHold would not compile, because then you could use the value
   non-linearly.
-}

linearHoldExtract :: LinearHolder a %1 -> a
linearHoldExtract :: forall a. LinearHolder a %1 -> a
linearHoldExtract (LinearHolder a
x) = a
x

linearIdentity3 :: a %1 -> a
linearIdentity3 :: forall a. a %1 -> a
linearIdentity3 = LinearHolder a %1 -> a
forall a. LinearHolder a %1 -> a
linearHoldExtract (LinearHolder a %1 -> a) -> (a %1 -> LinearHolder a) -> a %1 -> a
forall b c a. (b %1 -> c) -> (a %1 -> b) -> a %1 -> c
#. a %1 -> LinearHolder a
forall a. a %1 -> LinearHolder a
linearHold

{-
   For clarity, we include an example of using such linear constructors.
   Here, linearHoldExtract must use the inner component linearly.
   Therefore, it's impossible to implement the following function:

   linearHoldPair :: LinearHolder a %1-> (a,a)
   linearHoldPair (LinearHolder x) = (x,x)

   In fact, we have the following equivalence:

   (LinearHolder a  %1-> b) ≅ (a %1-> b)
-}

data LinearHolder2 where
  LinearHolder2 :: a %1 -> b -> LinearHolder2

linearHold' :: a %1 -> LinearHolder2
linearHold' :: forall a. a %1 -> LinearHolder2
linearHold' a
x = a -> String -> LinearHolder2
forall a b. a -> b -> LinearHolder2
LinearHolder2 a
x String
"hello"

-- linearHold' x = LinearHolder2 "hi" x -- fails to type check

{-
   We can have constructors with mixed arrows, of course.  Here, this
   means only the first value is bound linearly.  This is why the
   commented out line would fail to type check
-}

data ForcedUnlinear a where
  ForcedUnlinear :: a -> ForcedUnlinear a

forcedLinearPair :: ForcedUnlinear a %1 -> (a, a)
forcedLinearPair :: forall a. ForcedUnlinear a %1 -> (a, a)
forcedLinearPair (ForcedUnlinear a
x) = (a
x, a
x)

{-
   Above we define a data type ForcedUnlinear which does not use the
   linear arrow to hold it's argument. This means that even if an input of
   type `ForcedUnlinear a` is linear, the component does not have to be.
   Hence, we can write the function above but could not write something
   the following type:

   linearPair :: a %1-> (a,a)
-}

demote :: (ForcedUnlinear a %1 -> b) -> (a -> b)
demote :: forall a b. (ForcedUnlinear a %1 -> b) -> a -> b
demote ForcedUnlinear a %1 -> b
f a
x = ForcedUnlinear a %1 -> b
f (a -> ForcedUnlinear a
forall a. a -> ForcedUnlinear a
ForcedUnlinear a
x)

promote :: (a -> b) -> (ForcedUnlinear a %1 -> b)
promote :: forall a b. (a -> b) -> ForcedUnlinear a %1 -> b
promote a -> b
f (ForcedUnlinear a
x) = a -> b
f a
x

{-
   Another way of saying this is the following equivalence proven by the
   two functions above:

   (ForcedUnlinear a  %1-> b) ≅ (a -> b)

   In the Linear Haskell POPL '18 paper, this datatype is called
   Unrestricted.
-}