In the previous post, the denotational semantics I gave for the lambda calculus could not deal with self application, such as the program (λf.ff)(λg.42) whose result should be 42. The problem was that I defined function values to be tables of pairs of values, using a datatype definition, which rules out the possibility of cycles. In the above program, the table for λg.42 needs to include itself so that the application ff makes sense.
A straightforward way to solve this problem is to allow cycles by representing all the functions created by the program as a graph of tables. A function value will just contain an index, i.e. it will have the form fun[i] with index i, and the graph will map the index to a table. So we define values as follows. numbersn∈Nvaluesv∈V::=n∣fun[i]graph=N→(V×V)list Given a function value v and graph G, we write tab(v,G) for the table t when v=fun[i] and G(i)=t for some index i. So we modify the semantic function E to be defined as follows.
E:exp→env→(V×graph)setE[|n|](ρ)={(v,G)∣v=n}E[|x|](ρ)={(v,G)∣v=ρ(x)}E[|λx.e|](ρ)={(v,G)|∀v1v2.(v1,v2)∈tab(v,G)⇒v2∈E[|e|](ρ(x:=v1))}E[|e1e2|](ρ)={(v,G)|∃v1v2.v1∈E[|e1|](ρ)∧v2∈E[|e2|](ρ)∧(v2,v)∈tab(v1,G)}Example 1. Let's consider again the program (λf.ff)(λg.42) We'll define a graph G as follows. G(0)=[(fun[1],42)]G(1)=[(fun[1],42)] To show (42,G)∈E[|(λf.ff)(λg.42)|]∅ we need to show
- (fun[0],G)∈E[|(λf.ff)|]∅. So we need to show (42,G)∈E[|ff|](f:=fun[1]) which we have because (fun[1],42)∈fun[1].
- (fun[1],G)∈E[|(λg.42)|]∅. We need (42,G)∈E[|42|](f:=fun[1]) which is immediate.
- (fun[1],42)∈tab(fun[0]). This is immediate.
Example 2. We shall compute the factorial of 3 using the strict version of the Y combinator, that is, the Z combinator. R≡λv.(xx)vM≡λx.fRZ≡λf.MMF≡λn.ifn=0then1elsen×r(n−1)H≡λr.Ffact≡ZH We shall show that (6,G)∈E[|fact3|]∅ for some graph G that we need to construct. By instrumenting an interpreter for the lambda calculus and running fact3, we observe the following graph. G(0)=[(fun[1],fun[4])]for ZG(1)=[(fun[3],fun[4])]for HG(2)=[(fun[2],fun[4])]for MG(3)=[(0,1),(1,1),(2,2)]for RG(4)=[(0,1),(1,1),(2,2),(3,6)]for F We check all of the following (Tedious! I'm going to write a program to do this next time.):
- (fun[4],G)∈E[ZH]∅
- (3,6)∈G(4)
- (fun[0],G)∈E[|Z|]∅
- (fun[4],G)∈E[|MM|](f:=fun[1])
- (fun[2],G)∈E[|M|](f:=fun[1])
- (fun[4],G)∈E[|fR|](f:=fun[1],x:=fun[2])
- (fun[3],G)∈E[|R|](f:=fun[1],x:=fun[2])
- (1,G)∈E[|(xx)v|](f:=fun[1],x:=fun[2],v:=0) and (0,1)∈G(4)
- (1,G)∈E[|(xx)v|](f:=fun[1],x:=fun[2],v:=1) and (1,1)∈G(4)
- (2,G)∈E[|(xx)v|](f:=fun[1],x:=fun[2],v:=2) and (2,2)∈G(4)
- (fun[2],fun[4])∈G(2)
- (fun[1],G)∈E[|H|]∅
- (fun[4],G)∈E[|F|](r:=fun[3])
- (1,G)∈E[|ifn=0then1elsen×r(n−1)|](r:=fun[3],n:=0)
- (1,G)∈E[|ifn=0then1elsen×r(n−1)|](r:=fun[3],n:=1)
- (2,G)∈E[|ifn=0then1elsen×r(n−1)|](r:=fun[3],n:=2)
- (6,G)∈E[|ifn=0then1elsen×r(n−1)|](r:=fun[3],n:=3)
The next step is to update the proof of soundness wrt. the big-step semantics. The graphs will make it a bit more challenging. But hopefully they will make it possible to also prove completeness!
No comments:
Post a Comment