From 3ca81af2005a2269fbbe441f5c60e32895e3a4e3 Mon Sep 17 00:00:00 2001
From: Max New <maxsnew@gmail.com>
Date: Wed, 25 Jul 2018 09:15:31 -0400
Subject: [PATCH] some thoughts on polymorphism

---
 notes/polymorphic.org | 92 +++++++++++++++++++++++++++++++++++++++++++
 1 file changed, 92 insertions(+)
 create mode 100644 notes/polymorphic.org

diff --git a/notes/polymorphic.org b/notes/polymorphic.org
new file mode 100644
index 0000000..8af025f
--- /dev/null
+++ b/notes/polymorphic.org
@@ -0,0 +1,92 @@
+* Polymorphic GTT
+
+** Summary
+
+It seems like it's possibly *impossible* to combine the following
+properties:
+
+1. Graduality
+2. polymorphic casts to/from the dynamic type
+3. A safe type case for the dynamic type (i.e., intro/elim, beta,eta
+   for the dynamic type)
+4. Curry-style forall/exists, 
+5. Parametricity
+
+For the first 4, the following each seem to be valid workarounds, but
+each violates a principle of gradual/dynamic typing:
+1. Weaken graduality so we only have retraction, not projection
+2. Don't allow casting of polymorphic type variables, i.e., unknown
+   types are not "gradual"
+3. Disallow type case as a construct
+4. Use Church-style forall/exists, i.e., have the use of polymorphism
+   be *manifest* in the dynamic language, and instantiation is not
+   less than substitution.
+5. Parametricity theorems only apply to totally typed programs
+
+** Details
+First, we work in a GTT-like setting.
+Next, we consider the precision rules for incorporating polymorphism.
+
+First, we need to generalize type dynamism to open type dynamism.
+The following principles seem sensible for now
+1. Previous rules for GTT dynamism are *stable/fibred*: i.e., they
+   apply in any context. In particular, the dynamic type is the most
+   dynamic type *in any context*, meaning for instance that
+   X |- X <= ?
+2. Monotonicity: if X |- A <= B then for any A' <= B', A[A'/X] <=
+   B[B'/X]
+4. Functoriality: if X |- A <= B then forall X. A <= forall X. B
+5. Curry-style: if X |- A, then for any B, \forall X. A <= A[B/X]
+
+Contrast with the church-style rule which would be
+
+5. Church-style: (forall X. ?) <= ?
+
+The argument that something goes wrong comes down to what happens when
+we run type case when we've abstracted over some type variables?
+
+In particular what does
+
+    /\X. \x -> tycase (in_X x) { number? -> x + 1 | else. 0 } : forall X. X -> Num
+
+do?
+If instantiation is less dynamic than we know that choosing X = Num should get us:
+
+   (/\X. \x -> tycase (in_X x) { number? -> 1 | else. 0 }) [Num]
+   <=
+   \x -> tycase (in_Num x) { number? -> 1 | else. 0 }
+
+calling both sides with a number tells us that the LHS must be less
+dynamic than `ret 1`, so either it is equal to it or raises an error.
+On the other hand, instantiating X = String (really anything that
+doesn't satisfy number?) should get us that the LHS must be less
+dynamic than `ret 0`, but the parametricity theorem says that they
+need to produce the same answer, so they must both be error.
+
+on the other hand, consider the following slightly modified version:
+
+   (/\ X. \x -> tycase (in_X x) { number? -> 0 | else. 0 })
+
+should this raise an error?
+
+The runtime-sealing interpretation certainly would raise an error, but
+an eta principle for the dynamic type would say that this should be equivalent to
+
+   (/\ X. \x -> 0)
+
+i.e., that type-case is a *safe* operation, which it certainly is in
+a typical dynamically typed language.
+
+But how could we accomodate these semantics?
+It would seem to require that we run *both* sides of the computation
+and dynamically monitor them for equivalence, and that we would have
+to do this for every polymorphic type-case, possibly causing a huge
+runtime-overhead.
+
+This is possibly of interest as some kind of testing tool, but it's
+certainly *not* what sealing *does*, so we are left to ponder *why*
+sealing works the way it does, since it seems *unsound*: it rejects
+programs at runtime that are *semantically* polymorphic, but not
+satisfying the weirdly rigid notion of *syntactic* polymorphism: never
+interacting with a value in any way.
+
-- 
GitLab