module Data.SegTree 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 (rep1, rev1, unsafeShiftRL)
class (Monoid f, Semigroup s) => AsSemigroupEndo f s where
sendo :: f -> (s -> s)
data SegTree s f a = SegTree
{ forall s f a. SegTree s f a -> MVector s a
getSegTree :: !(UM.MVector s a)
, forall s f a. SegTree s f a -> MVector s f
getDualSegTree :: !(UM.MVector s f)
, forall s f a. SegTree s f a -> Int
sizeSegTree :: !Int
, forall s f a. SegTree s f a -> Int
heightSegTree :: !Int
}
newSegTree ::
(Monoid f, U.Unbox f, Monoid a, U.Unbox a, PrimMonad m) =>
Int ->
m (SegTree (PrimState m) f a)
newSegTree :: forall f a (m :: * -> *).
(Monoid f, Unbox f, Monoid a, Unbox a, PrimMonad m) =>
Int -> m (SegTree (PrimState m) f a)
newSegTree Int
n0 = do
seg <- Int -> a -> m (MVector (PrimState m) a)
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
Int -> a -> m (MVector (PrimState m) a)
UM.replicate (Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
n) a
forall a. Monoid a => a
mempty
dseg <- UM.replicate n mempty
return $ SegTree seg dseg n (63 - countLeadingZeros n)
where
!n :: Int
n = Int -> Int
extendToPowerOfTwo Int
n0
buildSegTree ::
(Monoid f, U.Unbox f, Monoid a, U.Unbox a, PrimMonad m) =>
U.Vector a ->
m (SegTree (PrimState m) f a)
buildSegTree :: forall f a (m :: * -> *).
(Monoid f, Unbox f, Monoid a, Unbox a, PrimMonad m) =>
Vector a -> m (SegTree (PrimState m) f a)
buildSegTree Vector a
xs = do
seg <- Int -> a -> m (MVector (PrimState m) a)
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
Int -> a -> m (MVector (PrimState m) a)
UM.replicate (Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
n) a
forall a. Monoid a => a
mempty
dseg <- UM.replicate n mempty
U.unsafeCopy (UM.unsafeSlice n (U.length xs) seg) xs
let st = MVector (PrimState m) a
-> MVector (PrimState m) f
-> Int
-> Int
-> SegTree (PrimState m) f a
forall s f a.
MVector s a -> MVector s f -> Int -> Int -> SegTree s f a
SegTree MVector (PrimState m) a
seg MVector (PrimState m) f
dseg Int
n (Int
63 Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int -> Int
forall b. FiniteBits b => b -> Int
countLeadingZeros Int
n)
rev1 (n - 1) $ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall a (m :: * -> *) f.
(Semigroup a, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pullSegTree SegTree (PrimState m) f a
st Int
i
return st
where
!n :: Int
n = 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
readSegTree ::
(AsSemigroupEndo f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
m a
readSegTree :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m a
readSegTree SegTree (PrimState m) f a
st Int
k0 = do
let !k :: Int
k = Int
k0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st (Int -> Int -> Int
forall a. Bits a => a -> Int -> a
unsafeShiftR Int
k 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 (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
k
{-# INLINE readSegTree #-}
writeSegTree ::
(AsSemigroupEndo f a, Semigroup a, U.Unbox a, U.Unbox f, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
a ->
m ()
writeSegTree :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Semigroup a, Unbox a, Unbox f,
PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> a -> m ()
writeSegTree SegTree (PrimState m) f a
st Int
k0 a
v = do
let !k :: Int
k = Int
k0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st (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 (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
k a
v
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rep1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall a (m :: * -> *) f.
(Semigroup a, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pullSegTree SegTree (PrimState m) f a
st (Int
k Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
{-# INLINE writeSegTree #-}
modifySegTree ::
(AsSemigroupEndo f a, Semigroup a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
(a -> a) ->
Int ->
m ()
modifySegTree :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Semigroup a, Unbox f, Unbox a,
PrimMonad m) =>
SegTree (PrimState m) f a -> (a -> a) -> Int -> m ()
modifySegTree SegTree (PrimState m) f a
st a -> a
f Int
k0 = do
let !k :: Int
k = Int
k0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st (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 (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) a -> a
f Int
k
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rep1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall a (m :: * -> *) f.
(Semigroup a, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pullSegTree SegTree (PrimState m) f a
st (Int
k Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
{-# INLINE modifySegTree #-}
mappendFromTo ::
(AsSemigroupEndo f a, Monoid a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
Int ->
m a
mappendFromTo :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Monoid a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> Int -> m a
mappendFromTo SegTree (PrimState m) f a
st Int
l0 Int
r0 = do
let !l :: Int
l = Int
l0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
!r :: Int
r = Int
r0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((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
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st (Int -> Int -> Int
forall a. Bits a => a -> Int -> a
unsafeShiftR Int
l 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
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st (Int -> Int -> Int
forall a. Bits a => a -> Int -> a
unsafeShiftR Int
r Int
i)
((a -> a -> Int -> Int -> m a) -> a -> a -> Int -> Int -> m a)
-> a -> a -> Int -> Int -> m a
forall a. (a -> a) -> a
fix
( \a -> a -> Int -> Int -> m a
loop !a
accL !a
accR !Int
l' !Int
r' -> do
if Int
l' Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
r'
then do
!accL' <-
if 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
then (a
accL a -> a -> a
forall a. Semigroup a => a -> a -> a
<>) (a -> a) -> m a -> m a
forall (m :: * -> *) a b. Monad m => (a -> b) -> m a -> m b
<$!> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
l'
else a -> m a
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return a
accL
!accR' <-
if r' .&. 1 == 1
then (<> accR) <$!> UM.unsafeRead (getSegTree st) (r' - 1)
else return accR
loop
accL'
accR'
((l' + 1) !>>. 1)
(r' !>>. 1)
else a -> m a
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return (a -> m a) -> a -> m a
forall a b. (a -> b) -> a -> b
$! a
accL a -> a -> a
forall a. Semigroup a => a -> a -> a
<> a
accR
)
a
forall a. Monoid a => a
mempty
a
forall a. Monoid a => a
mempty
Int
l
Int
r
{-# INLINE mappendFromTo #-}
mappendTo ::
(AsSemigroupEndo f a, Monoid a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
m a
mappendTo :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Monoid a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m a
mappendTo SegTree (PrimState m) f a
st = SegTree (PrimState m) f a -> Int -> Int -> m a
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Monoid a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> Int -> m a
mappendFromTo SegTree (PrimState m) f a
st Int
0
{-# INLINE mappendTo #-}
mappendAll :: (U.Unbox a, PrimMonad m) => SegTree (PrimState m) f a -> m a
mappendAll :: forall a (m :: * -> *) f.
(Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> m a
mappendAll SegTree (PrimState m) f a
st = MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
1
{-# INLINE mappendAll #-}
appAt ::
(AsSemigroupEndo f a, Semigroup a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
f ->
m ()
appAt :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Semigroup a, Unbox f, Unbox a,
PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> f -> m ()
appAt SegTree (PrimState m) f a
st Int
k f
f = SegTree (PrimState m) f a -> (a -> a) -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Semigroup a, Unbox f, Unbox a,
PrimMonad m) =>
SegTree (PrimState m) f a -> (a -> a) -> Int -> m ()
modifySegTree SegTree (PrimState m) f a
st (f -> a -> a
forall f s. AsSemigroupEndo f s => f -> s -> s
sendo f
f) Int
k
{-# INLINE appAt #-}
appFromTo ::
(AsSemigroupEndo f a, Semigroup a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
Int ->
f ->
m ()
appFromTo :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Semigroup a, Unbox f, Unbox a,
PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> Int -> f -> m ()
appFromTo SegTree (PrimState m) f a
st 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
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
!r :: Int
r = Int
r0 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((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
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st (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
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st ((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
SegTree (PrimState m) f a -> Int -> f -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> f -> m ()
evalAt SegTree (PrimState m) f a
st 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
SegTree (PrimState m) f a -> Int -> f -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> f -> m ()
evalAt SegTree (PrimState m) f a
st (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
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rep1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((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
SegTree (PrimState m) f a -> Int -> m ()
forall a (m :: * -> *) f.
(Semigroup a, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pullSegTree SegTree (PrimState m) f a
st (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
SegTree (PrimState m) f a -> Int -> m ()
forall a (m :: * -> *) f.
(Semigroup a, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pullSegTree SegTree (PrimState m) f a
st ((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)
{-# INLINE appFromTo #-}
upperBoundFrom ::
(AsSemigroupEndo f a, Monoid a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
(a -> Bool) ->
m Int
upperBoundFrom :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Monoid a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> (a -> Bool) -> m Int
upperBoundFrom SegTree (PrimState m) f a
st Int
l a -> Bool
p = do
let !n :: Int
n = SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st ((Int
l Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
n) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
violationNode <-
((a -> Int -> m (Maybe (a, Int)))
-> a -> Int -> m (Maybe (a, Int)))
-> a -> Int -> m (Maybe (a, Int))
forall a. (a -> a) -> a
fix
( \a -> Int -> m (Maybe (a, Int))
loopUp !a
acc !Int
cur -> do
let rightParent :: Int
rightParent = Int
cur Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int -> Int
forall b. FiniteBits b => b -> Int
countTrailingZeros Int
cur
!acc' <- (a
acc a -> a -> a
forall a. Semigroup a => a -> a -> a
<>) (a -> a) -> m a -> m a
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
rightParent
if p acc'
then do
let !cur' = Int
rightParent Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1
if cur' .&. negate cur' /= cur'
then loopUp acc' cur'
else return Nothing
else return $ Just (acc, rightParent)
)
a
forall a. Monoid a => a
mempty
(Int
l Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
n)
case violationNode of
Maybe (a, Int)
Nothing -> Int -> m Int
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return Int
n
Just (!a
acc0, !Int
cur0) -> do
((a -> Int -> m Int) -> a -> Int -> m Int) -> a -> Int -> m Int
forall a. (a -> a) -> a
fix
( \a -> Int -> m Int
loopDown !a
acc !Int
cur -> do
if Int
cur Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
n
then do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st Int
cur
let !leftChild :: Int
leftChild = Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
cur
!acc' <- (a
acc a -> a -> a
forall a. Semigroup a => a -> a -> a
<>) (a -> a) -> m a -> m a
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
leftChild
if p acc'
then loopDown acc' (leftChild + 1)
else loopDown acc leftChild
else Int -> m Int
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return (Int -> m Int) -> Int -> m Int
forall a b. (a -> b) -> a -> b
$! Int
cur Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
n
)
a
acc0
Int
cur0
{-# INLINE upperBoundFrom #-}
lowerBoundTo ::
(AsSemigroupEndo f a, Monoid a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
(a -> Bool) ->
m Int
lowerBoundTo :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Monoid a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> (a -> Bool) -> m Int
lowerBoundTo SegTree (PrimState m) f a
st Int
r a -> Bool
p = do
let !n :: Int
n = SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st
Int -> (Int -> m ()) -> m ()
forall (m :: * -> *). Monad m => Int -> (Int -> m ()) -> m ()
rev1 (SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
heightSegTree SegTree (PrimState m) f a
st) ((Int -> m ()) -> m ()) -> (Int -> m ()) -> m ()
forall a b. (a -> b) -> a -> b
$ \Int
i -> do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st ((Int
r Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int
i)
violationNode <-
((a -> Int -> m (Maybe (a, Int)))
-> a -> Int -> m (Maybe (a, Int)))
-> a -> Int -> m (Maybe (a, Int))
forall a. (a -> a) -> a
fix
( \a -> Int -> m (Maybe (a, Int))
loopUp !a
acc !Int
cur -> do
let leftParent :: Int
leftParent =
case Int
cur Int -> Int -> Int
forall a. Bits a => a -> Int -> a
!>>. Int -> Int
forall b. FiniteBits b => b -> Int
countTrailingZeros (Int -> Int
forall a. Bits a => a -> a
complement Int
cur) of
Int
0 -> Int
1
Int
v -> Int
v
!acc' <- (a -> a -> a
forall a. Semigroup a => a -> a -> a
<> a
acc) (a -> a) -> m a -> m a
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
leftParent
if p acc'
then do
if leftParent .&. negate leftParent /= leftParent
then loopUp acc' (leftParent - 1)
else return Nothing
else return $ Just (acc, leftParent)
)
a
forall a. Monoid a => a
mempty
(Int
r Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1 Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
n)
case violationNode of
Maybe (a, Int)
Nothing -> Int -> m Int
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return Int
0
Just (!a
acc0, !Int
cur0) ->
((a -> Int -> m Int) -> a -> Int -> m Int) -> a -> Int -> m Int
forall a. (a -> a) -> a
fix
( \a -> Int -> m Int
loopDown !a
acc !Int
cur -> do
if Int
cur Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< Int
n
then do
SegTree (PrimState m) f a -> Int -> m ()
forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st Int
cur
let !rightChild :: Int
rightChild = Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
cur Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1
!acc' <- (a -> a -> a
forall a. Semigroup a => a -> a -> a
<> a
acc) (a -> a) -> m a -> m a
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
rightChild
if p acc'
then loopDown acc' (rightChild - 1)
else loopDown acc rightChild
else Int -> m Int
forall a. a -> m a
forall (m :: * -> *) a. Monad m => a -> m a
return (Int -> m Int) -> Int -> m Int
forall a b. (a -> b) -> a -> b
$! Int
cur Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1 Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
n
)
a
acc0
Int
cur0
{-# INLINE lowerBoundTo #-}
evalAt ::
(AsSemigroupEndo f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
f ->
m ()
evalAt :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> f -> m ()
evalAt SegTree (PrimState m) f a
st Int
k f
f = 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 (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) (f -> a -> a
forall f s. AsSemigroupEndo f s => f -> s -> s
sendo f
f) Int
k
Bool -> m () -> m ()
forall (f :: * -> *). Applicative f => Bool -> f () -> f ()
when (Int
k Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
< SegTree (PrimState m) f a -> Int
forall s f a. SegTree s f a -> Int
sizeSegTree SegTree (PrimState m) f a
st) (m () -> m ()) -> m () -> m ()
forall a b. (a -> b) -> a -> b
$ do
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 (SegTree (PrimState m) f a -> MVector (PrimState m) f
forall s f a. SegTree s f a -> MVector s f
getDualSegTree SegTree (PrimState m) f a
st) (f
f f -> f -> f
forall a. Semigroup a => a -> a -> a
<>) Int
k
{-# INLINE evalAt #-}
pushSegTree ::
(AsSemigroupEndo f a, U.Unbox f, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
m ()
pushSegTree :: forall f a (m :: * -> *).
(AsSemigroupEndo f a, Unbox f, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pushSegTree SegTree (PrimState m) f a
st 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 (SegTree (PrimState m) f a -> MVector (PrimState m) f
forall s f a. SegTree s f a -> MVector s f
getDualSegTree SegTree (PrimState m) f a
st) Int
k
UM.unsafeWrite (getDualSegTree st) k mempty
evalAt st (2 * k) fk
evalAt st (2 * k + 1) fk
{-# INLINE pushSegTree #-}
pullSegTree ::
(Semigroup a, U.Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a ->
Int ->
m ()
pullSegTree :: forall a (m :: * -> *) f.
(Semigroup a, Unbox a, PrimMonad m) =>
SegTree (PrimState m) f a -> Int -> m ()
pullSegTree SegTree (PrimState m) f a
st Int
k = do
a -> a -> a
forall a. Semigroup a => a -> a -> a
(<>)
(a -> a -> a) -> m a -> m (a -> a)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) (Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
k)
m (a -> a) -> m a -> m a
forall a b. m (a -> b) -> m a -> m b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> MVector (PrimState m) a -> Int -> m a
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> m a
UM.unsafeRead (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) (Int
2 Int -> Int -> Int
forall a. Num a => a -> a -> a
* Int
k Int -> Int -> Int
forall a. Num a => a -> a -> a
+ Int
1)
m a -> (a -> m ()) -> m ()
forall a b. m a -> (a -> m b) -> m b
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= MVector (PrimState m) a -> Int -> a -> m ()
forall (m :: * -> *) a.
(PrimMonad m, Unbox a) =>
MVector (PrimState m) a -> Int -> a -> m ()
UM.unsafeWrite (SegTree (PrimState m) f a -> MVector (PrimState m) a
forall s f a. SegTree s f a -> MVector s a
getSegTree SegTree (PrimState m) f a
st) Int
k
{-# INLINE pullSegTree #-}
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