diff options
| author | Sebastian Graf <sgraf1337@gmail.com> | 2019-07-30 17:55:31 +0000 | 
|---|---|---|
| committer | Marge Bot <ben+marge-bot@smart-cactus.org> | 2019-08-13 17:30:15 -0400 | 
| commit | 328a0efaba621b3bad6476dff1cea58bf7ab5166 (patch) | |
| tree | 79ba68532e477b715e7cea9faa12f893e3f82a67 /compiler/utils/UniqFM.hs | |
| parent | 672cbab268342d7a8829698e838e369e0a0b3a19 (diff) | |
| download | haskell-328a0efaba621b3bad6476dff1cea58bf7ab5166.tar.gz | |
Add Foldable, Traversable instances for Uniq(D)FM
The `UniqDFM` is deterministic, of course, while we provide an unsafe
`NonDetUniqFM` wrapper for `UniqFM` to opt into nondeterministic instances.
Diffstat (limited to 'compiler/utils/UniqFM.hs')
| -rw-r--r-- | compiler/utils/UniqFM.hs | 31 | 
1 files changed, 27 insertions, 4 deletions
| diff --git a/compiler/utils/UniqFM.hs b/compiler/utils/UniqFM.hs index 33d73cc60c..19b506e883 100644 --- a/compiler/utils/UniqFM.hs +++ b/compiler/utils/UniqFM.hs @@ -26,7 +26,8 @@ of arguments of combining function.  module UniqFM (          -- * Unique-keyed mappings -        UniqFM,       -- abstract type +        UniqFM,           -- abstract type +        NonDetUniqFM(..), -- wrapper for opting into nondeterminism          -- ** Manipulating those mappings          emptyUFM, @@ -84,9 +85,8 @@ import Data.Functor.Classes (Eq1 (..))  newtype UniqFM ele = UFM (M.IntMap ele)    deriving (Data, Eq, Functor) -  -- We used to derive Traversable and Foldable, but they were nondeterministic -  -- and not obvious at the call site. You can use explicit nonDetEltsUFM -  -- and fold a list if needed. +  -- Nondeterministic Foldable and Traversable instances are accessible through +  -- use of the 'NonDetUniqFM' wrapper.    -- See Note [Deterministic UniqFM] in UniqDFM to learn about determinism.  emptyUFM :: UniqFM elt @@ -333,6 +333,29 @@ nonDetFoldUFM_Directly k z (UFM m) = M.foldrWithKey (k . getUnique) z m  nonDetUFMToList :: UniqFM elt -> [(Unique, elt)]  nonDetUFMToList (UFM m) = map (\(k, v) -> (getUnique k, v)) $ M.toList m +-- | A wrapper around 'UniqFM' with the sole purpose of informing call sites +-- that the provided 'Foldable' and 'Traversable' instances are +-- nondeterministic. +-- If you use this please provide a justification why it doesn't introduce +-- nondeterminism. +-- See Note [Deterministic UniqFM] in UniqDFM to learn about determinism. +newtype NonDetUniqFM ele = NonDetUniqFM { getNonDet :: UniqFM ele } +  deriving (Functor) + +-- | Inherently nondeterministic. +-- If you use this please provide a justification why it doesn't introduce +-- nondeterminism. +-- See Note [Deterministic UniqFM] in UniqDFM to learn about determinism. +instance Foldable NonDetUniqFM where +  foldr f z (NonDetUniqFM (UFM m)) = foldr f z m + +-- | Inherently nondeterministic. +-- If you use this please provide a justification why it doesn't introduce +-- nondeterminism. +-- See Note [Deterministic UniqFM] in UniqDFM to learn about determinism. +instance Traversable NonDetUniqFM where +  traverse f (NonDetUniqFM (UFM m)) = NonDetUniqFM . UFM <$> traverse f m +  ufmToIntMap :: UniqFM elt -> M.IntMap elt  ufmToIntMap (UFM m) = m | 
