Summer school 2008:Arithmetic expressions with let-binding (hypothetical evaluation)
This is Literate Twelf Code: here Status: %% OK %% Output: here. |
Summer school 2008 |
Previous: Call-by-value let-binding syntax |
Next: Typed arithmetic expressions |
Numbers and addition are as before.
Evaluation using a hypothetical judgement
We use the call-by-value syntax for expressions here.
Values, expressions, answers, and the first two cases of evaluation are as before:
val : type. %name val V. num : nat -> val. exp : type. %name exp E. ret : val -> exp. plus : exp -> exp -> exp. let : exp -> (val -> exp) -> exp. %%% evaluation, using hypothetical judgements ans : type. %name ans A. anum : nat -> ans. eval : exp -> ans -> type. %mode eval +E1 -E2. eval/val : eval (ret (num N)) (anum N). eval/plus : eval (plus E1 E2) (anum N) <- eval E1 (anum N1) <- eval E2 (anum N2) <- add N1 N2 N. eval/let : eval (let E1 ([x] E2 x)) A2 <- eval E1 A1 <- ({x:val} eval (ret x) A1 -> eval (E2 x) A2).
eval/let deserves some explanation: the second recursive call says that we evaluate (E2 x) in an extended LF context. In particular, we extend the context with x:val, a variable ranging over values, and a derivation of eval (ret x) A1. In the scope of these assumptions, the expression ret x therefore evaluates to A1. In the terminology of Practical Foundations for Programming Languages, eval is a hypothetical (because we add eval assumptions to the context) and general (because we add variables to the context) judgement. The context of eval is represented by the LF context, a technique called higher-order judgements.
Totality in non-empty worlds
Because evaluation recurs in an extend context, we must prove it total not just for the empty context, as we have done above, but for a world of a particular form.
If we tried to prove it total in the empty context, Twelf would complain:
%worlds () (eval _ _).
Twelf 1.7.1+ (r1896, built 05/05/15 at 12:56:43 on yazoo.plparty.org)%worlds () (eval _ _). Error: While checking constant eval/let: World violation for family eval: {x:val} {_:eval (ret x) A1} </: 1
%% ABORT %%
This error means "you said eval stays in the empty context, but it doesn't!".
In what contexts in eval total? Not in every context: if we ever assumed a variable x:val without also assuming eval (ret x) A for some A, then ret x would be an expression without a value. So we want to describe the invariant that the context looks like
x1:val, d1:eval (ret x) A1, ...... , xn:val, dn:eval (ret x) An
for some A1, ... , An.
We do this by
- defining a block eval_block describing that pattern
- stating eval for a world containing contexts made up of eval_blocks
%block eval_block : some {A:ans} block {x:val} {_:eval (ret x) A}. %worlds (eval_block) (eval _ _).
Now Twelf can verify the totality of eval:
%total E (eval E _).
Summer school 2008 |
Previous: Call-by-value let-binding syntax |
Next: Typed arithmetic expressions |