shrub/pkg/hs-urbit/lib/Data/Noun/Atom.hs

184 lines
4.4 KiB
Haskell
Raw Normal View History

2019-05-10 05:45:28 +03:00
{-# LANGUAGE MagicHash, GeneralizedNewtypeDeriving, UnboxedTuples #-}
module Data.Noun.Atom where
import ClassyPrelude
2019-05-11 00:59:45 +03:00
import Control.Lens
-- import Prelude ((^))
2019-05-10 05:45:28 +03:00
import GHC.Integer.GMP.Internals
import GHC.Natural
import GHC.Prim
import GHC.Word
2019-05-11 00:59:45 +03:00
import GHC.Int
import Data.Bits
import Test.QuickCheck.Arbitrary
import Test.QuickCheck.Gen
2019-05-15 08:09:53 +03:00
import Text.Printf
import Data.Flat
2019-05-11 00:59:45 +03:00
2019-06-30 06:05:45 +03:00
import Data.Hashable (Hashable)
2019-05-11 00:59:45 +03:00
--------------------------------------------------------------------------------
2019-05-10 05:45:28 +03:00
newtype Atom = MkAtom { unAtom :: Natural }
2019-06-30 06:05:45 +03:00
deriving newtype (Eq, Ord, Num, Bits, Enum, Real, Integral, Flat, Hashable)
2019-05-11 00:59:45 +03:00
instance Show Atom where
show (MkAtom a) = show a
2019-05-11 00:59:45 +03:00
2019-05-13 23:46:05 +03:00
{-
An Atom with a bit-offset.
-}
2019-05-11 00:59:45 +03:00
data Cursor = Cursor
{ _cOffset :: {-# UNPACK #-} !Int
, _cBuffer :: !Atom
2019-05-11 00:59:45 +03:00
}
deriving (Eq, Ord, Show)
data Slice = Slice
{ _sOffset :: {-# UNPACK #-} !Int
, _sWidth :: {-# UNPACK #-} !Int
, _sBuffer :: !Atom
2019-05-11 00:59:45 +03:00
}
deriving (Eq, Ord, Show)
makeLenses ''Cursor
makeLenses ''Slice
-- Instances -------------------------------------------------------------------
2019-05-15 04:30:44 +03:00
instance Arbitrary Natural where
arbitrary = fromInteger . abs <$> arbitrary
instance Arbitrary Atom where
arbitrary = do
arbitrary >>= \case
False -> MkAtom <$> arbitrary
2019-05-18 02:02:39 +03:00
True -> arbitrary <&> ((`mod` 16) . MkAtom)
-- Conversion ------------------------------------------------------------------
class IsAtom a where
toAtom :: a -> Atom
fromAtom :: Atom -> a
instance IsAtom Atom where
toAtom = id
fromAtom = id
2019-05-15 04:30:44 +03:00
instance IsAtom Natural where
toAtom = MkAtom
fromAtom (MkAtom a) = a
instance IsAtom Word8 where
toAtom = fromIntegral
fromAtom = fromIntegral
instance IsAtom Word16 where
toAtom = fromIntegral
fromAtom = fromIntegral
instance IsAtom Word32 where
toAtom = fromIntegral
fromAtom = fromIntegral
instance IsAtom Word64 where
toAtom = fromIntegral
fromAtom = fromIntegral
2019-05-15 08:09:53 +03:00
instance IsAtom Word where
toAtom = fromIntegral
fromAtom = fromIntegral
instance IsAtom Int where
toAtom = fromIntegral
fromAtom = fromIntegral
2019-05-15 04:30:44 +03:00
instance IsAtom Integer where
toAtom = fromIntegral
fromAtom = fromIntegral
2019-05-11 00:59:45 +03:00
--------------------------------------------------------------------------------
2019-05-10 05:45:28 +03:00
2019-05-13 23:46:05 +03:00
{-
TODO Support 32-bit archetectures.
-}
wordBitWidth# :: Word# -> Word#
wordBitWidth# w = minusWord# 64## (clz# w)
2019-05-10 05:45:28 +03:00
bigNatBitWidth# :: BigNat -> Word#
bigNatBitWidth# nat =
2019-05-10 05:45:28 +03:00
lswBits `plusWord#` ((int2Word# lastIdx) `timesWord#` 64##)
where
(# lastIdx, _ #) = subIntC# (sizeofBigNat# nat) 1#
lswBits = wordBitWidth# (indexBigNat# nat lastIdx)
2019-05-10 05:45:28 +03:00
atomBitWidth# :: Atom -> Word#
atomBitWidth# (MkAtom (NatS# gl)) = wordBitWidth# gl
atomBitWidth# (MkAtom (NatJ# bn)) = bigNatBitWidth# bn
2019-05-11 00:59:45 +03:00
bitWidth :: Num a => Atom -> a
bitWidth a = fromIntegral (W# (atomBitWidth# a))
2019-05-11 00:59:45 +03:00
--------------------------------------------------------------------------------
cursor :: Atom -> Atom -> Cursor
cursor offset buf = Cursor (fromIntegral offset) buf
fromCursor :: Cursor -> Atom
fromCursor (Cursor off buf) = shiftR buf off
bumpCursor :: Word -> Cursor -> Cursor
bumpCursor off = over cOffset (+ fromIntegral off)
instance IsAtom Cursor where
toAtom (Cursor off bits) = shiftR bits off
fromAtom = Cursor 0
2019-05-11 00:59:45 +03:00
--------------------------------------------------------------------------------
slice :: (Atom, Atom) -> Atom -> Atom
slice (offset, size) buf =
fromSlice (Slice (fromAtom offset) (fromAtom size) buf)
2019-05-11 00:59:45 +03:00
fromSlice :: Slice -> Atom
fromSlice (Slice off wid buf) = mask .&. (shiftR buf off)
where mask = shiftL (MkAtom 1) wid - 1
2019-05-11 00:59:45 +03:00
--------------------------------------------------------------------------------
takeBits :: Int -> Atom -> Atom
2019-05-11 00:59:45 +03:00
takeBits wid buf = mask .&. buf
where mask = shiftL (MkAtom 1) wid - 1
2019-05-11 00:59:45 +03:00
bitIdx :: Int -> Atom -> Bool
bitIdx idx buf = testBit buf idx
bitConcat :: Atom -> Atom -> Atom
2019-05-15 04:30:44 +03:00
bitConcat x y = x .|. shiftL y (bitWidth x)
2019-05-15 08:09:53 +03:00
-- Bit Buffers -----------------------------------------------------------------
data Buf = Buf !Int !Atom
instance Show Buf where
show (Buf sz bits) = "0b"
<> replicate (sz - bitWidth bits) '0'
<> printf "%b (%d bits)" (toInteger bits) sz
instance Semigroup Buf where
Buf xSz xBuf <> Buf ySz yBuf = Buf (xSz+ySz) (xBuf .|. shiftL yBuf xSz)
instance Monoid Buf where
mempty = Buf 0 0
instance IsAtom Buf where
toAtom (Buf _ bits) = bits
fromAtom bits = Buf (bitWidth bits) bits