Coercion #
Lean uses a somewhat elaborate system of typeclasses to drive the coercion system. Here a coercion means an invisible function that is automatically inserted to fix what would otherwise be a type error. For example, if we have:
def f (x : Nat) : Int := x
then this is clearly not type correct as is, because x
has type Nat
but
type Int
is expected, and normally you will get an error message saying exactly that.
But before it shows that message, it will attempt to synthesize an instance of
CoeT Nat x Int
, which will end up going through all the other typeclasses defined
below, to discover that there is an instance of Coe Nat Int
defined.
This instance is defined as:
instance : Coe Nat Int := ⟨Int.ofNat⟩
so Lean will elaborate the original function f
as if it said:
def f (x : Nat) : Int := Int.ofNat x
which is not a type error anymore.
You can also use the ↑
operator to explicitly indicate a coercion. Using ↑x
instead of x
in the example will result in the same output.
Because there are many polymorphic functions in Lean, it is often ambiguous where
the coercion can go. For example:
def f (x y : Nat) : Int := x + y
This could be either ↑x + ↑y
where +
is the addition on Int
, or ↑(x + y)
where +
is addition on Nat
, or even x + y
using a heterogeneous addition
with the type Nat → Nat → Int
. You can use the ↑
operator to disambiguate
between these possibilities, but generally Lean will elaborate working from the
"outside in", meaning that it will first look at the expression _ + _ : Int
and assign the +
to be the one for Int
, and then need to insert coercions
for the subterms ↑x : Int
and ↑y : Int
, resulting in the ↑x + ↑y
version.
Important typeclasses #
-
Coe α β
is the most basic class, and the usual one you will want to use when implementing a coercion for your own types. -
CoeDep α (x : α) β
allowsβ
to depend not only onα
but on the valuex : α
itself. This is useful when the coercion function is dependent. An example of a dependent coercion is the instance forProp → Bool
, because it only holds forDecidable
propositions. It is defined as:instance (p : Prop) [Decidable p] : CoeDep Prop p Bool := ...
-
CoeFun α (γ : α → Sort v)
is a coercion to a function.γ a
should be a (coercion-to-)function type, and this is triggered whenever an elementf : α
appears in an application likef x
which would not make sense sincef
does not have a function type. This is automatically turned intoCoeFun.coe f x
. -
CoeSort α β
is a coercion to a sort.β
must be a universe, and ifa : α
appears in a place where a type is expected, like(x : a)
ora → a
, then it will be turned into(x : CoeSort.coe a)
. -
CoeHead
is likeCoe
, but whileCoe
can be transitively chained in theCoeT
class,CoeHead
can only appear once and only at the start of such a chain. This is useful when the transitive instances are not well behaved. -
CoeTail
is similar: it can only appear at the end of a chain of coercions. -
CoeT α (x : α) β
itself is the combination of all the aforementioned classes (exceptCoeSort
andCoeFun
which have different triggers). You can implementCoeT
if you do not want this coercion to be transitively composed with any other coercions.
Note that unlike most operators like +
, ↑
is always eagerly unfolded at
parse time into its definition. So if we look at the definition of f
from
before, we see no trace of the CoeT.coe
function:
def f (x : Nat) : Int := x
#print f
-- def f : Nat → Int :=
-- fun (x : Nat) => Int.ofNat x
Coerces a value of type
α
to typeβ
. Accessible by the notation↑x
, or by double type ascription((x : α) : β)
.coe : α → β
Coe α β
is the typeclass for coercions from α
to β
. It can be transitively
chained with other Coe
instances, and coercion is automatically used when
x
has type α
but it is used in a context where β
is expected.
You can use the ↑x
operator to explicitly trigger coercion.
Instances
Coerces a value of type
α
to typeβ
. Accessible by the notation↑x
, or by double type ascription((x : α) : β)
.coe : α → β
Auxiliary class that contains the transitive closure of Coe
.
Users should generally not implement this directly.
Instances
Coerces a value of type
α
to typeβ
. Accessible by the notation↑x
, or by double type ascription((x : α) : β)
.coe : α → β
CoeHead α β
is for coercions that can only appear at the beginning of a
sequence of coercions. That is, β
can be further coerced via Coe β γ
and
CoeTail γ δ
instances but α
will only be the inferred type of the input.
Instances
Coerces a value of type
α
to typeβ
. Accessible by the notation↑x
, or by double type ascription((x : α) : β)
.coe : α → β
CoeTail α β
is for coercions that can only appear at the end of a
sequence of coercions. That is, α
can be further coerced via Coe σ α
and
CoeHead τ σ
instances but β
will only be the expected type of the expression.
Instances
The resulting value of type
β
. The inputx : α
is a parameter to the type class, so the value of typeβ
may possibly depend on additional typeclasses onx
.coe : β
CoeDep α (x : α) β
is a typeclass for dependent coercions, that is, the type β
can depend on x
(or rather, the value of x
is available to typeclass search
so an instance that relates β
to x
is allowed).
Dependent coercions do not participate in the transitive chaining process of regular coercions: they must exactly match the type mismatch on both sides.
Instances
The resulting value of type
β
. The inputx : α
is a parameter to the type class, so the value of typeβ
may possibly depend on additional typeclasses onx
.coe : β
CoeT
is the core typeclass which is invoked by Lean to resolve a type error.
It can also be triggered explicitly with the notation ↑x
or by double type
ascription ((x : α) : β)
.
A CoeT
chain has the "grammar" (CoeHead)? (Coe)* (CoeTail)? | CoeDep
,
except that the empty sequence is not allowed (identity coercions don't need
the coercion system at all).
Instances
Coerces a value
f : α
to typeγ f
, which should be either be a function type or anotherCoeFun
type, in order to resolve a mistyped applicationf x
.coe : (f : α) → γ f
CoeFun α (γ : α → Sort v)
is a coercion to a function. γ a
should be a
(coercion-to-)function type, and this is triggered whenever an element
f : α
appears in an application like f x
which would not make sense since
f
does not have a function type. This is automatically turned into CoeFun.coe f x
.
Instances
Coerces a value of type
α
toβ
, which must be a universe.coe : α → β
CoeSort α β
is a coercion to a sort. β
must be a universe, and if
a : α
appears in a place where a type is expected, like (x : a)
or a → a
,
then it will be turned into (x : CoeSort.coe a)
.
Instances
↑x
represents a coercion, which converts x
of type α
to type β
, using
typeclasses to resolve a suitable conversion function. You can often leave the
↑
off entirely, since coercion is triggered implicitly whenever there is a
type error, but in ambiguous cases it can be useful to use ↑
to disambiguate
between e.g. ↑x + ↑y
and ↑(x + y)
.
Equations
- One or more equations did not get rendered due to their size.
Equations
- coeOfHeafOfTCOfTail = { coe := fun a => CoeTail.coe (CoeTC.coe (CoeHead.coe a)) }
Equations
- coeOfHeadOfTC = { coe := fun a => CoeTC.coe (CoeHead.coe a) }
Equations
- coeOfTCOfTail = { coe := fun a => CoeTail.coe (CoeTC.coe a) }
Equations
- coeOfHeadOfTail = { coe := fun a => CoeTail.coe (CoeHead.coe a) }
Equations
- coeSortToCoeTail = { coe := CoeSort.coe }
Basic instances #
Equations
- boolToProp = { coe := fun b => b = true }
Equations
- decPropToBool p = { coe := decide p }
Equations
- subtypeCoe = { coe := fun v => v.val }
Coe bridge #
Helper definition used by the elaborator. It is not meant to be used directly by users.
This is used for coercions between monads, in the case where we want to apply a monad lift and a coercion on the result type at the same time.
Equations
- Lean.Internal.liftCoeM x = do let a ← liftM x pure (CoeT.coe a)
Helper definition used by the elaborator. It is not meant to be used directly by users.
This is used for coercing the result type under a monad.
Equations
- Lean.Internal.coeM x = do let a ← x pure (CoeT.coe a)
Equations
- instCoeDep a = { coe := CoeFun.coe a }
Equations
- instCoeTail = { coe := fun a => CoeFun.coe a }
Equations
- instCoeTail_1 = { coe := fun a => CoeSort.coe a }