COMPUTED-HINTS

computing advice to the theorem proving process
Major Section:  MISCELLANEOUS

General Form of :hints:
(hint1 hint2 ... hintk)
Each element, hinti, must be either a common hint or a computed hint.

A common hint is of the form

(goal-spec :key1 val1 ... :keyn valn)
where goal-spec is as specified in goal-spec and each :keyi and vali is as specified in hints. Among the ``common hints'' we include both the primitive hints and user-defined custom keyword hints (see custom-keyword-hints).

A computed hint may be a function symbol, fn, of three, four or seven arguments. Otherwise, a computed hint is a term with the following properties:

(a) the only free variables allowed in the term are ID, CLAUSE, WORLD, STABLE-UNDER-SIMPLIFICATIONP, HIST, PSPV, CTX, and STATE;

(b) the output signature of the term is either (MV * * STATE), which is to be treated as an error triple (see below), or is *, denoting a single non-stobj value; and

(c) in the former case of (b) above, the term is single-threaded in STATE.

If a computed hint is a function symbol fn, whose arity n is therefore three, four, or seven, then it is treated as the term resulting from applying that fn to the first n variables shown in (a) above. Notice that it must then return a single non-stobj value, not an error triple, since state is not one of the first seven variables shown in (a).

Note: Error triples are an ACL2 idiom for implementing ``errors.'' If a computation returns (mv erp val state) in a context in which ACL2 is respecting the error triple convention (see ld-error-triples and see ld-error-action), then an error is deemed to have occurred if erp is non-nil. The computation is expected to have printed an appropriate error message to state and further computation is aborted. On the other hand, if a computation returns an error triple in which erp is nil, then ``value'' of the computation is taken to be the second component, val, of the triple (along with the possibly modified state), and computation continues.

The function symbol cases are treated as abbreviations of the term (fn ID CLAUSE WORLD), (fn ID CLAUSE WORLD STABLE-UNDER-SIMPLIFICATIONP), or (fn ID CLAUSE WORLD STABLE-UNDER-SIMPLIFICATIONP HIST PSPV CTX) as appropriate for the arity of fn. (Note that this tells you which argument of fn is which.) Moreover, in these cases the value returned must be a single ordinary (non-stobj) value, not an error triple. In the discussion below we assume all computed hints are of the term form. Indeed, we almost assume all computed hints are of the 3 and 4 argument forms. We only comment briefly on the 7 argument form in using-computed-hints-8.

The semantics of a computed hint term is as follows. On every subgoal, the term is evaluated in an environment in which the variables mentioned in (a) above are bound to context-sensitive values explained below. Either the computed hint signals an error, in which the proof attempt aborts, or else it returns a value, val and a new state, state. Any changes to those parts of state that affect logical soundness are undone; more specifically, the values of symbols (sometimes called ``state global variables'') in the list *protected-system-state-globals* in the global table of the state (see state) are restored when changed during evaluation. The value, val, of a non-erroneous computed hint calculation is either nil, which means the computed hint did not apply to the subgoal in question, or it is an alternating list of :keyi vali pairs as specified in hints. With one exception, those new hints are applied to the given subgoal as though the user had typed them explicitly.

The exception is that the first keyword in the returned val is allowed to be :COMPUTED-HINT-REPLACEMENT. Its value should be nil, t, or a list of terms. If this keyword is not present, the default value of nil is provided. We explain :COMPUTED-HINT-REPLACEMENT below.

The evaluation of a hint term is done with guard checking turned off (see set-guard-checking); e.g., the form (car 23) in a computed hint returns nil as per the axioms.

When a non-nil value is returned, the keyword/value pairs (other than the optional :COMPUTED-HINT-REPLACEMENT) are used as the hint for the subgoal in question. Thus, your job as the programmer of computed hints is either to cause an error, typically by invoking er, or to return a non-erroneous value triple whose value is the list of keys and values you would have typed had you supplied a common hint for the subgoal. (In particular, any theory expressions in it are evaluated with respect to the global current-theory, not whatever theory is active at the subgoal in question.) If the generated list of keywords and values is illegal, an error will be signaled and the proof attempt will be aborted.

The purpose of the :COMPUTED-HINT-REPLACEMENT keyword and its value, chr, is to change the list of hints. If chr is nil, then the hint which was applied is removed from the list of hints that is passed down to the children of the subgoal in question. This is the default. If chr is t, then the hint is left in the list of hints. This means that the same hint may act on the children of the subgoal. Otherwise, chr must be a list of terms, each of which is treated as a computed hint. The hint which was applied is deleted from the list of hints and the hints in chr are added to the list of hints passed to the children of the subgoal. The ability to compute new hints and pass them down allows strange and wonderful behavior.

For these purposes, the goals produced by induction and the top-level goals of forcing rounds are not considered children; all original hints are available to them.

Only the first hint applicable to a goal, as specified in the user-supplied list of :hints followed by the default-hints-table, will be applied to that goal. (For an advanced exception, see override-hints.)

It remains only to describe the bindings of the free variables.

Suppose the theorem prover is working on some clause, clause, named by some goal-spec, e.g., "Subgoal *1/2'''" in some logical world, world. Corresponding to the printed goal-spec is an internal data structure called a ``clause identifier'' id. See clause-identifier.

In the case of a common hint, the hint applies if the goal-spec of the hint is the same as the goal-spec of the clause in question.

In the case of a computed hint, the variable ID is bound to the clause id, the variable CLAUSE is bound to the (translated form of the) clause, and the variable WORLD is bound to the current ACL2 world. The variable STABLE-UNDER-SIMPLIFICATIONP is bound to t or nil. It is bound to t only if the clause is known to be stable under simplification. That is, the simplifier has been applied to the clause and did not change it. Such a clause is sometimes known as a ``simplification checkpoint.'' It is frequently useful to inject hints (e.g., to enable a rule or provide a :use hint) only when the goal in question has stabilized. If a hint is provided, the processing of the clause starts over with simplification.

As for CTX and STATE, they are provided so that you can pass them to the er macro to print error messages. We recommend not writing computed hints that otherwise change STATE!

The remaining variables, HIST and PSPV are not documented yet. Only users familiar with the internals of ACL2 are likely to need them or understand their values.

For some instruction about how to use computed hints, see using-computed-hints.