{-# OPTIONS_GHC -fno-warn-incomplete-patterns #-} {-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE ScopedTypeVariables #-} module HaskellWorks.Data.RankSelect.CsPoppySpec (spec) where import Control.Monad import Control.Monad.IO.Class import Data.List (isSuffixOf) import Data.Maybe import Data.Word import GHC.Exts import HaskellWorks.Data.AtIndex import HaskellWorks.Data.Bits.BitLength import HaskellWorks.Data.Bits.BitRead import HaskellWorks.Data.Bits.BitShow import HaskellWorks.Data.Bits.PopCount.PopCount1 import HaskellWorks.Data.FromForeignRegion import HaskellWorks.Data.RankSelect.Base.Select1 import HaskellWorks.Data.RankSelect.BasicGen import HaskellWorks.Data.RankSelect.CsPoppy import HaskellWorks.Data.Take import HaskellWorks.Hspec.Hedgehog import Hedgehog import Prelude hiding (length, take, drop) import System.Directory import System.IO.MMap import System.IO.Unsafe import Test.Hspec import qualified Data.Vector.Storable as DVS import qualified HaskellWorks.Hedgehog.Gen as G import qualified Hedgehog.Gen as G import qualified Hedgehog.Range as R {-# ANN module ("HLint: ignore Redundant do" :: String) #-} {-# ANN module ("HLint: ignore Reduce duplication" :: String) #-} newtype ShowVector a = ShowVector a deriving (Eq, BitShow) corpusFiles :: [FilePath] corpusFiles = unsafePerformIO $ do entries <- listDirectory "data" let files = ("data/" ++) <$> (".ib" `isSuffixOf`) `filter` entries return files {-# NOINLINE corpusFiles #-} loadVector64 :: FilePath -> IO (DVS.Vector Word64) loadVector64 filename = fromForeignRegion <$> mmapFileForeignPtr filename ReadOnly Nothing instance BitShow a => Show (ShowVector a) where show = bitShow spec :: Spec spec = describe "HaskellWorks.Data.RankSelect.CsPoppySpec" $ do genRank1Select1Spec (undefined :: CsPoppy) describe "rank1 for Vector Word64 is equivalent to rank1 for CsPoppy" $ do it "on empty bitvector" $ require $ withTests 1 $ property $ do let v = DVS.empty Nice w <- forAll $ pure $ Nice $ makeCsPoppy v let i = 0 rank1 v i === rank1 w i it "on one basic block" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 1 8) (G.word64 R.constantBounded) i <- forAll $ G.word64 (R.linear 0 (length v * 8)) Nice w <- forAll $ pure $ Nice $ makeCsPoppy v rank1 v i === rank1 w i it "on two basic blocks" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 9 16) (G.word64 R.constantBounded) i <- forAll $ G.word64 (R.linear 0 (length v * 8)) Nice w <- forAll $ pure $ Nice $ makeCsPoppy v rank1 v i === rank1 w i it "on three basic blocks" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 17 24) (G.word64 R.constantBounded) i <- forAll $ G.word64 (R.linear 0 (length v * 8)) Nice w <- forAll $ pure $ Nice $ makeCsPoppy v rank1 v i === rank1 w i describe "select1 for Vector Word64 is equivalent to select1 for CsPoppy" $ do xit "on empty bitvector" $ require $ withTests 1 $ property $ do let v = DVS.empty Nice w <- forAll $ pure $ Nice $ makeCsPoppy v let i = 0 select1 w i === select1 v i xit "on one full zero basic block" $ require $ withTests 1 $ property $ do let v = fromList [0, 0, 0, 0, 0, 0, 0, 0] :: DVS.Vector Word64 Nice w <- forAll $ pure $ Nice $ makeCsPoppy v select1 w 0 === select1 v 0 xit "on one basic block" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 1 8) (G.word64 R.constantBounded) i <- forAll $ G.word64 (R.linear 0 (popCount1 v)) Nice w <- forAll $ pure $ Nice $ makeCsPoppy v select1 w i === select1 v i xit "on two basic blocks" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 9 16) (G.word64 R.constantBounded) i <- forAll $ G.word64 (R.linear 0 (popCount1 v)) Nice w <- forAll $ pure $ Nice $ makeCsPoppy v select1 w i === select1 v i xit "on three basic blocks" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 17 24) (G.word64 R.constantBounded) i <- forAll $ G.word64 (R.linear 0 (popCount1 v)) Nice w <- forAll $ pure $ Nice $ makeCsPoppy v select1 w i === select1 v i describe "Rank select over large buffer" $ do xit "Rank works" $ require $ property $ do let cs = fromJust (bitRead (take 4096 (cycle "10"))) :: DVS.Vector Word64 let ps = makeCsPoppy cs (rank1 ps `map` [1 .. 4096]) === [(x - 1) `div` 2 + 1 | x <- [1 .. 4096]] xit "Rank is consistent with pop count" $ require $ property $ do v <- forAll $ G.storableVector (R.linear 0 1024) (G.word64 R.constantBounded) Nice rsbs <- forAll $ pure $ Nice (makeCsPoppy v) rank1 rsbs (bitLength rsbs) === popCount1 rsbs it "Select works" $ do let v = DVS.fromList [0x1101001000100001, 0x2202002000200002, 0x3303003000300003] :: DVS.Vector Word64 let pc = popCount1 v require $ property $ do Nice csPoppy <- forAll $ pure $ Nice (makeCsPoppy v) s <- forAll $ G.word64 (R.linear 1 pc) select1 csPoppy s === select1 v s describe "Corpus" $ do describe "Shrunk Rank" $ do forM_ corpusFiles $ \corpusFile -> do xit corpusFile $ do fileV <- liftIO $ loadVector64 corpusFile require $ property $ do let v = DVS.take 768 fileV Nice csPoppy <- forAll $ pure $ Nice $ makeCsPoppy v maxPos <- forAll $ pure $ fromIntegral $ DVS.length (csPoppyBits csPoppy) r <- forAll $ G.word64 (R.linear 1 maxPos) rank1 csPoppy r === rank1 v r describe "Rank" $ do forM_ corpusFiles $ \corpusFile -> do it corpusFile $ do v <- liftIO $ loadVector64 corpusFile let csPoppy = makeCsPoppy v let maxPos = fromIntegral $ DVS.length (csPoppyBits csPoppy) require $ property $ do r <- forAll $ G.word64 (R.linear 1 maxPos) rank1 csPoppy r === rank1 v r describe "Select" $ do forM_ corpusFiles $ \corpusFile -> do it corpusFile $ do fileV <- liftIO $ loadVector64 corpusFile let csPoppy = makeCsPoppy fileV let pc = popCount1 (csPoppyBits csPoppy) require $ property $ do _ <- forAll $ pure $ Nice csPoppy s <- forAll $ G.word64 (R.linear 1 pc) select1 csPoppy s === select1 fileV s describe "Select straw man" $ do forM_ corpusFiles $ \corpusFile -> do it corpusFile $ do fileV <- liftIO $ loadVector64 corpusFile let csPoppy = makeCsPoppy fileV let pc = popCount1 (csPoppyBits csPoppy) require $ property $ do _ <- forAll $ pure $ csPoppyLayerS csPoppy s <- forAll $ G.word64 (R.linear 1 pc) select1 csPoppy s === select1 fileV s