In System F, the kind of a polymorphic type is *
(as that's the only kind in System F anyway...), so e.g. for the following closed type:
[] ⊢ (forall α : *. α → α) : *
I would like to represent System F in Agda, and because everything is in *
, I thought I'd interpret types (like the above) as Agda Set
s; so something like
evalTy : RepresentationOfAWellKindedClosedType → Set
However, Agda doesn't have polymorphic types, so the above type, in Agda, would need to be a (large!) Π type:
idType = (α : Set) → α → α
which means it is not in Set₀:
idType : Set
idType = (α : Set) → α → α
poly-level.agda:4,12-29
Set₁ != Set
when checking that the expression (α : Set) → α → α has type Set
Is there a way out of this, or is System F not embeddable in this sense into Agda?
Instead of
you can write
(András Kovács, care to add an answer with references to your embedding of predicative System F?)
This is enough for embedding, but I've seen a lot of
Setω
errors and they have traumatised me, so now I'm trying to avoid dependent universes as much as possible.You can embed polymorphic types into
Set₁
and monomorphic types intoSet
, so you can embed any type intoSet₁
using the ugly lifting mechanism. I tried this several times and it always was awful.The thing I would try is to define
evalTy
asand then eliminate it at the type level like this:
You can run this elimination:
Thus you introduce a universe dependency only at the end, when you actually need to compute something.
E.g.
ofSomeSet
is a dependent function, but not a "universally dependent", you can write e.g.f = ofSomeSet ∘ suc
and it's a perfectly typeable expression. This doesn't work with universes dependencies:You can also enhance
[_,_]ᵀ
to make it mappable like I did here, but this all is probably overkill and you should just useNote that I'm talking only about the predicative fragment of System F. Full System F is not embeddable as Dominique Devriese explains in his comments to the question.
However I feel like we can embed more than the predicative fragment, if we first normalize a System F term. E.g.
id [∀ α : *. α → α] id
is not directly embeddable, but after normalization it becomes justid
, which is embeddable.However it should be possible to embed
id [∀ α : *. α → α] id
even without normalization by transforming it intoΛ α. id [α → α] (id [α])
, which is what Agda does with implicit arguments (right?). So it's not clear to me what exactly we can't embed.