{-# LANGUAGE CPP, BangPatterns, PatternGuards, DeriveDataTypeable #-}
{-# OPTIONS_GHC -Wno-incomplete-uni-patterns #-}
{-# OPTIONS_HADDOCK hide #-}
module Codec.Archive.Tar.Index.StringTable (
StringTable(..),
lookup,
index,
construct,
StringTableBuilder,
empty,
insert,
inserts,
finalise,
unfinalise,
serialise,
serialiseSize,
deserialiseV1,
deserialiseV2,
index'
) where
import Data.Typeable (Typeable)
import Prelude hiding (lookup, id)
import Data.List hiding (lookup, insert)
import Data.Function (on)
import Data.Word (Word32)
import Data.Int (Int32)
import Data.Bits
import Data.Monoid (Monoid(..))
import Data.Monoid ((<>))
import Control.Exception (assert)
import qualified Data.Array.Unboxed as A
import qualified Data.Array.Base as A
import Data.Array.Unboxed ((!))
import qualified Data.Map.Strict as Map
import Data.Map.Strict (Map)
import qualified Data.ByteString as BS
import qualified Data.ByteString.Unsafe as BS
import qualified Data.ByteString.Lazy as LBS
import Data.ByteString.Builder as BS
import Data.ByteString.Builder.Extra as BS (byteStringCopy)
import GHC.IO (unsafePerformIO)
import Unsafe.Coerce (unsafeCoerce)
import Codec.Archive.Tar.Index.Utils
data StringTable id = StringTable
{-# UNPACK #-} !BS.ByteString
{-# UNPACK #-} !(A.UArray Int32 Word32)
{-# UNPACK #-} !(A.UArray Int32 Int32)
{-# UNPACK #-} !(A.UArray Int32 Int32)
deriving (Int -> StringTable id -> ShowS
[StringTable id] -> ShowS
StringTable id -> String
(Int -> StringTable id -> ShowS)
-> (StringTable id -> String)
-> ([StringTable id] -> ShowS)
-> Show (StringTable id)
forall id. Int -> StringTable id -> ShowS
forall id. [StringTable id] -> ShowS
forall id. StringTable id -> String
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: forall id. Int -> StringTable id -> ShowS
showsPrec :: Int -> StringTable id -> ShowS
$cshow :: forall id. StringTable id -> String
show :: StringTable id -> String
$cshowList :: forall id. [StringTable id] -> ShowS
showList :: [StringTable id] -> ShowS
Show, Typeable)
instance (Eq id, Enum id) => Eq (StringTable id) where
StringTable id
tbl1 == :: StringTable id -> StringTable id -> Bool
== StringTable id
tbl2 = StringTable id -> StringTableBuilder id
forall id. Enum id => StringTable id -> StringTableBuilder id
unfinalise StringTable id
tbl1 StringTableBuilder id -> StringTableBuilder id -> Bool
forall a. Eq a => a -> a -> Bool
== StringTable id -> StringTableBuilder id
forall id. Enum id => StringTable id -> StringTableBuilder id
unfinalise StringTable id
tbl2
lookup :: Enum id => StringTable id -> BS.ByteString -> Maybe id
lookup :: forall id. Enum id => StringTable id -> ByteString -> Maybe id
lookup (StringTable ByteString
bs UArray Int32 Word32
offsets UArray Int32 Int32
ids UArray Int32 Int32
_ixs) ByteString
str =
Int32 -> Int32 -> ByteString -> Maybe id
forall {a}. Enum a => Int32 -> Int32 -> ByteString -> Maybe a
binarySearch Int32
0 (Int32
topBoundInt32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
-Int32
1) ByteString
str
where
(Int32
0, Int32
topBound) = UArray Int32 Word32 -> (Int32, Int32)
forall i. Ix i => UArray i Word32 -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Int32 Word32
offsets
binarySearch :: Int32 -> Int32 -> ByteString -> Maybe a
binarySearch !Int32
a !Int32
b !ByteString
key
| Int32
a Int32 -> Int32 -> Bool
forall a. Ord a => a -> a -> Bool
> Int32
b = Maybe a
forall a. Maybe a
Nothing
| Bool
otherwise = case ByteString -> ByteString -> Ordering
forall a. Ord a => a -> a -> Ordering
compare ByteString
key (ByteString -> UArray Int32 Word32 -> Int32 -> ByteString
index' ByteString
bs UArray Int32 Word32
offsets Int32
mid) of
Ordering
LT -> Int32 -> Int32 -> ByteString -> Maybe a
binarySearch Int32
a (Int32
midInt32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
-Int32
1) ByteString
key
Ordering
EQ -> a -> Maybe a
forall a. a -> Maybe a
Just (a -> Maybe a) -> a -> Maybe a
forall a b. (a -> b) -> a -> b
$! Int -> a
forall a. Enum a => Int -> a
toEnum (Int32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (UArray Int32 Int32
ids UArray Int32 Int32 -> Int32 -> Int32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! Int32
mid))
Ordering
GT -> Int32 -> Int32 -> ByteString -> Maybe a
binarySearch (Int32
midInt32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
+Int32
1) Int32
b ByteString
key
where mid :: Int32
mid = (Int32
a Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
+ Int32
b) Int32 -> Int32 -> Int32
forall a. Integral a => a -> a -> a
`div` Int32
2
index' :: BS.ByteString -> A.UArray Int32 Word32 -> Int32 -> BS.ByteString
index' :: ByteString -> UArray Int32 Word32 -> Int32 -> ByteString
index' ByteString
bs UArray Int32 Word32
offsets Int32
i = Int -> ByteString -> ByteString
BS.unsafeTake Int
len (ByteString -> ByteString)
-> (ByteString -> ByteString) -> ByteString -> ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> ByteString -> ByteString
BS.unsafeDrop Int
start (ByteString -> ByteString) -> ByteString -> ByteString
forall a b. (a -> b) -> a -> b
$ ByteString
bs
where
start, end, len :: Int
start :: Int
start = Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (UArray Int32 Word32
offsets UArray Int32 Word32 -> Int32 -> Word32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! Int32
i)
end :: Int
end = Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (UArray Int32 Word32
offsets UArray Int32 Word32 -> Int32 -> Word32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! (Int32
iInt32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
+Int32
1))
len :: Int
len = Int
end Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
start
index :: Enum id => StringTable id -> id -> BS.ByteString
index :: forall id. Enum id => StringTable id -> id -> ByteString
index (StringTable ByteString
bs UArray Int32 Word32
offsets UArray Int32 Int32
_ids UArray Int32 Int32
ixs) =
ByteString -> UArray Int32 Word32 -> Int32 -> ByteString
index' ByteString
bs UArray Int32 Word32
offsets (Int32 -> ByteString) -> (id -> Int32) -> id -> ByteString
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (UArray Int32 Int32
ixs UArray Int32 Int32 -> Int32 -> Int32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
!) (Int32 -> Int32) -> (id -> Int32) -> id -> Int32
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int -> Int32) -> (id -> Int) -> id -> Int32
forall b c a. (b -> c) -> (a -> b) -> a -> c
. id -> Int
forall a. Enum a => a -> Int
fromEnum
construct :: Enum id => [BS.ByteString] -> StringTable id
construct :: forall id. Enum id => [ByteString] -> StringTable id
construct = StringTableBuilder id -> StringTable id
forall id. Enum id => StringTableBuilder id -> StringTable id
finalise (StringTableBuilder id -> StringTable id)
-> ([ByteString] -> StringTableBuilder id)
-> [ByteString]
-> StringTable id
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (StringTableBuilder id -> ByteString -> StringTableBuilder id)
-> StringTableBuilder id -> [ByteString] -> StringTableBuilder id
forall b a. (b -> a -> b) -> b -> [a] -> b
forall (t :: * -> *) b a.
Foldable t =>
(b -> a -> b) -> b -> t a -> b
foldl' (\StringTableBuilder id
tbl ByteString
s -> (StringTableBuilder id, id) -> StringTableBuilder id
forall a b. (a, b) -> a
fst (ByteString -> StringTableBuilder id -> (StringTableBuilder id, id)
forall id.
Enum id =>
ByteString -> StringTableBuilder id -> (StringTableBuilder id, id)
insert ByteString
s StringTableBuilder id
tbl)) StringTableBuilder id
forall id. StringTableBuilder id
empty
data StringTableBuilder id = StringTableBuilder
!(Map BS.ByteString id)
{-# UNPACK #-} !Word32
deriving (StringTableBuilder id -> StringTableBuilder id -> Bool
(StringTableBuilder id -> StringTableBuilder id -> Bool)
-> (StringTableBuilder id -> StringTableBuilder id -> Bool)
-> Eq (StringTableBuilder id)
forall id.
Eq id =>
StringTableBuilder id -> StringTableBuilder id -> Bool
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: forall id.
Eq id =>
StringTableBuilder id -> StringTableBuilder id -> Bool
== :: StringTableBuilder id -> StringTableBuilder id -> Bool
$c/= :: forall id.
Eq id =>
StringTableBuilder id -> StringTableBuilder id -> Bool
/= :: StringTableBuilder id -> StringTableBuilder id -> Bool
Eq, Int -> StringTableBuilder id -> ShowS
[StringTableBuilder id] -> ShowS
StringTableBuilder id -> String
(Int -> StringTableBuilder id -> ShowS)
-> (StringTableBuilder id -> String)
-> ([StringTableBuilder id] -> ShowS)
-> Show (StringTableBuilder id)
forall id. Show id => Int -> StringTableBuilder id -> ShowS
forall id. Show id => [StringTableBuilder id] -> ShowS
forall id. Show id => StringTableBuilder id -> String
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: forall id. Show id => Int -> StringTableBuilder id -> ShowS
showsPrec :: Int -> StringTableBuilder id -> ShowS
$cshow :: forall id. Show id => StringTableBuilder id -> String
show :: StringTableBuilder id -> String
$cshowList :: forall id. Show id => [StringTableBuilder id] -> ShowS
showList :: [StringTableBuilder id] -> ShowS
Show, Typeable)
empty :: StringTableBuilder id
empty :: forall id. StringTableBuilder id
empty = Map ByteString id -> Word32 -> StringTableBuilder id
forall id. Map ByteString id -> Word32 -> StringTableBuilder id
StringTableBuilder Map ByteString id
forall k a. Map k a
Map.empty Word32
0
insert :: Enum id => BS.ByteString -> StringTableBuilder id -> (StringTableBuilder id, id)
insert :: forall id.
Enum id =>
ByteString -> StringTableBuilder id -> (StringTableBuilder id, id)
insert ByteString
str builder :: StringTableBuilder id
builder@(StringTableBuilder Map ByteString id
smap Word32
nextid) =
case ByteString -> Map ByteString id -> Maybe id
forall k a. Ord k => k -> Map k a -> Maybe a
Map.lookup ByteString
str Map ByteString id
smap of
Just id
id -> (StringTableBuilder id
builder, id
id)
Maybe id
Nothing -> let !id :: id
id = Int -> id
forall a. Enum a => Int -> a
toEnum (Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral Word32
nextid)
!smap' :: Map ByteString id
smap' = ByteString -> id -> Map ByteString id -> Map ByteString id
forall k a. Ord k => k -> a -> Map k a -> Map k a
Map.insert ByteString
str id
id Map ByteString id
smap
in (Map ByteString id -> Word32 -> StringTableBuilder id
forall id. Map ByteString id -> Word32 -> StringTableBuilder id
StringTableBuilder Map ByteString id
smap' (Word32
nextidWord32 -> Word32 -> Word32
forall a. Num a => a -> a -> a
+Word32
1), id
id)
inserts :: Enum id => [BS.ByteString] -> StringTableBuilder id -> (StringTableBuilder id, [id])
inserts :: forall id.
Enum id =>
[ByteString]
-> StringTableBuilder id -> (StringTableBuilder id, [id])
inserts [ByteString]
bss StringTableBuilder id
builder = (StringTableBuilder id
-> ByteString -> (StringTableBuilder id, id))
-> StringTableBuilder id
-> [ByteString]
-> (StringTableBuilder id, [id])
forall (t :: * -> *) s a b.
Traversable t =>
(s -> a -> (s, b)) -> s -> t a -> (s, t b)
mapAccumL ((ByteString
-> StringTableBuilder id -> (StringTableBuilder id, id))
-> StringTableBuilder id
-> ByteString
-> (StringTableBuilder id, id)
forall a b c. (a -> b -> c) -> b -> a -> c
flip ByteString -> StringTableBuilder id -> (StringTableBuilder id, id)
forall id.
Enum id =>
ByteString -> StringTableBuilder id -> (StringTableBuilder id, id)
insert) StringTableBuilder id
builder [ByteString]
bss
finalise :: Enum id => StringTableBuilder id -> StringTable id
finalise :: forall id. Enum id => StringTableBuilder id -> StringTable id
finalise (StringTableBuilder Map ByteString id
smap Word32
_) =
(ByteString
-> UArray Int32 Word32
-> UArray Int32 Int32
-> UArray Int32 Int32
-> StringTable id
forall id.
ByteString
-> UArray Int32 Word32
-> UArray Int32 Int32
-> UArray Int32 Int32
-> StringTable id
StringTable ByteString
strs UArray Int32 Word32
offsets UArray Int32 Int32
ids UArray Int32 Int32
ixs)
where
strs :: ByteString
strs = [ByteString] -> ByteString
BS.concat (Map ByteString id -> [ByteString]
forall k a. Map k a -> [k]
Map.keys Map ByteString id
smap)
offsets :: UArray Int32 Word32
offsets = (Int32, Int32) -> [Word32] -> UArray Int32 Word32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [e] -> a i e
A.listArray (Int32
0, Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Map ByteString id -> Int
forall k a. Map k a -> Int
Map.size Map ByteString id
smap))
([Word32] -> UArray Int32 Word32)
-> ([ByteString] -> [Word32])
-> [ByteString]
-> UArray Int32 Word32
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Word32 -> ByteString -> Word32)
-> Word32 -> [ByteString] -> [Word32]
forall b a. (b -> a -> b) -> b -> [a] -> [b]
scanl (\Word32
off ByteString
str -> Word32
off Word32 -> Word32 -> Word32
forall a. Num a => a -> a -> a
+ Int -> Word32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int
BS.length ByteString
str)) Word32
0
([ByteString] -> UArray Int32 Word32)
-> [ByteString] -> UArray Int32 Word32
forall a b. (a -> b) -> a -> b
$ Map ByteString id -> [ByteString]
forall k a. Map k a -> [k]
Map.keys Map ByteString id
smap
ids :: UArray Int32 Int32
ids = (Int32, Int32) -> [Int32] -> UArray Int32 Int32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [e] -> a i e
A.listArray (Int32
0, Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Map ByteString id -> Int
forall k a. Map k a -> Int
Map.size Map ByteString id
smap) Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1)
([Int32] -> UArray Int32 Int32)
-> ([id] -> [Int32]) -> [id] -> UArray Int32 Int32
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (id -> Int32) -> [id] -> [Int32]
forall a b. (a -> b) -> [a] -> [b]
map (Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int -> Int32) -> (id -> Int) -> id -> Int32
forall b c a. (b -> c) -> (a -> b) -> a -> c
. id -> Int
forall a. Enum a => a -> Int
fromEnum)
([id] -> UArray Int32 Int32) -> [id] -> UArray Int32 Int32
forall a b. (a -> b) -> a -> b
$ Map ByteString id -> [id]
forall k a. Map k a -> [a]
Map.elems Map ByteString id
smap
ixs :: UArray Int32 Int32
ixs = (Int32, Int32) -> [(Int32, Int32)] -> UArray Int32 Int32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [(i, e)] -> a i e
A.array (UArray Int32 Int32 -> (Int32, Int32)
forall i. Ix i => UArray i Int32 -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Int32 Int32
ids) [ (Int32
id,Int32
ix) | (Int32
ix,Int32
id) <- UArray Int32 Int32 -> [(Int32, Int32)]
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> [(i, e)]
A.assocs UArray Int32 Int32
ids ]
unfinalise :: Enum id => StringTable id -> StringTableBuilder id
unfinalise :: forall id. Enum id => StringTable id -> StringTableBuilder id
unfinalise (StringTable ByteString
strs UArray Int32 Word32
offsets UArray Int32 Int32
ids UArray Int32 Int32
_) =
Map ByteString id -> Word32 -> StringTableBuilder id
forall id. Map ByteString id -> Word32 -> StringTableBuilder id
StringTableBuilder Map ByteString id
smap Word32
nextid
where
smap :: Map ByteString id
smap = [(ByteString, id)] -> Map ByteString id
forall k a. Eq k => [(k, a)] -> Map k a
Map.fromAscList
[ (ByteString -> UArray Int32 Word32 -> Int32 -> ByteString
index' ByteString
strs UArray Int32 Word32
offsets Int32
ix, Int -> id
forall a. Enum a => Int -> a
toEnum (Int32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (UArray Int32 Int32
ids UArray Int32 Int32 -> Int32 -> Int32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> i -> e
! Int32
ix)))
| Int32
ix <- [Int32
0..Int32
h] ]
(Int32
0,Int32
h) = UArray Int32 Int32 -> (Int32, Int32)
forall i. Ix i => UArray i Int32 -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Int32 Int32
ids
nextid :: Word32
nextid = Int32 -> Word32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Int32
hInt32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
+Int32
1)
serialise :: StringTable id -> BS.Builder
serialise :: forall id. StringTable id -> Builder
serialise (StringTable ByteString
strs UArray Int32 Word32
offs UArray Int32 Int32
ids UArray Int32 Int32
ixs) =
let (Int32
_, !Int32
ixEnd) = UArray Int32 Word32 -> (Int32, Int32)
forall i. Ix i => UArray i Word32 -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Int32 Word32
offs in
Word32 -> Builder
BS.word32BE (Int -> Word32
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int
BS.length ByteString
strs))
Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> Word32 -> Builder
BS.word32BE (Int32 -> Word32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int32
ixEnd Word32 -> Word32 -> Word32
forall a. Num a => a -> a -> a
+ Word32
1)
Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> ByteString -> Builder
BS.byteStringCopy ByteString
strs
Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> (Word32 -> Builder -> Builder) -> Builder -> [Word32] -> Builder
forall a b. (a -> b -> b) -> b -> [a] -> b
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr (\Word32
n Builder
r -> Word32 -> Builder
BS.word32BE Word32
n Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> Builder
r) Builder
forall a. Monoid a => a
mempty (UArray Int32 Word32 -> [Word32]
forall (a :: * -> * -> *) e i. (IArray a e, Ix i) => a i e -> [e]
A.elems UArray Int32 Word32
offs)
Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> (Int32 -> Builder -> Builder) -> Builder -> [Int32] -> Builder
forall a b. (a -> b -> b) -> b -> [a] -> b
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr (\Int32
n Builder
r -> Int32 -> Builder
BS.int32BE Int32
n Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> Builder
r) Builder
forall a. Monoid a => a
mempty (UArray Int32 Int32 -> [Int32]
forall (a :: * -> * -> *) e i. (IArray a e, Ix i) => a i e -> [e]
A.elems UArray Int32 Int32
ids)
Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> (Int32 -> Builder -> Builder) -> Builder -> [Int32] -> Builder
forall a b. (a -> b -> b) -> b -> [a] -> b
forall (t :: * -> *) a b.
Foldable t =>
(a -> b -> b) -> b -> t a -> b
foldr (\Int32
n Builder
r -> Int32 -> Builder
BS.int32BE Int32
n Builder -> Builder -> Builder
forall a. Semigroup a => a -> a -> a
<> Builder
r) Builder
forall a. Monoid a => a
mempty (UArray Int32 Int32 -> [Int32]
forall (a :: * -> * -> *) e i. (IArray a e, Ix i) => a i e -> [e]
A.elems UArray Int32 Int32
ixs)
serialiseSize :: StringTable id -> Int
serialiseSize :: forall id. StringTable id -> Int
serialiseSize (StringTable ByteString
strs UArray Int32 Word32
offs UArray Int32 Int32
_ids UArray Int32 Int32
_ixs) =
let (Int32
_, !Int32
ixEnd) = UArray Int32 Word32 -> (Int32, Int32)
forall i. Ix i => UArray i Word32 -> (i, i)
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
a i e -> (i, i)
A.bounds UArray Int32 Word32
offs
in Int
4 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
2
Int -> Int -> Int
forall a. Num a => a -> a -> a
+ ByteString -> Int
BS.length ByteString
strs
Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
4 Int -> Int -> Int
forall a. Num a => a -> a -> a
* (Int32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int32
ixEnd Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
8 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int32
ixEnd
deserialiseV1 :: BS.ByteString -> Maybe (StringTable id, BS.ByteString)
deserialiseV1 :: forall id. ByteString -> Maybe (StringTable id, ByteString)
deserialiseV1 ByteString
bs
| ByteString -> Int
BS.length ByteString
bs Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
8
, let lenStrs :: Int
lenStrs = Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int -> Word32
readWord32BE ByteString
bs Int
0)
lenArr :: Int
lenArr = Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int -> Word32
readWord32BE ByteString
bs Int
1)
lenTotal :: Int
lenTotal= Int
8 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
lenStrs Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
4 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
lenArr
, ByteString -> Int
BS.length ByteString
bs Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
lenTotal
, let strs :: ByteString
strs = Int -> ByteString -> ByteString
BS.unsafeTake Int
lenStrs (Int -> ByteString -> ByteString
BS.unsafeDrop Int
8 ByteString
bs)
arr :: UArray Int32 Word32
arr = (Int32, Int32) -> [(Int32, Word32)] -> UArray Int32 Word32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [(i, e)] -> a i e
A.array (Int32
0, Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1)
[ (Int32
i, ByteString -> Int -> Word32
readWord32BE ByteString
bs Int
off)
| (Int32
i, Int
off) <- [Int32] -> [Int] -> [(Int32, Int)]
forall a b. [a] -> [b] -> [(a, b)]
zip [Int32
0 .. Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1]
[Int
offArrS,Int
offArrSInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
4 .. Int
offArrE]
]
ids :: UArray Int32 Int32
ids = (Int32, Int32) -> [(Int32, Int32)] -> UArray Int32 Int32
forall (a :: * -> * -> *) e i.
(IArray a e, Ix i) =>
(i, i) -> [(i, e)] -> a i e
A.array (Int32
0, Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1)
[ (Int32
i,Int32
i) | Int32
i <- [Int32
0 .. Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1] ]
ixs :: UArray Int32 Int32
ixs = UArray Int32 Int32
ids
offArrS :: Int
offArrS = Int
8 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
lenStrs
offArrE :: Int
offArrE = Int
offArrS Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
4 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
lenArr Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1
!stringTable :: StringTable id
stringTable = ByteString
-> UArray Int32 Word32
-> UArray Int32 Int32
-> UArray Int32 Int32
-> StringTable id
forall id.
ByteString
-> UArray Int32 Word32
-> UArray Int32 Int32
-> UArray Int32 Int32
-> StringTable id
StringTable ByteString
strs UArray Int32 Word32
arr UArray Int32 Int32
ids UArray Int32 Int32
ixs
!bs' :: ByteString
bs' = Int -> ByteString -> ByteString
BS.drop Int
lenTotal ByteString
bs
= (StringTable id, ByteString) -> Maybe (StringTable id, ByteString)
forall a. a -> Maybe a
Just (StringTable id
forall {id}. StringTable id
stringTable, ByteString
bs')
| Bool
otherwise
= Maybe (StringTable id, ByteString)
forall a. Maybe a
Nothing
deserialiseV2 :: BS.ByteString -> Maybe (StringTable id, BS.ByteString)
deserialiseV2 :: forall id. ByteString -> Maybe (StringTable id, ByteString)
deserialiseV2 ByteString
bs
| ByteString -> Int
BS.length ByteString
bs Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
8
, let lenStrs :: Int
lenStrs = Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int -> Word32
readWord32BE ByteString
bs Int
0)
lenArr :: Int
lenArr = Word32 -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (ByteString -> Int -> Word32
readWord32BE ByteString
bs Int
1)
lenTotal :: Int
lenTotal= Int
8
Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
lenStrs
Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
4 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
lenArr
Int -> Int -> Int
forall a. Num a => a -> a -> a
+(Int
4 Int -> Int -> Int
forall a. Num a => a -> a -> a
* (Int
lenArr Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)) Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
2
, ByteString -> Int
BS.length ByteString
bs Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
>= Int
lenTotal
, let strs :: ByteString
strs = Int -> ByteString -> ByteString
BS.unsafeTake Int
lenStrs (Int -> ByteString -> ByteString
BS.unsafeDrop Int
8 ByteString
bs)
offs_bs :: ByteString
offs_bs = Int -> ByteString -> ByteString
BS.unsafeDrop (Int
8 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
lenStrs) ByteString
bs
ids_bs :: ByteString
ids_bs = Int -> ByteString -> ByteString
BS.unsafeDrop (Int
lenArr Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
4) ByteString
offs_bs
ixs_bs :: ByteString
ixs_bs = Int -> ByteString -> ByteString
BS.unsafeDrop ((Int
lenArr Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
4) ByteString
ids_bs
castArray :: A.UArray i Word32 -> A.UArray i Int32
castArray :: forall i. UArray i Word32 -> UArray i Int32
castArray (A.UArray i
a i
b Int
c ByteArray#
d) = (i -> i -> Int -> ByteArray# -> UArray i Int32
forall i e. i -> i -> Int -> ByteArray# -> UArray i e
A.UArray i
a i
b Int
c ByteArray#
d)
(UArray Int32 Word32
offs, UArray Int32 Int32
ids, UArray Int32 Int32
ixs) = IO (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
-> (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
forall a. IO a -> a
unsafePerformIO (IO (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
-> (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32))
-> IO (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
-> (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
forall a b. (a -> b) -> a -> b
$ do
!UArray Int32 Word32
r1 <- Int32 -> ByteString -> IO (UArray Int32 Word32)
forall i.
(Integral i, Num i) =>
i -> ByteString -> IO (UArray i Word32)
beToLe (Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr) ByteString
offs_bs
!UArray Int32 Int32
r2 <- UArray Int32 Word32 -> UArray Int32 Int32
forall i. UArray i Word32 -> UArray i Int32
castArray (UArray Int32 Word32 -> UArray Int32 Int32)
-> IO (UArray Int32 Word32) -> IO (UArray Int32 Int32)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Int32 -> ByteString -> IO (UArray Int32 Word32)
forall i.
(Integral i, Num i) =>
i -> ByteString -> IO (UArray i Word32)
beToLe (Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1) ByteString
ids_bs
!UArray Int32 Int32
r3 <- UArray Int32 Word32 -> UArray Int32 Int32
forall i. UArray i Word32 -> UArray i Int32
castArray (UArray Int32 Word32 -> UArray Int32 Int32)
-> IO (UArray Int32 Word32) -> IO (UArray Int32 Int32)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Int32 -> ByteString -> IO (UArray Int32 Word32)
forall i.
(Integral i, Num i) =>
i -> ByteString -> IO (UArray i Word32)
beToLe (Int -> Int32
forall a b. (Integral a, Num b) => a -> b
fromIntegral Int
lenArr Int32 -> Int32 -> Int32
forall a. Num a => a -> a -> a
- Int32
1) ByteString
ixs_bs
(UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
-> IO (UArray Int32 Word32, UArray Int32 Int32, UArray Int32 Int32)
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return (UArray Int32 Word32
r1, UArray Int32 Int32
r2, UArray Int32 Int32
r3)
!stringTable :: StringTable id
stringTable = ByteString
-> UArray Int32 Word32
-> UArray Int32 Int32
-> UArray Int32 Int32
-> StringTable id
forall id.
ByteString
-> UArray Int32 Word32
-> UArray Int32 Int32
-> UArray Int32 Int32
-> StringTable id
StringTable ByteString
strs UArray Int32 Word32
offs UArray Int32 Int32
ids UArray Int32 Int32
ixs
!bs_left :: ByteString
bs_left = Int -> ByteString -> ByteString
BS.drop Int
lenTotal ByteString
bs
= (StringTable id, ByteString) -> Maybe (StringTable id, ByteString)
forall a. a -> Maybe a
Just (StringTable id
forall {id}. StringTable id
stringTable, ByteString
bs_left)
| Bool
otherwise
= Maybe (StringTable id, ByteString)
forall a. Maybe a
Nothing