module EVM.RLP where
import Prelude hiding (drop, head)
import EVM.Types
import Data.Bits (shiftR)
import Data.ByteString (ByteString, drop, head)
import qualified Data.ByteString as BS
data RLP = BS ByteString | List [RLP] deriving Eq
instance Show RLP where
show (BS str) = show (ByteStringS str)
show (List list) = show list
slice :: Int -> Int -> ByteString -> ByteString
slice offset size bs = BS.take size $ BS.drop offset bs
itemInfo :: ByteString -> (Int, Int, Bool, Bool)
itemInfo bs | bs == mempty = (0, 0, False, False)
| otherwise = case head bs of
x | 0 <= x && x < 128 -> (0, 1, False, True)
x | 128 <= x && x < 184 -> (1, num x - 128, False, (BS.length bs /= 2) || (127 < (head $ drop 1 bs)))
x | 184 <= x && x < 192 -> (1 + pre, len, False, (len > 55) && head (drop 1 bs) /= 0)
where pre = num $ x - 183
len = num $ word $ slice 1 pre bs
x | 192 <= x && x < 248 -> (1, num $ x - 192, True, True)
x -> (1 + pre, len, True, (len > 55) && head (drop 1 bs) /= 0)
where pre = num $ x - 247
len = num $ word $ slice 1 pre bs
rlpdecode :: ByteString -> Maybe RLP
rlpdecode bs | optimal && pre + len == BS.length bs = if isList
then do
items <- mapM
(\(s, e) -> rlpdecode $ slice s e content) $
rlplengths content 0 len
Just (List items)
else Just (BS content)
| otherwise = Nothing
where (pre, len, isList, optimal) = itemInfo bs
content = drop pre bs
rlplengths :: ByteString -> Int -> Int -> [(Int,Int)]
rlplengths bs acc top | acc < top = let (pre, len, _, _) = itemInfo bs
in (acc, pre + len) : rlplengths (drop (pre + len) bs) (acc + pre + len) top
| otherwise = []
rlpencode :: RLP -> ByteString
rlpencode (BS bs) = if BS.length bs == 1 && head bs < 128 then bs
else encodeLen 128 bs
rlpencode (List items) = encodeLen 192 (mconcat $ map rlpencode items)
encodeLen :: Int -> ByteString -> ByteString
encodeLen offset bs | BS.length bs <= 55 = prefix (BS.length bs) <> bs
| otherwise = prefix lenLen <> lenBytes <> bs
where
lenBytes = asBE $ BS.length bs
prefix n = BS.singleton $ num $ offset + n
lenLen = BS.length lenBytes + 55
rlpList :: [RLP] -> ByteString
rlpList n = rlpencode $ List n
octets :: W256 -> ByteString
octets x =
BS.pack $ dropWhile (== 0) [fromIntegral (shiftR x (8 * i)) | i <- reverse [0..31]]
octets160 :: Addr -> ByteString
octets160 x =
BS.pack $ dropWhile (== 0) [fromIntegral (shiftR x (8 * i)) | i <- reverse [0..19]]
rlpWord256 :: W256 -> RLP
rlpWord256 0 = BS mempty
rlpWord256 n = BS $ octets n
rlpWord160 :: Addr -> RLP
rlpWord160 0 = BS mempty
rlpWord160 n = BS $ octets160 n