aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
-rw-r--r--src/Helper.hs20
-rw-r--r--src/Parser.hs34
-rw-r--r--std/Float.bruijn19
-rw-r--r--std/Math/Rational.bruijn87
4 files changed, 140 insertions, 20 deletions
diff --git a/src/Helper.hs b/src/Helper.hs
index 695d63f..3617cfc 100644
--- a/src/Helper.hs
+++ b/src/Helper.hs
@@ -20,6 +20,9 @@ import Data.Maybe ( fromMaybe
, isNothing
)
import GHC.Generics ( Generic )
+import GHC.Real ( denominator
+ , numerator
+ )
import Text.Megaparsec
invalidProgramState :: a
@@ -370,7 +373,7 @@ humanifyString :: Expression -> Maybe String
humanifyString e = do
es <- unlistify e
str <- mapM binaryToChar' es
- pure str
+ pure $ "\"" <> str <> "\""
humanifyPair :: Expression -> Maybe String
humanifyPair e = do
@@ -381,6 +384,21 @@ humanifyPair e = do
---
+floatToRational :: Rational -> Expression
+floatToRational f = Abstraction
+ (Application (Application (Bruijn 0) (decimalToTernary p))
+ (decimalToTernary $ q - 1)
+ )
+ where
+ p = numerator f
+ q = denominator f
+
+floatToReal :: Rational -> Expression
+floatToReal f = Bruijn 0
+
+floatToComplex :: Rational -> Expression
+floatToComplex f = Bruijn 0
+
-- Dec to Bal3 in Bruijn encoding: reversed application with 0=>0; 1=>1; T=>2; end=>3
-- e.g. 0=0=[[[[3]]]]; 2=1T=[[[[2 (1 3)]]]] -5=T11=[[[[1 (1 (2 3))]]]]
decimalToTernary :: Integer -> Expression
diff --git a/src/Parser.hs b/src/Parser.hs
index 57d6f58..a63b4a2 100644
--- a/src/Parser.hs
+++ b/src/Parser.hs
@@ -8,6 +8,7 @@ import Control.Monad ( ap
, void
)
import Data.Void
+import GHC.Real ( (%) )
import Helper
import Text.Megaparsec hiding ( parseTest )
import Text.Megaparsec.Char
@@ -135,6 +136,38 @@ parseNumeral = do
number :: Parser Integer
number = ap sign nat
+-- parsed float a.b to rational p/q
+convertToRational :: Integer -> Integer -> Rational
+convertToRational a b =
+ let denominator :: Integer
+ denominator = 10 ^ length (show b)
+ numerator = b + a * denominator
+ common = gcd numerator denominator
+ in (numerator `div` common) % (denominator `div` common)
+
+parseFloat :: Parser Expression
+parseFloat = do
+ _ <- string "(" <?> "float start"
+ num <- signedFloat <?> "signed float"
+ base <- try (oneOf "frc") <|> return 'f'
+ _ <- string ")" <?> "float end"
+ pure $ f base num
+ where
+ f 'f' = floatToRational
+ f 'r' = floatToReal
+ f 'c' = floatToComplex -- TODO: imaginary
+ f _ = invalidProgramState
+ sign :: Parser (Rational -> Rational)
+ sign = (char '-' >> return negate) <|> (char '+' >> return id)
+ float :: Parser Rational
+ float = do
+ a <- read <$> some digitChar <?> "digits"
+ _ <- char '.' <?> "float delimiter"
+ b <- read <$> some digitChar <?> "digits"
+ return $ convertToRational a b
+ signedFloat :: Parser Rational
+ signedFloat = ap sign float
+
specialEscape :: Parser Char
specialEscape =
choice (zipWith (\c r -> r <$ char c) "bnfrt\\\"/" "\b\n\f\r\t\\\"/")
@@ -201,6 +234,7 @@ parseSingleton =
let parseSingletonExpression =
parseBruijn
<|> try parseNumeral
+ <|> try parseFloat
<|> parseString
<|> try parseChar
<|> parseQuote
diff --git a/std/Float.bruijn b/std/Float.bruijn
deleted file mode 100644
index c272d2b..0000000
--- a/std/Float.bruijn
+++ /dev/null
@@ -1,19 +0,0 @@
-# MIT License, Copyright (c) 2022 Marvin Borner
-# Arbitrary-precision floating-point arithmetic implementation using
-# (+3.14) = pair (+3) (+14)
-
-# This is completely WIP and only intended as a proof of concept
-
-:import std/Combinator .
-:import std/Number .
-:import std/Pair .
-
-pi (+3) : (+14159)
-
-# generates a float from a normal balanced ternary number
-float! \…:… (+0)
-
-# adds two floating numbers
-# TODO: Carry support
-# - needed: mod, div (?) -> ternary carry != decimal carry
-add zip-with …+…
diff --git a/std/Math/Rational.bruijn b/std/Math/Rational.bruijn
new file mode 100644
index 0000000..4c2f046
--- /dev/null
+++ b/std/Math/Rational.bruijn
@@ -0,0 +1,87 @@
+# ideas by u/DaVinci103
+# MIT License, Copyright (c) 2024 Marvin Borner
+
+# (p : q) ⇔ (1 / (q + 1))
+
+:import std/Combinator .
+:import std/Logic .
+:import std/Pair .
+:import std/Math N
+
+# returns true if two rational numbers are equal
+eq? &[[&[[N.eq? (N.mul 3 N.++0) (N.mul N.++2 1)]]]] ⧗ Rational → Rational → Boolean
+
+…=?… eq?
+
+:test (((+1) : (+3)) =? ((+2) : (+7))) (true)
+:test ((+0.5) =? (+0.5)) (true)
+:test ((+42.0) =? (+42.0)) (true)
+:test ((+0.4) =? (+0.5)) (false)
+
+# finds smallest equivalent representation of a rational number
+compress &[[[(N.div 2 0) : N.--(N.div N.++1 0)] (N.gcd 1 N.++0)]] ⧗ Rational → Rational
+
+%‣ compress
+
+:test (%((+4) : (+1)) =? (+2.0)) (true)
+:test (%((+4) : (+7)) =? (+0.5)) (true)
+
+# adds two rational numbers
+add &[[&[[p : q]]]] ⧗ Rational → Rational → Rational
+ p N.add (N.mul 3 N.++0) (N.mul 1 N.++2)
+ q N.add (N.mul 2 0) (N.add 2 0)
+
+…+… add
+
+:test ((+0.5) + (+0.5) =? (+1.0)) (true)
+:test ((+1.8) + (+1.2) =? (+3.0)) (true)
+:test ((-1.8) + (+1.2) =? (-0.6)) (true)
+
+# subtracts two rational numbers
+sub &[[&[[p : q]]]] ⧗ Rational → Rational → Rational
+ p N.sub (N.mul 3 N.++0) (N.mul 1 N.++2)
+ q N.add (N.mul 2 0) (N.add 2 0)
+
+…-… sub
+
+:test ((+0.5) - (+0.5) =? (+0.0)) (true)
+:test ((+3.0) - (+1.8) =? (+1.2)) (true)
+:test ((+1.8) - (-1.2) =? (+3.0)) (true)
+
+# negates a rational number
+negate &[[N.-1 : 0]] ⧗ Rational → Rational
+
+-‣ negate
+
+:test (-(+0.0) =? (+0.0)) (true)
+:test (-(+4.2) =? (-4.2)) (true)
+:test (-(-4.2) =? (+4.2)) (true)
+
+# multiplies two rational numbers
+mul &[[&[[p : q]]]] ⧗ Rational → Rational → Rational
+ p N.mul 3 1
+ q N.add (N.mul 2 0) (N.add 2 0)
+
+…⋅… mul
+
+:test ((+5.0) ⋅ (+5.0) =? (+25.0)) (true)
+:test ((+1.8) ⋅ (+1.2) =? (+2.16)) (true)
+
+# finds the multiplicative inverse of a rational number
+invert &[[N.compare-case eq gt lt 1 (+0)]] ⧗ Rational → Rational
+ eq Ω
+ gt N.++0 : N.--1
+ lt N.-(N.++0) : N.--(N.-1)
+
+~‣ invert
+
+:test (~(+0.5) =? (+2.0)) (true)
+:test (~(-0.5) =? (-2.0)) (true)
+
+# divides two rational numbers
+div [[1 ⋅ ~0]] ⧗ Rational → Rational → Rational
+
+…/… div
+
+:test ((+8.0) / (+4.0) =? (+2.0)) (true)
+:test ((+18.0) / (+12.0) =? (+1.5)) (true)