Processing math: 100%

Monday, December 19, 2016

Take 2: Graph of Tables for the Denotational Semantics of the Lambda Calculus

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. numbersnNvaluesvV::=nfun[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:expenv(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)v2E[|e|](ρ(x:=v1))}E[|e1e2|](ρ)={(v,G)|v1v2.v1E[|e1|](ρ)v2E[|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

  1. (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].
  2. (fun[1],G)E[|(λg.42)|]. We need (42,G)E[|42|](f:=fun[1]) which is immediate.
  3. (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(n1)Hλr.FfactZH 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(n1)|](r:=fun[3],n:=0)
  • (1,G)E[|ifn=0then1elsen×r(n1)|](r:=fun[3],n:=1)
  • (2,G)E[|ifn=0then1elsen×r(n1)|](r:=fun[3],n:=2)
  • (6,G)E[|ifn=0then1elsen×r(n1)|](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