module Data.ByteString.ZAlgorithm where
import Control.Monad
import qualified Data.ByteString as B
import qualified Data.ByteString.Unsafe as B
import Data.Function (fix)
import qualified Data.Vector.Unboxed as U
import qualified Data.Vector.Unboxed.Mutable as UM
zAlgorithm :: B.ByteString -> U.Vector Int
zAlgorithm :: ByteString -> Vector Int
zAlgorithm ByteString
bs = (forall s. ST s (MVector s Int)) -> Vector Int
forall a. Unbox a => (forall s. ST s (MVector s a)) -> Vector a
U.create ((forall s. ST s (MVector s Int)) -> Vector Int)
-> (forall s. ST s (MVector s Int)) -> Vector Int
forall a b. (a -> b) -> a -> b
$ do
z <- Int -> ST s (MVector (PrimState (ST s)) Int)
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
Int -> m (MVector (PrimState m) a)
UM.unsafeNew Int
n
fix
( \Int -> Int -> Int -> ST s ()
loop !Int
zl !Int
zr !Int
l -> Bool -> ST s () -> ST s ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
l Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
n) (ST s () -> ST s ()) -> ST s () -> ST s ()
forall a b. (a -> b) -> a -> b
$ do
if Int
zr Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
l
then do
let !r :: Int
r = Int -> Int -> Int
lcp Int
0 Int
l
MVector (PrimState (ST s)) Int -> Int -> Int -> ST s ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> a -> m ()
UM.unsafeWrite MVector s Int
MVector (PrimState (ST s)) Int
z Int
l (Int
r Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
l)
Int -> Int -> Int -> ST s ()
loop Int
l Int
r (Int
l Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
else do
zk <- MVector (PrimState (ST s)) Int -> Int -> ST s Int
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead MVector s Int
MVector (PrimState (ST s)) Int
z (Int
l Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
zl)
if zk < zr - l
then do
UM.unsafeWrite z l zk
loop zl zr (l + 1)
else do
let !r = Int -> Int -> Int
lcp (Int
zr Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
l) Int
zr
UM.unsafeWrite z l (r - l)
loop l r (l + 1)
)
0
1
1
when (n > 0) $ do
UM.write z 0 n
return z
where
!n :: Int
n = ByteString -> Int
B.length ByteString
bs
lcp :: Int -> Int -> Int
lcp :: Int -> Int -> Int
lcp = ((Int -> Int -> Int) -> Int -> Int -> Int) -> Int -> Int -> Int
forall a. (a -> a) -> a
fix (((Int -> Int -> Int) -> Int -> Int -> Int) -> Int -> Int -> Int)
-> ((Int -> Int -> Int) -> Int -> Int -> Int) -> Int -> Int -> Int
forall a b. (a -> b) -> a -> b
$ \Int -> Int -> Int
loop !Int
l !Int
r ->
if Int
r Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
n Bool -> Bool -> Bool
&& ByteString -> Int -> Word8
B.unsafeIndex ByteString
bs Int
l Word8 -> Word8 -> Bool
forall a. Eq a => a -> a -> Bool
== ByteString -> Int -> Word8
B.unsafeIndex ByteString
bs Int
r
then Int -> Int -> Int
loop (Int
l Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1) (Int
r Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
else Int
r