The squash type { A }
provides a
way to assert that A
is true without providing the inhabitant. The
type { A }
contains ()
exactly when A
contains something.
Destructing a hypothesis of type { A }
produces a new hypothesis of
type A
, but that new hypothesis is hidden, which means that it can
be used only in very limited ways.
Hidden variables are a device in the proof engine. They ensure that the hidden variable does not appear in the proof’s computational content. To write a program that refers to unknown inhabitants, one requires proof irrelevance.
We say that a function fn x . M
is parametric if the argument x
does not contribute to the computational or semantic meaning of M
.
We call this notion irrelevance, saying “x
appears only in
irrelevant positions within M
”, or, more briefly, “x
is irrelevant
in M
”. Intuitively, x
is irrelevant in M
when fully unfolding
all the constants in M
and normalizing the result results in a
normal form that does not contain x
.
The type parametric (x : A) . B
contains the parametric dependent
functions from A
to B
. It is introduced by fn
as usual, and
eliminated by parametric application: f Ap m
. Importantly, the
argument in parametric application is an irrelevant position.
Some other irrelevant arguments to important constants are x
and A
in f ap x
, abort x
, M :> A
, and fn (x : A) . M
.
One important use for parametric functions is in impredicativity. The semantic requirements of impredicative quantification demand that an impredicative function’s argument not be available computationally in its body. In Istari’s primitive impredicative functions, the argument is not passed into the “function” at all. However, impredicative functions are isomorphic to parametric functions, which do allow the argument to be manipulated, albeit irrelevantly. (Note that the isomorphism does not mean that parametric functions are impredicative, so we cannot simply dispense with impredicative functions.)
This makes for a similar situation as in the polymorphic lambda
calculus. In the polymorphic lambda calculus, a polymorphic
function’s type argument can only be used in a lambda abstraction’s
type annotation, or in a polymorphic application. In Istari, both of
those (written fn (x : A) . M
and f Ap t
) are irrelevant
positions, and are thus permissible uses of a parametric argument.
Weak sums are a particularly important application of parametric functions. Weak sums are isomorphic to impredicative existentials, but the latter behaves like a union type and can be very difficult to use practically.
The proof irrelevance modality is yet another application. It allows a computation to manipulate unknown (i.e., proof irrelevant) objects.
The introduction rule for parametric functions states:
G |- (fn x . M) : parametric A (fn x . B)
>>
G |- A : type
G |- irrelevant (fn x . M)
G, x : A |- M : B
The second premise states that x
is irrelevant in M
. That is the
case when M
is syntactically equivalent to [unavailable / x]M
.
The typechecker will normally discharge irrelevance premises
automatically.
Consider parametric application of a known function:
(fn x . M) Ap N
This reduces to [N / x]M
, but only when x
is irrelevant in M
.
The reduce
tactic will not do this, as it only handles unconditional
reductions. Instead, use the reduceParam
tactic.
Suppose foo
is a newly defined constant. To tell Istari that its
first and second arguments are irrelevant, call:
recordIrrelevance /foo/ [0, 1];
Istari will check that arguments 0 and 1 are indeed irrelevant, then
record that information in the database. The list of indices must be
strictly increasing. Once the irrelevance is recorded, parametric
variables can be used in foo
’s first and second arguments.