aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorMarvin Borner2023-02-24 15:31:35 +0100
committerMarvin Borner2023-02-24 15:31:35 +0100
commit3f20e501464fc31d0b10bbe004a2aae71aea38a4 (patch)
tree0eeaafbf3ef6df8627d38444821a7d3472c35572
parent23536b41ba46d45d2c61cb77bbad5bbfa6cd1ce3 (diff)
More functions for new representations
-rw-r--r--std/Binary.bruijn78
-rw-r--r--std/List.bruijn16
-rw-r--r--std/Number.bruijn43
3 files changed, 85 insertions, 52 deletions
diff --git a/std/Binary.bruijn b/std/Binary.bruijn
index 9fd95da..ec62e60 100644
--- a/std/Binary.bruijn
+++ b/std/Binary.bruijn
@@ -121,6 +121,8 @@ abstract! [0 z a¹ a⁰] ⧗ Binary → AbstractBinary
→^‣ abstract!
+:test (→^(+2b)) ([[[0 [[[1 [[[2]]]]]]]]])
+
# converts the abstracted binary representation back to normal
normal! ω [[0 z a¹ a⁰]] ⧗ AbstractBinary → Binary
z (+0b)
@@ -129,6 +131,8 @@ normal! ω [[0 z a¹ a⁰]] ⧗ AbstractBinary → Binary
→_‣ normal!
+:test (→_[[[0 [[[1 [[[2]]]]]]]]]) ((+2b))
+
# returns true if two binary numbers are equal
# → ignores leading 0s!
# also: ⋀?‣ ∘∘ (ψ* zip-with xnor? list!)
@@ -142,7 +146,7 @@ eq? [[abs 1 →^0]] ⧗ Binary → Binary → Boolean
:test ((+0b) =? (+0b)) (true)
:test ([[[1 (0 2)]]] =? [[[1 (0 2)]]]) (true)
-:test ([[[1 (0 2)]]] =? (+2b)) (false)
+:test ([[[1 2]]] =? (+2b)) (false)
# returns true if two binary numbers are not equal
not-eq? not! ∘∘ eq? ⧗ Binary → Binary → Boolean
@@ -176,6 +180,30 @@ dec [~(0 z a¹ a⁰)] ⧗ Binary → Binary
:test (--(+1b)) ([[[0 2]]])
:test (--(+3b)) ((+2b))
+# flips the bits of a binary number (1's complement)
+complement [[[[3 2 0 1]]]] ⧗ Binary → Binary
+
+~‣ complement
+
+:test (~(+0b) =? (+0b)) (true)
+:test (~(+1b) =? (+0b)) (true)
+:test (~(+42b)) ([[[1 (0 (1 (0 (1 (0 2)))))]]])
+
+# inverts a binary number by complementing and incrementing (2's complement)
+# don't forget to pad the number with zeroes if needed
+invert ++‣ ∘ ~‣ ⧗ Binary → Binary
+
+-‣ invert
+
+:test (-(+0b)) ((+1b))
+:test (-(+1b)) ((+1b))
+
+# pads a binary number with zeroes until its as long a another binary number
+# TODO: this could probably be done without list magic
+pad [[binary! (pad-right ∀(list! %1) b⁰ (list! 0))]] ⧗ Binary → Binary → Binary
+
+:test (pad (+5b) [[[1 2]]]) ([[[1 (0 (0 2))]]])
+
# adds two binary numbers (can introduce leading 0s)
# second argument gets abstracted (performance)
add [[abs 1 →^0]] ⧗ Binary → Binary → Binary
@@ -190,15 +218,18 @@ add [[abs 1 →^0]] ⧗ Binary → Binary → Binary
…+… add
-:test (((+0b) + (+0b)) =? (+0b)) (true)
-:test (((+0b) + (+3b)) =? (+3b)) (true)
-:test (((+1b) + (+2b)) =? (+3b)) (true)
-:test (((+42b) + (+1b)) =? (+43b)) (true)
+:test ((+0b) + (+0b) =? (+0b)) (true)
+:test ((+0b) + (+3b) =? (+3b)) (true)
+:test ((+1b) + (+2b) =? (+3b)) (true)
+:test ((+42b) + (+1b) =? (+43b)) (true)
+:test ((+1b) + (+42b) =? (+43b)) (true)
# subs two binary numbers (can introduce leading 0s)
# second argument gets abstracted (performance)
-sub [[abs 1 →^0]] ⧗ Binary → Binary → Binary
+# TODO: fix sub*; implementation is completely wrong
+sub* [[abs 1 →^0]] ⧗ Binary → Binary → Binary
abs [c (0 z a¹ a⁰)]
+ c¹ [1 ↑¹(3 0 b⁰) ↑¹(3 0 b⁰)]
a¹ [[[1 (c¹ 1) c¹' c¹]]]
c¹ [1 ↑⁰(3 0 b¹) ↑¹(3 0 b⁰)]
c¹' [1 ↑¹(3 0 b⁰) ↑¹(3 0 b⁰)]
@@ -208,31 +239,14 @@ sub [[abs 1 →^0]] ⧗ Binary → Binary → Binary
z [[0 --(→_1) →_1]]
c [[1 0 b⁰]]
+# subs two binary numbers
+# uses addition but with two's complement
+# TODO: isn't very performant ⇒ replace with sub*
+# TODO: gives wrong results if b>=a in a-b
+sub [[-((pad 1 0) + -(pad 0 1))]] ⧗ Binary → Binary → Binary
+
…-… sub
-:test ((+5b) - (+0b)) ((+5b))
-:test ((+8b) - (+3b)) ((+5b))
-:test ((+10b) - (+5b)) ((+5b))
-:test (((+20b) - (+1b))) ((+19b))
-
-# :test (%((+20b) - (+0b))) ((+20b))
-# :test (%((+20b) - (+1b))) ((+19b))
-# :test (%((+19b) - (+1b))) ((+18b))
-# :test (%((+19b) - (+2b))) ((+17b))
-# :test (%((+18b) - (+2b))) ((+16b))
-# :test (%((+18b) - (+3b))) ((+15b))
-# :test (%((+17b) - (+3b))) ((+14b))
-# :test (%((+17b) - (+4b))) ((+13b))
-# :test (%((+16b) - (+4b))) ((+12b))
-# :test (%((+16b) - (+5b))) ((+11b))
-# :test (%((+15b) - (+5b))) ((+10b))
-# :test (%((+15b) - (+6b))) ((+9b))
-# :test (%((+14b) - (+6b))) ((+8b))
-# :test (%((+14b) - (+7b))) ((+7b))
-# :test (%((+13b) - (+7b))) ((+6b))
-# :test (%((+13b) - (+8b))) ((+5b))
-# :test (%((+12b) - (+8b))) ((+4b))
-# :test (%((+12b) - (+9b))) ((+3b))
-# :test (%((+11b) - (+9b))) ((+2b))
-# :test (%((+11b) - (+10b))) ((+1b))
-# :test (%((+10b) - (+10b))) ((+0b))
+:test ((+42b) - (+12b) =? (+30b)) (true)
+:test ((+3b) - (+0b) =? (+3b)) (true)
+:test ((+3b) - (+2b) =? (+1b)) (true)
diff --git a/std/List.bruijn b/std/List.bruijn
index c3f36b2..986b6bc 100644
--- a/std/List.bruijn
+++ b/std/List.bruijn
@@ -112,7 +112,7 @@ reverse foldl \cons empty ⧗ (List a) → (List a)
:test (<~>((+1) : ((+2) : ((+3) : empty)))) ((+3) : ((+2) : ((+1) : empty)))
# creates list out of n terms
-# TODO: fix for balanced ternary
+# TODO: use balanced ternary
# TODO: fix/remove incorrect type?
list [0 [[[2 (0 : 1)]]] reverse empty] ⧗ (List a) → Unary → b → (List b)
@@ -185,9 +185,7 @@ init z [[rec]] ⧗ (List a) → (List a)
# concatenates a list of lists to one list
concat foldr append empty ⧗ (List a) → (List a) → (List a)
-# TODO: ?
-# :test (concat ((((+1) : ((+2) : empty)) : ((+3) : ((+4) : empty))) : empty)) ((+1) : ((+2) : ((+3) : ((+4) : empty))))
-
+:test (concat (((+1) : ((+2) : empty)) : (((+3) : ((+4) : empty)) : empty))) ((+1) : ((+2) : ((+3) : ((+4) : empty))))
:test (concat ("a" : ("b" : empty))) ("ab")
# maps a function returning list of list and concatenates
@@ -330,6 +328,16 @@ replicate \(g take repeat) ⧗ Number → a → (List a)
:test (replicate (+3) (+4)) ((+4) : ((+4) : ((+4) : empty)))
+# pads a list at end until its length is n
+pad-right [[[0 ++ (replicate |(2 - ∀0) 1)]]] ⧗ Number → a → (List a) → (List a)
+
+:test (pad-right (+6) 'a' "hah") ("hahaaa")
+
+# pads a list at start until its length is n
+pad-left [[[(replicate (2 - ∀0) 1) ++ 0]]] ⧗ Number → a → (List a) → (List a)
+
+:test (pad-left (+6) 'a' "hah") ("aaahah")
+
# returns an infinite list repeating a finite list
cycle z [[rec]] ⧗ (List a) → (List a)
rec 0 ++ (1 0)
diff --git a/std/Number.bruijn b/std/Number.bruijn
index f1f4432..dfed84c 100644
--- a/std/Number.bruijn
+++ b/std/Number.bruijn
@@ -269,12 +269,12 @@ add [[abs 1 →^0]] ⧗ Number → Number → Number
…+… add
-:test (((-42) + (-1)) =? (-43)) (true)
-:test (((-5) + (+6)) =? (+1)) (true)
-:test (((-1) + (+0)) =? (-1)) (true)
-:test (((+0) + (+0)) =? (+0)) (true)
-:test (((+1) + (+2)) =? (+3)) (true)
-:test (((+42) + (+1)) =? (+43)) (true)
+:test ((-42) + (-1) =? (-43)) (true)
+:test ((-5) + (+6) =? (+1)) (true)
+:test ((-1) + (+0) =? (-1)) (true)
+:test ((+0) + (+0) =? (+0)) (true)
+:test ((+1) + (+2) =? (+3)) (true)
+:test ((+42) + (+1) =? (+43)) (true)
# subs two balanced ternary numbers (can introduce leading 0s)
# second argument gets abstracted (performance)
@@ -282,12 +282,12 @@ sub [[1 + -0]] ⧗ Number → Number → Number
…-… sub
-:test (((-42) - (-1)) =? (-41)) (true)
-:test (((-5) - (+6)) =? (-11)) (true)
-:test (((-1) - (+0)) =? (-1)) (true)
-:test (((+0) - (+0)) =? (+0)) (true)
-:test (((+1) - (+2)) =? (-1)) (true)
-:test (((+42) - (+1)) =? (+41)) (true)
+:test ((-42) - (-1) =? (-41)) (true)
+:test ((-5) - (+6) =? (-11)) (true)
+:test ((-1) - (+0) =? (-1)) (true)
+:test ((+0) - (+0) =? (+0)) (true)
+:test ((+1) - (+2) =? (-1)) (true)
+:test ((+42) - (+1) =? (+41)) (true)
# returns true if number is greater than other number
# larger numbers should be second argument (performance)
@@ -356,10 +356,10 @@ mul [[1 z a⁻ a⁺ a⁰]] ⧗ Number → Number → Number
…⋅… mul
-:test (((+42) ⋅ (+0)) =? (+0)) (true)
-:test (((-1) ⋅ (+42)) =? (-42)) (true)
-:test (((+3) ⋅ (+11)) =? (+33)) (true)
-:test (((+42) ⋅ (-4)) =? (-168)) (true)
+:test ((+42) ⋅ (+0) =? (+0)) (true)
+:test ((-1) ⋅ (+42) =? (-42)) (true)
+:test ((+3) ⋅ (+11) =? (+33)) (true)
+:test ((+42) ⋅ (-4) =? (-168)) (true)
# divs a balanced ternary number by three (rshifts least significant trit)
div³ [~(0 z a⁻ a⁺ a⁰)] ⧗ Number → Number
@@ -438,5 +438,16 @@ mod ~‣ ∘∘ quot-rem ⧗ Number → Number
# returns max number of two
max [[(1 ≤? 0) 0 1]] ⧗ Number → Number → Number
+:test (max (+5) (+2)) ((+5))
+
# returns min number of two
min [[(1 ≤? 0) 1 0]] ⧗ Number → Number → Number
+
+:test (min (+5) (+2)) ((+2))
+
+# clamps a number between two numbers
+clamp [[[min 1 (max 0 2)]]] ⧗ Number → Number → Number
+
+:test (clamp (+0) (+5) (+3)) ((+3))
+:test (clamp (+0) (+5) (-2)) ((+0))
+:test (clamp (+0) (+5) (+7)) ((+5))