{-# LANGUAGE RecordWildCards #-}
module Data.SegTree.Dual where
import Control.Monad
import Control.Monad.Primitive
import Data.Bits
import Data.Function
import qualified Data.Vector.Unboxed as U
import qualified Data.Vector.Unboxed.Mutable as UM
import My.Prelude
import Data.Monoid.Action (MonoidAction (..))
data DualSegTree s f a = DualSegTree
{ forall s f a. DualSegTree s f a -> MVector s f
dualSegTreeDST :: !(UM.MVector s f)
, forall s f a. DualSegTree s f a -> MVector s a
primalsDST :: !(UM.MVector s a)
, forall s f a. DualSegTree s f a -> Int
dualSizeDST :: !Int
, forall s f a. DualSegTree s f a -> Int
primalSizeDST :: !Int
, forall s f a. DualSegTree s f a -> Int
heightDST :: !Int
}
buildDualSegTree ::
(Monoid f, U.Unbox f, U.Unbox a, PrimMonad m) =>
U.Vector a ->
m (DualSegTree (PrimState m) f a)
buildDualSegTree :: forall f a (m :: * -> *).
(Monoid f, Unbox f, Unbox a, PrimMonad m) =>
Vector a -> m (DualSegTree (PrimState m) f a)
buildDualSegTree Vector a
xs = do
dualSegTreeDST <- Int -> f -> m (MVector (PrimState m) f)
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
Int -> a -> m (MVector (PrimState m) a)
UM.replicate Int
dualSizeDST f
forall a. Monoid a => a
mempty
primalsDST <- U.thaw xs
return $ DualSegTree{..}
where
!dualSizeDST :: Int
dualSizeDST = Int -> Int
extendToPowerOfTwo (Int -> Int) -> Int -> Int
forall a b. (a -> b) -> a -> b
$ Vector a -> Int
forall a. Unbox a => Vector a -> Int
U.length Vector a
xs
primalSizeDST :: Int
primalSizeDST = Vector a -> Int
forall a. Unbox a => Vector a -> Int
U.length Vector a
xs
heightDST :: Int
heightDST = Int -> Int
forall b. FiniteBits b => b -> Int
countTrailingZeros Int
dualSizeDST
{-# INLINE buildDualSegTree #-}
freezeDualSegTree ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
m (U.Vector a)
freezeDualSegTree :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> m (Vector a)
freezeDualSegTree
DualSegTree
{ dualSegTreeDST :: forall s f a. DualSegTree s f a -> MVector s f
dualSegTreeDST = MVector (PrimState m) f
dseg
, primalsDST :: forall s f a. DualSegTree s f a -> MVector s a
primalsDST = MVector (PrimState m) a
pseg
, dualSizeDST :: forall s f a. DualSegTree s f a -> Int
dualSizeDST = Int
dsize
} = do
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rep1 ((Int
dsize Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
1) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
fi <- MVector (PrimState m) f -> Int -> m f
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead MVector (PrimState m) f
dseg Int
i
UM.unsafeModify dseg (fi <>) (2 * i)
UM.unsafeModify dseg (fi <>) (2 * i + 1)
((Int -> a -> m ()) -> MVector (PrimState m) a -> m ())
-> MVector (PrimState m) a -> (Int -> a -> m ()) -> m ()
forall a b c. (a -> b -> c) -> b -> a -> c
flip (Int -> a -> m ()) -> MVector (PrimState m) a -> m ()
forall (m :: * -> *) a b.
(PrimMonad m, Unbox a) =>
(Int -> a -> m b) -> MVector (PrimState m) a -> m ()
UM.imapM_ MVector (PrimState m) a
pseg ((Int -> a -> m ()) -> m ()) -> (Int -> a -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i a
x -> do
f <- MVector (PrimState m) f -> Int -> m f
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead MVector (PrimState m) f
dseg (Int -> m f) -> Int -> m f
forall a b. (a -> b) -> a -> b
$ (Int
i Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
dsize) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
1
UM.unsafeWrite pseg i (mact f x)
MVector (PrimState m) f -> f -> m ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> a -> m ()
UM.set MVector (PrimState m) f
dseg f
forall a. Monoid a => a
mempty
MVector (PrimState m) a -> m (Vector a)
forall a (m :: * -> *).
(Unbox a, PrimMonad m) =>
MVector (PrimState m) a -> m (Vector a)
U.freeze MVector (PrimState m) a
pseg
{-# INLINE freezeDualSegTree #-}
readDualSegTree ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
Int ->
m a
readDualSegTree :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m a
readDualSegTree
seg :: DualSegTree (PrimState m) f a
seg@DualSegTree
{ primalsDST :: forall s f a. DualSegTree s f a -> MVector s a
primalsDST = MVector (PrimState m) a
pseg
, dualSizeDST :: forall s f a. DualSegTree s f a -> Int
dualSizeDST = Int
dsize
, heightDST :: forall s f a. DualSegTree s f a -> Int
heightDST = Int
height
}
Int
k0 = do
let !k :: Int
k = Int
k0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
dsize
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 Int
height ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
DualSegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m ()
pushSegTree DualSegTree (PrimState m) f a
seg (Int
k Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead MVector (PrimState m) a
pseg Int
k0
{-# INLINE readDualSegTree #-}
writeDualSegTree ::
(MonoidAction f a, U.Unbox a, U.Unbox f, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
Int ->
a ->
m ()
writeDualSegTree :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox a, Unbox f, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> a -> m ()
writeDualSegTree
seg :: DualSegTree (PrimState m) f a
seg@DualSegTree
{ primalsDST :: forall s f a. DualSegTree s f a -> MVector s a
primalsDST = MVector (PrimState m) a
pseg
, dualSizeDST :: forall s f a. DualSegTree s f a -> Int
dualSizeDST = Int
dsize
, heightDST :: forall s f a. DualSegTree s f a -> Int
heightDST = Int
height
}
Int
k0
a
v = do
let !k :: Int
k = Int
k0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
dsize
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 Int
height ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
DualSegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m ()
pushSegTree DualSegTree (PrimState m) f a
seg (Int
k Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
MVector (PrimState m) a -> Int -> a -> m ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> a -> m ()
UM.unsafeWrite MVector (PrimState m) a
pseg Int
k0 a
v
{-# INLINE writeDualSegTree #-}
modifyDualSegTree ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
(a -> a) ->
Int ->
m ()
modifyDualSegTree :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> (a -> a) -> Int -> m ()
modifyDualSegTree
seg :: DualSegTree (PrimState m) f a
seg@DualSegTree
{ primalsDST :: forall s f a. DualSegTree s f a -> MVector s a
primalsDST = MVector (PrimState m) a
pseg
, dualSizeDST :: forall s f a. DualSegTree s f a -> Int
dualSizeDST = Int
dsize
, heightDST :: forall s f a. DualSegTree s f a -> Int
heightDST = Int
height
}
a -> a
f
Int
k0 = do
let !k :: Int
k = Int
k0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
dsize
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 Int
height ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
DualSegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m ()
pushSegTree DualSegTree (PrimState m) f a
seg (Int
k Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
MVector (PrimState m) a -> (a -> a) -> Int -> m ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> (a -> a) -> Int -> m ()
UM.unsafeModify MVector (PrimState m) a
pseg a -> a
f Int
k0
{-# INLINE modifyDualSegTree #-}
appAt ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
Int ->
f ->
m ()
appAt :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> f -> m ()
appAt DualSegTree (PrimState m) f a
st Int
k f
f = DualSegTree (PrimState m) f a -> (a -> a) -> Int -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> (a -> a) -> Int -> m ()
modifyDualSegTree DualSegTree (PrimState m) f a
st (f -> a -> a
forall f a. MonoidAction f a => f -> a -> a
mact f
f) Int
k
{-# INLINE appAt #-}
appFromTo ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
Int ->
Int ->
f ->
m ()
appFromTo :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> Int -> f -> m ()
appFromTo
seg :: DualSegTree (PrimState m) f a
seg@DualSegTree
{ dualSizeDST :: forall s f a. DualSegTree s f a -> Int
dualSizeDST = Int
dsize
, heightDST :: forall s f a. DualSegTree s f a -> Int
heightDST = Int
height
}
Int
l0
Int
r0
f
f = Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
l0 Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
r0) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
let !l :: Int
l = Int
l0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
dsize
!r :: Int
r = Int
r0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
dsize
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 Int
height ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when ((Int
l Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!<<. Int
i Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
/= Int
l) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
DualSegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m ()
pushSegTree DualSegTree (PrimState m) f a
seg (Int
l Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when ((Int
r Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!<<. Int
i Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
/= Int
r) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
DualSegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m ()
pushSegTree DualSegTree (PrimState m) f a
seg ((Int
r Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
((Int -> Int -> m ()) -> Int -> Int -> m ()) -> Int -> Int -> m ()
forall a. (a -> a) -> a
fix
( \Int -> Int -> m ()
loop !Int
l' !Int
r' -> Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
l' Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
r') (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
l' Int -> Int -> Int
forall a. Bits a => a -> a -> a
.&. Int
1 Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
1) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
DualSegTree (PrimState m) f a -> Int -> f -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> f -> m ()
evalAt DualSegTree (PrimState m) f a
seg Int
l' f
f
Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
r' Int -> Int -> Int
forall a. Bits a => a -> a -> a
.&. Int
1 Int -> Int -> Bool
forall a. Eq a => a -> a -> Bool
== Int
1) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
DualSegTree (PrimState m) f a -> Int -> f -> m ()
forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> f -> m ()
evalAt DualSegTree (PrimState m) f a
seg (Int
r' Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) f
f
Int -> Int -> m ()
loop ((Int
l' Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
1) (Int
r' Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
1)
)
Int
l
Int
r
{-# INLINE appFromTo #-}
evalAt ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
Int ->
f ->
m ()
evalAt :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> f -> m ()
evalAt
DualSegTree
{ dualSegTreeDST :: forall s f a. DualSegTree s f a -> MVector s f
dualSegTreeDST = MVector (PrimState m) f
dseg
, primalsDST :: forall s f a. DualSegTree s f a -> MVector s a
primalsDST = MVector (PrimState m) a
pseg
, dualSizeDST :: forall s f a. DualSegTree s f a -> Int
dualSizeDST = Int
dsize
, primalSizeDST :: forall s f a. DualSegTree s f a -> Int
primalSizeDST = Int
psize
}
Int
k
f
f = do
if Int
k Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
dsize
then MVector (PrimState m) f -> (f -> f) -> Int -> m ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> (a -> a) -> Int -> m ()
UM.unsafeModify MVector (PrimState m) f
dseg (f
f f -> f -> f
forall a. Semigroup a => a -> a -> a
<>) Int
k
else do
let !k' :: Int
k' = Int
k Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
dsize
Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
k' Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
psize) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
MVector (PrimState m) a -> (a -> a) -> Int -> m ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> (a -> a) -> Int -> m ()
UM.unsafeModify MVector (PrimState m) a
pseg (f -> a -> a
forall f a. MonoidAction f a => f -> a -> a
mact f
f) Int
k'
{-# INLINE evalAt #-}
pushSegTree ::
(MonoidAction f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a ->
Int ->
m ()
pushSegTree :: forall f a (m :: * -> *).
(MonoidAction f a, Unbox f, Unbox a, PrimMonad m) =>
DualSegTree (PrimState m) f a -> Int -> m ()
pushSegTree seg :: DualSegTree (PrimState m) f a
seg@DualSegTree{dualSegTreeDST :: forall s f a. DualSegTree s f a -> MVector s f
dualSegTreeDST = MVector (PrimState m) f
dseg} Int
k = do
fk <- MVector (PrimState m) f -> Int -> m f
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead MVector (PrimState m) f
dseg Int
k
UM.unsafeWrite dseg k mempty
evalAt seg (2 * k) fk
evalAt seg (2 * k + 1) fk
{-# INLINE pushSegTree #-}
extendToPowerOfTwo :: Int -> Int
extendToPowerOfTwo :: Int -> Int
extendToPowerOfTwo Int
x
| Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
1 = Int -> Int -> Int
unsafeShiftRL (-Int
1) (Int -> Int
forall b. FiniteBits b => b -> Int
countLeadingZeros (Int
x Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1)) Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1
| Bool
otherwise = Int
1