aboutsummaryrefslogtreecommitdiffhomepage
path: root/std/Meta.bruijn
diff options
context:
space:
mode:
authorMarvin Borner2024-02-16 16:46:45 +0100
committerMarvin Borner2024-02-16 16:56:55 +0100
commitf3dc81930ebbc6727f9a796f71dceffbcb753752 (patch)
tree01dc563ac42b5d45091a4adc180e016078d38091 /std/Meta.bruijn
parenta852b3e28f7f080c1912e9a7c07cb85636bce89a (diff)
Added strong Rosenberg encoding for meta encoding
Originally got the idea from @sgoguen (YC) and was further inspired by @hbr from one of their blog posts[1]. The current encoding has some holes and bad performance, so I might switch to a different one in the future. The wiki and metaprogramming blog post have been updated accordingly. [1] https://hbr.github.io/Lambda-Calculus/computability/text.html
Diffstat (limited to 'std/Meta.bruijn')
-rw-r--r--std/Meta.bruijn30
1 files changed, 30 insertions, 0 deletions
diff --git a/std/Meta.bruijn b/std/Meta.bruijn
index 31d781a..3b67c35 100644
--- a/std/Meta.bruijn
+++ b/std/Meta.bruijn
@@ -5,6 +5,8 @@
:import std/Combinator .
:import std/Number/Unary .
:import std/List .
+:import std/Number/Pairing .
+:import std/Number/Conversion .
:import std/Pair P
# constructor of meta abstraction
@@ -223,3 +225,31 @@ map [[0 case-idx case-app case-abs]] ⧗ (Meta → Meta) → Meta → Meta
:test (map swap `[0 1]) (`[1 0])
:test (map inc `0) (`1)
:test (map (map inc) `[0]) (`[1])
+
+# ternary encoding of meta terms
+# inspired by Helmut Brandl's and Steve Goguen's work
+# uses ternary for lower space complexity
+encode fold idx-cb app-cb abs-cb ⧗ Meta → Number
+ idx-cb (pair-ternary (+0)) ∘ unary-to-ternary
+ app-cb (pair-ternary (+1)) ∘∘ pair-ternary
+ abs-cb pair-ternary (+2)
+
+:test (ternary-to-unary (encode `0)) ((+0u))
+:test (ternary-to-unary (encode `1)) ((+1u))
+:test (ternary-to-unary (encode `(0 0))) ((+3u))
+:test (ternary-to-unary (encode `[1])) ((+7u))
+:test (ternary-to-unary (encode `[0])) ((+8u))
+
+# decode ternary encoding back to meta term
+# TODO: improve performance (maybe with different unpairing function or better sqrt)
+decode z [[unpair-ternary 0 go]] ⧗ Number → Meta
+ go [[(case-idx : (case-app : {}case-abs)) !! 1 0]]
+ case-idx idx ∘ ternary-to-unary
+ case-app [unpair-ternary 0 (ψ app 4)]
+ case-abs abs ∘ 3
+
+:test (decode (+0)) (`0)
+:test (decode (+1)) (`1)
+:test (decode (+3)) (`(0 0))
+:test (decode (+7)) (`[1])
+:test (decode (+8)) (`[0])