aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/Binary.hs
diff options
context:
space:
mode:
authorMarvin Borner2022-07-12 22:36:35 +0200
committerMarvin Borner2022-07-12 22:36:35 +0200
commit495127cde69e0008b8a1c428d5ddf0d805eb25bb (patch)
tree92e1d75db3c79d41768573da524a057269668208 /src/Binary.hs
parent269a7832ece9c1997d0431c41d04e91d46813a96 (diff)
Sync
Diffstat (limited to 'src/Binary.hs')
-rw-r--r--src/Binary.hs78
1 files changed, 56 insertions, 22 deletions
diff --git a/src/Binary.hs b/src/Binary.hs
index fbf0a07..974bf8d 100644
--- a/src/Binary.hs
+++ b/src/Binary.hs
@@ -6,8 +6,13 @@ module Binary
) where
import Control.Applicative
+import Data.Binary ( decode
+ , encode
+ )
import qualified Data.BitString as Bit
+import qualified Data.ByteString.Lazy as Byte
import Data.Char
+import Data.Int ( Int8 )
import Helper
toBinary :: Expression -> String
@@ -16,36 +21,65 @@ toBinary (Abstraction exp ) = "00" ++ toBinary exp
toBinary (Application exp1 exp2) = "01" ++ (toBinary exp1) ++ (toBinary exp2)
-- Stolen from John Tromp
-fromBinary :: String -> Expression
-fromBinary = foldr
- (\x -> Abstraction . (Application . Application (Bruijn 0) . code $ x))
- nil
+-- fromBinary :: String -> Expression
+-- fromBinary = foldr
+-- (\x -> Abstraction . (Application . Application (Bruijn 0) . code $ x))
+-- nil
+-- where
+-- nil = code '1'
+-- code '0' = Abstraction (Abstraction (Bruijn 1))
+-- code '1' = Abstraction (Abstraction (Bruijn 0))
+-- code x = fromBinary (showsBin 8 (ord x) "")
+-- showsBin n x = if n == 0
+-- then id
+-- else let (x', b) = divMod x 2 in showsBin (n - 1) x' . (intToDigit b :)
+
+-- https://github.com/ljedrz/blc/blob/master/src/encoding/binary.rs
+fromBinary' :: String -> (Expression, String)
+fromBinary' = \case
+ -- "" -> (Bruijn 0, "")
+ '0' : '0' : rst ->
+ let (exp, rst) = fromBinary' rst in (Abstraction exp, rst)
+ '0' : '1' : rst ->
+ let (exp1, rst1) = fromBinary' rst
+ (exp2, rst2) = fromBinary' rst1
+ in (Application exp1 exp2, rst2)
+ '1' : '0' : rst -> binaryBruijn rst
+ '1' : '1' : rst -> binaryBruijn rst
+ _ -> (Bruijn (-1), "")
where
- nil = code '1'
- code '0' = Abstraction (Abstraction (Bruijn 1))
- code '1' = Abstraction (Abstraction (Bruijn 0))
- code x = fromBinary (showsBin 8 (ord x) "")
- showsBin n x = if n == 0
- then id
- else let (x', b) = divMod x 2 in showsBin (n - 1) x' . (intToDigit b :)
+ binaryBruijn rst =
+ let idx = length . takeWhile (== '1')
+ in case length rst of
+ 0 -> (Bruijn $ idx rst, "")
+ _ -> (Bruijn $ idx rst, drop ((idx rst) + 1) rst)
--- TODO: Fix weird endianess things
-padBitList :: [Bool] -> [Bool]
-padBitList lst | length lst `mod` 8 == 0 = lst
- | otherwise = padBitList ([False] ++ lst)
+fromBinary :: String -> Expression
+fromBinary = fst . fromBinary'
+-- 1 byte indicating bit-padding at end + n bytes filled with bits
+-- TODO: technically only 1 nibble is needed (versioning/sth?)
toBitString :: String -> Bit.BitString
-toBitString = Bit.fromList . padBitList . map
- (\case
- '0' -> False
- '1' -> True
- )
+toBitString str = Bit.concat
+ [ Bit.bitString $ Byte.toStrict $ encode
+ (fromIntegral $ length str `mod` 8 :: Int8)
+ , Bit.fromList $ map
+ (\case
+ '0' -> False
+ '1' -> True
+ )
+ str
+ ]
+-- TODO: Fix this
fromBitString :: Bit.BitString -> String
-fromBitString =
+fromBitString bits =
map
(\case
False -> '0'
True -> '1'
)
- . Bit.toList
+ $ Bit.toList
+ $ Bit.take (Bit.length bits - pad bits)
+ $ Bit.drop 8 bits
+ where pad bits = decode $ Bit.realizeBitStringLazy $ Bit.take 8 bits