diff --git a/containers/src/Data/IntMap/Internal.hs b/containers/src/Data/IntMap/Internal.hs index 3b8fbb446..b973d8e1b 100644 --- a/containers/src/Data/IntMap/Internal.hs +++ b/containers/src/Data/IntMap/Internal.hs @@ -3078,10 +3078,64 @@ foldlFB = foldlWithKey -- > fromList [(5,"c"), (3,"b"), (5, "a")] == fromList [(5,"a"), (3,"b")] fromList :: [(Key,a)] -> IntMap a -fromList xs - = Foldable.foldl' ins empty xs - where - ins t (k,x) = insert k x t +fromList = insertAll Nil +-- GHC wants to inline this, because it's tiny, but that doesn't accomplish +-- anything because it expands to a recursive function. +{-# NOINLINE fromList #-} + +-- [Note: fromList] +-- +-- The obvious way to build a map from a list is just to fold over the list +-- inserting each entry into the accumulator map. The problem is that this +-- rebuilds the path from the root *every single time*. To avoid this, we +-- insert as many elements as we can into the current subtree, backing out +-- one level at a time when necessary. + +data Inserted a = Inserted !(IntMap a) ![(Key, a)] + +insertAll :: IntMap a -> [(Key, a)] -> IntMap a +insertAll m [] = m +insertAll m ((k,x) : kxs) + | Inserted m' r <- insertSome m k x kxs + = insertAll m' r + +-- | Insert at least one entry into an 'IntMap' or subtree. If +-- others fit in the same resulting subtree, insert them too. +-- Return the new map and remaining values. +insertSome :: IntMap a -> Key -> a -> [(Key, a)] -> Inserted a +insertSome t@(Bin p m l r) !k x kxs + | nomatch k p m + = insertMany (link k (Tip k x) p t) kxs + + | zero k m + , Inserted l' kxs' <- insertSome l k x kxs + = insertMany (Bin p m l' r) kxs' + + | Inserted r' kxs' <- insertSome r k x kxs + = insertMany (Bin p m l r') kxs' + +insertSome t@(Tip ky _) k x kxs + | k == ky + = insertMany (Tip k x) kxs + | otherwise + = insertMany (link k (Tip k x) ky t) kxs + +insertSome Nil k x kxs = insertMany (Tip k x) kxs + + +-- | Try to insert some entries into a subtree of an 'IntMap'. If +-- they belong in some other subtree, just don't insert them. +insertMany :: IntMap a -> [(Key, a)] -> Inserted a +insertMany t [] = Inserted t [] +insertMany t@(Bin p m _ _) kxs@((k, x) : kxs') + | nomatch k p m + = Inserted t kxs + | otherwise + = insertSome t k x kxs' +insertMany t@(Tip ky _) kxs@((k, x) : kxs') + | k==ky = insertSome t k x kxs' + | otherwise = Inserted t kxs +insertMany Nil kxs = Inserted Nil kxs -- Unused case -- | /O(n*min(n,W))/. Create a map from a list of key\/value pairs with a combining function. See also 'fromAscListWith'. -- @@ -3099,10 +3153,61 @@ fromListWith f xs -- > fromListWithKey f [] == empty fromListWithKey :: (Key -> a -> a -> a) -> [(Key,a)] -> IntMap a -fromListWithKey f xs - = Foldable.foldl' ins empty xs - where - ins t (k,x) = insertWithKey f k x t +-- See [Note: fromList] +fromListWithKey f = insertAllWithKey f Nil +-- GHC wants to inline this because it's tiny, but doing so is useless +-- because it inlines to a recursive function. +{-# NOINLINE fromListWithKey #-} + +insertAllWithKey + :: (Key -> a -> a -> a) + -> IntMap a -> [(Key, a)] -> IntMap a +insertAllWithKey _f m [] = m +insertAllWithKey f m ((k,x) : kxs) + | Inserted m' r <- insertSomeWithKey f m k x kxs + = insertAllWithKey f m' r + +-- | Insert at least one entry into an 'IntMap'. If others fit +-- inside, insert them too. Return the new map and remaining +-- values. +insertSomeWithKey + :: (Key -> a -> a -> a) + -> IntMap a -> Key -> a -> [(Key, a)] -> Inserted a +insertSomeWithKey f t@(Bin p m l r) !k x kxs + | nomatch k p m + = insertManyWithKey f (link k (Tip k x) p t) kxs + + | zero k m + , Inserted l' kxs' <- insertSomeWithKey f l k x kxs + = insertManyWithKey f (Bin p m l' r) kxs' + + | Inserted r' kxs' <- insertSomeWithKey f r k x kxs + = insertManyWithKey f (Bin p m l r') kxs' + +insertSomeWithKey f t@(Tip ky y) k !x kxs + | k == ky + , y' <- f k x y + = insertManyWithKey f (Tip k y') kxs + | otherwise + = insertManyWithKey f (link k (Tip k x) ky t) kxs + +insertSomeWithKey f Nil k x kxs = insertManyWithKey f (Tip k x) kxs + +-- | Try to insert some entries into an 'IntMap', but only if +-- they fit +insertManyWithKey + :: (Key -> a -> a -> a) + -> IntMap a -> [(Key, a)] -> Inserted a +insertManyWithKey _f t [] = Inserted t [] +insertManyWithKey f t@(Bin p m _ _) kxs@((k, x) : kxs') + | nomatch k p m + = Inserted t kxs + | otherwise + = insertSomeWithKey f t k x kxs' +insertManyWithKey f t@(Tip ky _) kxs@((k, x) : kxs') + | k==ky = insertSomeWithKey f t k x kxs' + | otherwise = Inserted t kxs +insertManyWithKey _f Nil kxs = Inserted Nil kxs -- | /O(n)/. Build a map from a list of key\/value pairs where -- the keys are in ascending order. diff --git a/containers/src/Data/IntMap/Strict/Internal.hs b/containers/src/Data/IntMap/Strict/Internal.hs index 9957a041f..ac83bf207 100644 --- a/containers/src/Data/IntMap/Strict/Internal.hs +++ b/containers/src/Data/IntMap/Strict/Internal.hs @@ -1,5 +1,6 @@ {-# LANGUAGE CPP #-} {-# LANGUAGE BangPatterns #-} +{-# LANGUAGE PatternGuards #-} #include "containers.h" @@ -1066,10 +1067,53 @@ fromSet f (IntSet.Tip kx bm) = buildTree f kx bm (IntSet.suffixBitMask + 1) -- > fromList [(5,"c"), (3,"b"), (5, "a")] == fromList [(5,"a"), (3,"b")] fromList :: [(Key,a)] -> IntMap a -fromList xs - = Foldable.foldl' ins empty xs - where - ins t (k,x) = insert k x t +fromList = insertAll Nil +{-# NOINLINE fromList #-} + +data Inserted a = Inserted !(IntMap a) ![(Key, a)] + +insertAll :: IntMap a -> [(Key, a)] -> IntMap a +insertAll m [] = m +insertAll m ((k,x) : kxs) + | Inserted m' r <- insertSome m k x kxs + = insertAll m' r + +-- | Insert at least one entry into an 'IntMap'. If others fit +-- inside, insert them too. Return the new map and remaining +-- values. +insertSome :: IntMap a -> Key -> a -> [(Key, a)] -> Inserted a +insertSome t@(Bin p m l r) !k x kxs + | nomatch k p m + = insertMany (link k (Tip k x) p t) kxs + + | zero k m + , Inserted l' kxs' <- insertSome l k x kxs + = insertMany (Bin p m l' r) kxs' + + | Inserted r' kxs' <- insertSome r k x kxs + = insertMany (Bin p m l r') kxs' + +insertSome t@(Tip ky _) k !x kxs + | k == ky + = insertMany (Tip k x) kxs + | otherwise + = insertMany (link k (Tip k x) ky t) kxs + +insertSome Nil k x kxs = insertMany (Tip k x) kxs + +-- | Try to insert some entries into an 'IntMap', but only if +-- they fit +insertMany :: IntMap a -> [(Key, a)] -> Inserted a +insertMany t [] = Inserted t [] +insertMany t@(Bin p m _ _) kxs@((k, x) : kxs') + | nomatch k p m + = Inserted t kxs + | otherwise + = insertSome t k x kxs' +insertMany t@(Tip ky _) kxs@((k, x) : kxs') + | k==ky = insertSome t k x kxs' + | otherwise = Inserted t kxs +insertMany Nil kxs = Inserted Nil kxs -- | /O(n*min(n,W))/. Create a map from a list of key\/value pairs with a combining function. See also 'fromAscListWith'. -- @@ -1086,10 +1130,59 @@ fromListWith f xs -- > fromListWith (++) [] == empty fromListWithKey :: (Key -> a -> a -> a) -> [(Key,a)] -> IntMap a -fromListWithKey f xs - = Foldable.foldl' ins empty xs - where - ins t (k,x) = insertWithKey f k x t +fromListWithKey f = insertAllWithKey f Nil +{-# NOINLINE fromListWithKey #-} + +insertAllWithKey + :: (Key -> a -> a -> a) + -> IntMap a -> [(Key, a)] -> IntMap a +insertAllWithKey _f m [] = m +insertAllWithKey f m ((k,x) : kxs) + | Inserted m' r <- insertSomeWithKey f m k x kxs + = insertAllWithKey f m' r + +-- | Insert at least one entry into an 'IntMap'. If others fit +-- inside, insert them too. Return the new map and remaining +-- values. +insertSomeWithKey + :: (Key -> a -> a -> a) + -> IntMap a -> Key -> a -> [(Key, a)] -> Inserted a +insertSomeWithKey f t@(Bin p m l r) !k x kxs + | nomatch k p m + = insertManyWithKey f (link k (Tip k x) p t) kxs + + | zero k m + , Inserted l' kxs' <- insertSomeWithKey f l k x kxs + = insertManyWithKey f (Bin p m l' r) kxs' + + | Inserted r' kxs' <- insertSomeWithKey f r k x kxs + = insertManyWithKey f (Bin p m l r') kxs' + +insertSomeWithKey f t@(Tip ky y) k x kxs + | k == ky + , !y' <- f k x y + = insertManyWithKey f (Tip k y') kxs + | otherwise + = x `seq` insertManyWithKey f (link k (Tip k x) ky t) kxs + +insertSomeWithKey f Nil k x kxs = x `seq` insertManyWithKey f (Tip k x) kxs + +-- | Try to insert some entries into an 'IntMap', but only if +-- they fit +insertManyWithKey + :: (Key -> a -> a -> a) + -> IntMap a -> [(Key, a)] -> Inserted a +insertManyWithKey _f t [] = Inserted t [] +insertManyWithKey f t@(Bin p m _ _) kxs@((k, x) : kxs') + | nomatch k p m + = Inserted t kxs + | otherwise + = insertSomeWithKey f t k x kxs' +insertManyWithKey f t@(Tip ky _) kxs@((k, x) : kxs') + | k==ky + = insertSomeWithKey f t k x kxs' + | otherwise = Inserted t kxs +insertManyWithKey _f Nil kxs = Inserted Nil kxs -- | /O(n)/. Build a map from a list of key\/value pairs where -- the keys are in ascending order. diff --git a/containers/src/Data/IntSet/Internal.hs b/containers/src/Data/IntSet/Internal.hs index 3bc157ba1..53c0642d0 100644 --- a/containers/src/Data/IntSet/Internal.hs +++ b/containers/src/Data/IntSet/Internal.hs @@ -1,5 +1,6 @@ {-# LANGUAGE CPP #-} {-# LANGUAGE BangPatterns #-} +{-# LANGUAGE PatternGuards #-} #if __GLASGOW_HASKELL__ {-# LANGUAGE MagicHash, DeriveDataTypeable, StandaloneDeriving #-} #endif @@ -1052,10 +1053,56 @@ foldlFB = foldl -- | /O(n*min(n,W))/. Create a set from a list of integers. fromList :: [Key] -> IntSet -fromList xs - = Foldable.foldl' ins empty xs - where - ins t x = insert x t +-- See [Note: fromList] in Data.IntMap.Internal +fromList = insertAll Nil +{-# NOINLINE fromList #-} + +data Inserted = Inserted !IntSet ![Key] + +insertAll :: IntSet -> [Key] -> IntSet +insertAll m [] = m +insertAll m (k : kxs) + | Inserted m' r <- insertSome m (prefixOf k) (bitmapOf k) kxs + = insertAll m' r + +-- | Insert at least one entry into an 'IntSet' or subtree. If +-- others fit in the same resulting subtree, insert them too. +-- Return the new set and remaining values. +insertSome :: IntSet -> Prefix -> BitMap -> [Key] -> Inserted +insertSome t@(Bin p m l r) !k !x kxs + | nomatch k p m + = insertMany (link k (Tip k x) p t) kxs + + | zero k m + , Inserted l' kxs' <- insertSome l k x kxs + = insertMany (Bin p m l' r) kxs' + + | Inserted r' kxs' <- insertSome r k x kxs + = insertMany (Bin p m l r') kxs' + +insertSome t@(Tip ky y) k x kxs + | k == ky + = insertMany (Tip k (x .|. y)) kxs + | otherwise + = insertMany (link k (Tip k x) ky t) kxs + +insertSome Nil k x kxs = insertMany (Tip k x) kxs + +-- | Try to insert some entries into a subtree of an 'IntMap'. If +-- they belong in some other subtree, just don't insert them. +insertMany :: IntSet -> [Key] -> Inserted +insertMany t [] = Inserted t [] +insertMany t@(Bin p m _ _) kxs@(kx : kxs') + | nomatch (prefixOf kx) p m + = Inserted t kxs + | otherwise + = insertSome t (prefixOf kx) (bitmapOf kx) kxs' +insertMany t@(Tip ky _) kxs@(kx : kxs') + | prefixOf kx==ky + = insertSome t (prefixOf kx) (bitmapOf kx) kxs' + | otherwise + = Inserted t kxs +insertMany Nil kxs = Inserted Nil kxs -- Unused case -- | /O(n)/. Build a set from an ascending list of elements. -- /The precondition (input list is ascending) is not checked./