diff options
author | Marvin Borner | 2023-02-24 15:31:35 +0100 |
---|---|---|
committer | Marvin Borner | 2023-02-24 15:31:35 +0100 |
commit | 3f20e501464fc31d0b10bbe004a2aae71aea38a4 (patch) | |
tree | 0eeaafbf3ef6df8627d38444821a7d3472c35572 | |
parent | 23536b41ba46d45d2c61cb77bbad5bbfa6cd1ce3 (diff) |
More functions for new representations
-rw-r--r-- | std/Binary.bruijn | 78 | ||||
-rw-r--r-- | std/List.bruijn | 16 | ||||
-rw-r--r-- | std/Number.bruijn | 43 |
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)) |