aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/Binary.hs
blob: fbf0a0798beb5e0f24d6a560acc266d411854583 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
module Binary
  ( toBinary
  , fromBinary
  , toBitString
  , fromBitString
  ) where

import           Control.Applicative
import qualified Data.BitString                as Bit
import           Data.Char
import           Helper

toBinary :: Expression -> String
toBinary (Bruijn      x        ) = (replicate (x + 1) '1') ++ "0"
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
 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 :)

-- TODO: Fix weird endianess things
padBitList :: [Bool] -> [Bool]
padBitList lst | length lst `mod` 8 == 0 = lst
               | otherwise               = padBitList ([False] ++ lst)

toBitString :: String -> Bit.BitString
toBitString = Bit.fromList . padBitList . map
  (\case
    '0' -> False
    '1' -> True
  )

fromBitString :: Bit.BitString -> String
fromBitString =
  map
      (\case
        False -> '0'
        True  -> '1'
      )
    . Bit.toList