-
Notifications
You must be signed in to change notification settings - Fork 225
Delta Cheatsheet
This document aims to be a complete reference for when definitions do and do not get unfolded in Lean. It will take several weeks until it even begins to resemble such a document.
-
A definition can be unfolded eagerly during
is_def_eq.Example:
a =?= bcould in principle unfold all[reducible]layers of bothaandbeagerly before proceeding. This is not currently done. Use-case:id (xs ++ ys) =?= id (id (xs ++ ys))might just always unfold all theids before proceeding, regardless of definitional depth or any other heuristic. -
A definition can be unfolded lazily during
is_def_eq.Example: if
g := f, thenf a =?= g acould detect that the depth ofgis larger than the depth offand so unfoldgfirst before proceeding. Use-case:f1 (f2 ... (fN x)...) =?= f2 (f3 ... (fN x))would only need to unfoldf1to confirm that the two are definitionally equal. This idiom is ubiquitous with type-class instances. -
A definition can be unfolded because a Pi type or a Sort is expected or required.
Example: we may need to extract the universe level of a Sort in the app-builder.
-
A definition can be unfolded if it matches a particular pattern.
Example: we might have reflexivity-simplification rules for definitions that wrap recursors that only trigger if it is known which pattern in the definition matches.
-
A definition can be unfolded because of a unification hint.
Example: In
f =?= g, it might be the case that neitherfnorgwould normally be unfolded but that the pair(f, g)triggers a unification hint that causes one or both of them to be unfolded. Use-case: we wantn + 1 =?= succ nto succeed even if+would never otherwise get reduced. -
A definition can be unfolded during normalization if it is fully applied.
Example:
composehas the[unfold_full]tag, and is unfolded during normalization if it is applied to all 3 arguments (i.e.compose f g xnormalizes tof (g x)). Note that the number of arguments is not well defined in general and depends on the delta/normalization strategy being used. Also note that we can simulate this behaviour easily with a single reflexivity-simplification rule. -
A definition can be unfolded during normalization if a specified argument is a constructor.
Example:
pred : nat -> nathas the[unfold 1]tag and gets unfolded during normalization if its argument is zero or succ. -
A definition can be unfolded during normalization if an outer application is expecting a constructor at that position.
Example:
pairhas the[constructor]tag and so gets unfolded during normalization if an application ofpairoccurs in the major premise ofpair.rec. -
A definition can be unfolded eagerly and permanently by blast.
Example: blast currently unfolds every
[reducible]definition appearing in the context during initialization. Most blast modules that track lemmas in the environment (e.g. the simplifier) also unfold[reducible]definitions eagerly in the lemmas that they track. -
A definition can be unfolded "monotonically" by blast.
Example: given
H : f xandf x := g x x, blast might add the hypothesesH' : g x x := HandHH' : H = H' := rfl. We call this monotonic because we have not lost the original folded version. Note that we can use reflexivity-simplification rules monotonically as well.
-
Some data structures may be indexed by constant names or head symbols, and are hence inherently not robust to delta reduction.
Example: if
f := gbutfis not eagerly reduced by blast, then a[backward]lemmaHwith resultgwould not be considered for a backwards action for goalfeven ifis_def_eq(f, g). Note thatHwould not be tried even iffwere monotonically unfolded so that the context containedHfg : f = g. However,Hwould fire if e.g. asubstaction replaced the goal withg. This still seems brittle though. It would be more robust to lookup the[backward]rules for every term in the CC equivalence class of the goal.
-
Sometimes we may expect
is_def_eqto fail and want to prevent a lot of wasted unfolding.Example: if we are doing search
-
Sometimes the user may say "I am willing to suffer a long delay if it turns out I am wrong about
_ =?= _being true"Example:
exacttactic -
Sometimes we know something is going to work, but we don't know what.
Example: even though we know something is going to work, the
assumptiontactic might still not want to unfold aggressively inH1 : P (fact 100), H2 : P 0 |- P 0, or else risk wasting time computingfact 100. I thinkassumptionshould probably be conservative and the user should useexactif reductions are needed, orblastif the hypothesis in question contains system-generated names.
-
Above all we want the delta strategy to be simple and stable enough that this cheat-sheet can exist, can be maintained, and can be understood.
-
We want the strategy to be flexibly robust -- that is, we want the user to be able to say "I'll pay for the compute time, just please don't fail for some stupid reason relating to
delta". -
When running in more performance-stingy contexts, we want it to be easy for the user to figure out what went wrong (what did not get unfolded in the right place) and figure out what annotations need to be tweaked or hints provided to address the problem.
-
Rely on lazy unfolding in the unifier as little as possible, and rely on eager unfolding + "monotonic" unfolding (+ cc) in blast as much as possible.
-
Any unification problem that can be solved in conservative mode should be solvable in liberal mode as well.
Explanation: if liberal mode started unfolding things more aggressively from the beginning, it might no longer trigger an essential unification hint. Note that unification hints can involve "magic" solutions and are not subsumed by more aggressive unfolding. Solution: we keep an enum
{conservative, liberal-next, liberal-now}. In various parts of the unifier, we checkif (liberal-now) ..., and at the end ofon_is_def_eq_failure, if we areliberal-next, we recurse withliberal-now. We may want to make this more graded, so that there are e.g. many different levels of liberalness.
-
Four levels of reducibility, as there are now (not suggesting these names):
- eager
- lazy
- liberal
- never
-
These reducibility levels can be associated with definitions as well as with reflexivity-simplification lemmas, and perhaps with
[unfold <i1> ... <im>]annotations as well. -
The unifier does eager unfolding with the
eagerannotated definitions and reflexivity-simplification lemmas, and lazy unfolding with thelazyones. In liberal mode, if it fails it considers lazy-unfolding/normalizing things with theliberaltag. -
Blast eagerly (and permanently) unfolds the
eagerones, and monotonically unfolds thelazyones. In liberal mode, if it fails it considers monotonically unfolding theliberalones.Issue: there are many definitions that are likely to be tagged as
lazythat blast has no business monotonically unfolding, e.g. those related to instances. There may need to be a[no-blast]annotation or something like it that means "don't bother monotonically instantiating me".Issue: If hints are provided, does blast still monotonically instantiate? That is, is monotonic instantiation treated as if the
.deflemmas were tagged as[simp?]for ematching, and are ignored as are all other[simp?]lemmas when specific lemmas are passed as hints? -
relaxed-whnfis used as it is now, so that whenPiorSortis needed we find it no matter what the annotations. -
[constructor]and recursor-normalization is done in tactics and blast actions and not in unification, details TBD. -
We may also need a
conservativemode for unification (to be used during search, e.g. type class resolution) that does not even do lazy unfolding, and only considers theeagerrules.
-
definition left_inverse (g : B → A) (f : A → B) : Prop := ∀x, g (f x) = xComments: it seems weird to make this
[eager]and not allow blast to ever see it, because it is obviously a useful abstraction to humans. There is also useful lexical information in the name, that (a) makes the analogy to other variants of inverses more clear and (b) makes it much easier to locate informal proofs online. Suggested annotation:[lazy]Reason: There are not that many lemmas that use it, and the property it hides is not that complicated and very likely to be useful. We will very often want to atleast monotonically unfold it to expose thePi.