summaryrefslogtreecommitdiff
path: root/compiler
diff options
context:
space:
mode:
authorJohn Ericson <John.Ericson@Obsidian.Systems>2022-02-06 01:21:50 +0000
committerMarge Bot <ben+marge-bot@smart-cactus.org>2022-02-23 13:58:36 -0500
commitc7f32f768980b831d4969ec40fb7a4d19a51aff8 (patch)
tree667cda3dafbd51cdc22df16fc585606c33d1b0cb /compiler
parent59b7f764489d3eb765e0b40e916b1438ff76e1fa (diff)
downloadhaskell-c7f32f768980b831d4969ec40fb7a4d19a51aff8.tar.gz
Prepare rechecking logic for new type in a few ways
Combine `MustCompile and `NeedsCompile` into a single case. `CompileReason` is put inside to destinguish the two. This makes a number of things easier. `Semigroup RecompileRequired` is no longer used, to make sure we skip doing work where possible. `recompThen` is very similar, but helps remember. `checkList` is rewritten with `recompThen`.
Diffstat (limited to 'compiler')
-rw-r--r--compiler/GHC/Driver/Backpack.hs12
-rw-r--r--compiler/GHC/Driver/Main.hs79
-rw-r--r--compiler/GHC/Driver/Make.hs4
-rw-r--r--compiler/GHC/Driver/Pipeline.hs10
-rw-r--r--compiler/GHC/Iface/Recomp.hs141
5 files changed, 129 insertions, 117 deletions
diff --git a/compiler/GHC/Driver/Backpack.hs b/compiler/GHC/Driver/Backpack.hs
index b4e530a3e9..917fb6837e 100644
--- a/compiler/GHC/Driver/Backpack.hs
+++ b/compiler/GHC/Driver/Backpack.hs
@@ -570,20 +570,20 @@ mkBackpackMsg = do
in case node of
InstantiationNode _ _ ->
case recomp of
- MustCompile -> showMsg (text "Instantiating ") empty
UpToDate
| verbosity (hsc_dflags hsc_env) >= 2 -> showMsg (text "Skipping ") empty
| otherwise -> return ()
- RecompBecause reason -> showMsg (text "Instantiating ")
- (text " [" <> pprWithUnitState state (ppr reason) <> text "]")
+ NeedsRecompile reason0 -> showMsg (text "Instantiating ") $ case reason0 of
+ MustCompile -> empty
+ RecompBecause reason -> text " [" <> pprWithUnitState state (ppr reason) <> text "]"
ModuleNode _ _ ->
case recomp of
- MustCompile -> showMsg (text "Compiling ") empty
UpToDate
| verbosity (hsc_dflags hsc_env) >= 2 -> showMsg (text "Skipping ") empty
| otherwise -> return ()
- RecompBecause reason -> showMsg (text "Compiling ")
- (text " [" <> pprWithUnitState state (ppr reason) <> text "]")
+ NeedsRecompile reason0 -> showMsg (text "Compiling ") $ case reason0 of
+ MustCompile -> empty
+ RecompBecause reason -> text " [" <> pprWithUnitState state (ppr reason) <> text "]"
LinkNode _ _ -> showMsg (text "Linking ") empty
-- | 'PprStyle' for Backpack messages; here we usually want the module to
diff --git a/compiler/GHC/Driver/Main.hs b/compiler/GHC/Driver/Main.hs
index fc9b96f2e7..8fd8dad634 100644
--- a/compiler/GHC/Driver/Main.hs
+++ b/compiler/GHC/Driver/Main.hs
@@ -745,38 +745,43 @@ hscRecompStatus
Just hscMessage -> hscMessage hsc_env mod_index what (ModuleNode [] mod_summary)
Nothing -> return ()
- -- First check to see if the interface file agrees with the
- -- source file.
+ -- First check to see if the interface file agrees with the
+ -- source file.
+ --
+ -- Save the interface that comes back from checkOldIface.
+ -- In one-shot mode we don't have the old iface until this
+ -- point, when checkOldIface reads it from the disk.
(recomp_iface_reqd, mb_checked_iface)
<- {-# SCC "checkOldIface" #-}
liftIO $ checkOldIface hsc_env mod_summary mb_old_iface
- -- Check to see whether the expected build products already exist.
- -- If they don't exists then we trigger recompilation.
- let lcl_dflags = ms_hspp_opts mod_summary
- (recomp_obj_reqd, mb_linkable) <-
- case () of
- -- No need for a linkable, we're good to go
- _ | NoBackend <- backend lcl_dflags -> return (UpToDate, Nothing)
- -- Interpreter can use either already loaded bytecode or loaded object code
- | not (backendProducesObject (backend lcl_dflags)) -> do
- res <- liftIO $ checkByteCode old_linkable
- case res of
- (_, Just{}) -> return res
- _ -> liftIO $ checkObjects lcl_dflags old_linkable mod_summary
- -- Need object files for making object files
- | backendProducesObject (backend lcl_dflags) -> liftIO $ checkObjects lcl_dflags old_linkable mod_summary
- | otherwise -> pprPanic "hscRecompStatus" (text $ show $ backend lcl_dflags)
- let recomp_reqd = recomp_iface_reqd `mappend` recomp_obj_reqd
- -- save the interface that comes back from checkOldIface.
- -- In one-shot mode we don't have the old iface until this
- -- point, when checkOldIface reads it from the disk.
let mb_old_hash = fmap (mi_iface_hash . mi_final_exts) mb_checked_iface
- msg recomp_reqd
- case mb_checked_iface of
- Just iface | not (recompileRequired recomp_reqd) ->
- return $ HscUpToDate iface mb_linkable
- _ ->
- return $ HscRecompNeeded mb_old_hash
+ case recomp_iface_reqd of
+ NeedsRecompile _ -> do
+ msg recomp_iface_reqd
+ return $ HscRecompNeeded mb_old_hash
+ UpToDate -> do
+ -- Check to see whether the expected build products already exist.
+ -- If they don't exists then we trigger recompilation.
+ let lcl_dflags = ms_hspp_opts mod_summary
+ (recomp_obj_reqd, mb_linkable) <-
+ case () of
+ -- No need for a linkable, we're good to go
+ _ | NoBackend <- backend lcl_dflags -> return (UpToDate, Nothing)
+ -- Interpreter can use either already loaded bytecode or loaded object code
+ | not (backendProducesObject (backend lcl_dflags)) -> do
+ let res = checkByteCode old_linkable
+ case res of
+ (_, Just{}) -> return res
+ _ -> liftIO $ checkObjects lcl_dflags old_linkable mod_summary
+ -- Need object files for making object files
+ | backendProducesObject (backend lcl_dflags) -> liftIO $ checkObjects lcl_dflags old_linkable mod_summary
+ | otherwise -> pprPanic "hscRecompStatus" (text $ show $ backend lcl_dflags)
+ msg recomp_obj_reqd
+ case (mb_checked_iface, recomp_obj_reqd) of
+ (Just iface, UpToDate) ->
+ return $ HscUpToDate iface mb_linkable
+ _ ->
+ return $ HscRecompNeeded mb_old_hash
-- | Check that the .o files produced by compilation are already up-to-date
-- or not.
@@ -795,7 +800,7 @@ checkObjects dflags mb_old_linkable summary = do
checkDynamicObj k = if dt_enabled
then case (>=) <$> mb_dyn_obj_date <*> mb_if_date of
Just True -> k
- _ -> return (RecompBecause MissingDynObjectFile, Nothing)
+ _ -> return (needsRecompileBecause MissingDynObjectFile, Nothing)
-- Not in dynamic-too mode
else k
@@ -808,18 +813,18 @@ checkObjects dflags mb_old_linkable summary = do
| isObjectLinkable old_linkable, linkableTime old_linkable == obj_date
-> return $ (UpToDate, Just old_linkable)
_ -> (UpToDate,) . Just <$> findObjectLinkable this_mod obj_fn obj_date
- _ -> return (RecompBecause MissingObjectFile, Nothing)
+ _ -> return (needsRecompileBecause MissingObjectFile, Nothing)
-- | Check to see if we can reuse the old linkable, by this point we will
-- have just checked that the old interface matches up with the source hash, so
-- no need to check that again here
-checkByteCode :: Maybe Linkable -> IO (RecompileRequired, Maybe Linkable)
+checkByteCode :: Maybe Linkable -> (RecompileRequired, Maybe Linkable)
checkByteCode mb_old_linkable =
case mb_old_linkable of
Just old_linkable
| not (isObjectLinkable old_linkable)
- -> return $ (UpToDate, Just old_linkable)
- _ -> return $ (RecompBecause MissingBytecode, Nothing)
+ -> (UpToDate, Just old_linkable)
+ _ -> (needsRecompileBecause MissingBytecode, Nothing)
--------------------------------------------------------------
-- Compilers
@@ -1126,7 +1131,7 @@ oneShotMsg :: Logger -> RecompileRequired -> IO ()
oneShotMsg logger recomp =
case recomp of
UpToDate -> compilationProgressMsg logger $ text "compilation IS NOT required"
- _ -> return ()
+ NeedsRecompile _ -> return ()
batchMsg :: Messager
batchMsg = batchMsgWith (\_ _ _ _ -> empty)
@@ -1136,12 +1141,12 @@ batchMultiMsg = batchMsgWith (\_ _ _ node -> brackets (ppr (moduleGraphNodeUnitI
batchMsgWith :: (HscEnv -> (Int, Int) -> RecompileRequired -> ModuleGraphNode -> SDoc) -> Messager
batchMsgWith extra hsc_env_start mod_index recomp node =
case recomp of
- MustCompile -> showMsg (text herald) empty
UpToDate
| logVerbAtLeast logger 2 -> showMsg (text "Skipping") empty
| otherwise -> return ()
- RecompBecause reason -> showMsg (text herald)
- (text " [" <> pprWithUnitState state (ppr reason) <> text "]")
+ NeedsRecompile reason0 -> showMsg (text herald) $ case reason0 of
+ MustCompile -> empty
+ (RecompBecause reason) -> text " [" <> pprWithUnitState state (ppr reason) <> text "]"
where
herald = case node of
LinkNode {} -> "Linking"
diff --git a/compiler/GHC/Driver/Make.hs b/compiler/GHC/Driver/Make.hs
index 6023d3a914..fe1af07f93 100644
--- a/compiler/GHC/Driver/Make.hs
+++ b/compiler/GHC/Driver/Make.hs
@@ -82,7 +82,7 @@ import GHC.Parser.Header
import GHC.Iface.Load ( cannotFindModule )
import GHC.IfaceToCore ( typecheckIface )
-import GHC.Iface.Recomp ( RecompileRequired ( MustCompile ) )
+import GHC.Iface.Recomp ( RecompileRequired(..), CompileReason(..) )
import GHC.Data.Bag ( listToBag )
import GHC.Data.Graph.Directed
@@ -1142,7 +1142,7 @@ upsweep_inst :: HscEnv
-> IO ()
upsweep_inst hsc_env mHscMessage mod_index nmods uid iuid = do
case mHscMessage of
- Just hscMessage -> hscMessage hsc_env (mod_index, nmods) MustCompile (InstantiationNode uid iuid)
+ Just hscMessage -> hscMessage hsc_env (mod_index, nmods) (NeedsRecompile MustCompile) (InstantiationNode uid iuid)
Nothing -> return ()
runHsc hsc_env $ ioMsgMaybe $ hoistTcRnMessage $ tcRnCheckUnit hsc_env $ VirtUnit iuid
pure ()
diff --git a/compiler/GHC/Driver/Pipeline.hs b/compiler/GHC/Driver/Pipeline.hs
index ab1fb9f76f..6a5ebc74d9 100644
--- a/compiler/GHC/Driver/Pipeline.hs
+++ b/compiler/GHC/Driver/Pipeline.hs
@@ -480,7 +480,7 @@ linkingNeeded logger dflags unit_env staticLink linkables pkg_deps = do
exe_file = exeFileName platform staticLink (outputFile_ dflags)
e_exe_time <- tryIO $ getModificationUTCTime exe_file
case e_exe_time of
- Left _ -> return MustCompile
+ Left _ -> return $ NeedsRecompile MustCompile
Right t -> do
-- first check object files and extra_ld_inputs
let extra_ld_inputs = [ f | FileOption _ f <- ldInputs dflags ]
@@ -488,7 +488,7 @@ linkingNeeded logger dflags unit_env staticLink linkables pkg_deps = do
let (errs,extra_times) = partitionEithers e_extra_times
let obj_times = map linkableTime linkables ++ extra_times
if not (null errs) || any (t <) obj_times
- then return (RecompBecause ObjectsChanged)
+ then return $ needsRecompileBecause ObjectsChanged
else do
-- next, check libraries. XXX this only checks Haskell libraries,
@@ -498,16 +498,16 @@ linkingNeeded logger dflags unit_env staticLink linkables pkg_deps = do
lib <- unitHsLibs (ghcNameVersion dflags) (ways dflags) c ]
pkg_libfiles <- mapM (uncurry (findHSLib platform (ways dflags))) pkg_hslibs
- if any isNothing pkg_libfiles then return (RecompBecause LibraryChanged) else do
+ if any isNothing pkg_libfiles then return $ needsRecompileBecause LibraryChanged else do
e_lib_times <- mapM (tryIO . getModificationUTCTime)
(catMaybes pkg_libfiles)
let (lib_errs,lib_times) = partitionEithers e_lib_times
if not (null lib_errs) || any (t <) lib_times
- then return (RecompBecause LibraryChanged)
+ then return $ needsRecompileBecause LibraryChanged
else do
res <- checkLinkInfo logger dflags unit_env pkg_deps exe_file
if res
- then return (RecompBecause FlagsChanged)
+ then return $ needsRecompileBecause FlagsChanged
else return UpToDate
diff --git a/compiler/GHC/Iface/Recomp.hs b/compiler/GHC/Iface/Recomp.hs
index 2c1943074c..542901bad1 100644
--- a/compiler/GHC/Iface/Recomp.hs
+++ b/compiler/GHC/Iface/Recomp.hs
@@ -1,3 +1,4 @@
+{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE MultiWayIf #-}
{-# LANGUAGE TupleSections #-}
{-# LANGUAGE LambdaCase #-}
@@ -6,7 +7,10 @@
module GHC.Iface.Recomp
( checkOldIface
, RecompileRequired(..)
+ , needsRecompileBecause
+ , recompThen
, RecompReason (..)
+ , CompileReason(..)
, recompileRequired
, addFingerprints
)
@@ -111,17 +115,28 @@ Basic idea:
-}
data RecompileRequired
+ -- | everything is up to date, recompilation is not required
= UpToDate
- -- ^ everything is up to date, recompilation is not required
- | MustCompile
- -- ^ The .hs file has been modified, or the .o/.hi file does not exist
+ -- | Need to compile the module
+ | NeedsRecompile !CompileReason
+ deriving (Eq)
+
+needsRecompileBecause :: RecompReason -> RecompileRequired
+needsRecompileBecause = NeedsRecompile . RecompBecause
+
+data CompileReason
+ -- | The .hs file has been touched, or the .o/.hi file does not exist
+ = MustCompile
+ -- | The .o/.hi files are up to date, but something else has changed
+ -- to force recompilation; the String says what (one-line summary)
| RecompBecause !RecompReason
- -- ^ The .o/.hi files are up to date, but something else has changed
- -- to force recompilation; the String says what (one-line summary)
deriving (Eq)
instance Outputable RecompileRequired where
ppr UpToDate = text "UpToDate"
+ ppr (NeedsRecompile reason) = ppr reason
+
+instance Outputable CompileReason where
ppr MustCompile = text "MustCompile"
ppr (RecompBecause r) = text "RecompBecause" <+> ppr r
@@ -200,8 +215,15 @@ recompileRequired _ = True
recompThen :: Monad m => m RecompileRequired -> m RecompileRequired -> m RecompileRequired
recompThen ma mb = ma >>= \case
- UpToDate -> mb
- mc -> pure mc
+ UpToDate -> mb
+ rr@(NeedsRecompile _) -> pure rr
+
+checkList :: Monad m => [m RecompileRequired] -> m RecompileRequired
+checkList = \case
+ [] -> return UpToDate
+ (check : checks) -> check `recompThen` checkList checks
+
+----------------------
-- | Top level function to check if the version of an old interface file
-- is equivalent to the current source file the user asked us to compile.
@@ -262,10 +284,10 @@ check_old_iface hsc_env mod_summary maybe_iface
UpToDate -> do
maybe_dyn_iface <- liftIO $ loadIface (setDynamicNow dflags) (msDynHiFilePath mod_summary)
case maybe_dyn_iface of
- Nothing -> return (RecompBecause MissingDynHiFile, Nothing)
+ Nothing -> return (needsRecompileBecause MissingDynHiFile, Nothing)
Just dyn_iface | mi_iface_hash (mi_final_exts dyn_iface)
/= mi_iface_hash (mi_final_exts normal_iface)
- -> return (RecompBecause MismatchedDynHiFile, Nothing)
+ -> return (needsRecompileBecause MismatchedDynHiFile, Nothing)
Just {} -> return res
_ -> return res
check_dyn_hi _ recomp_check = recomp_check
@@ -283,19 +305,19 @@ check_old_iface hsc_env mod_summary maybe_iface
-- avoid reading an interface; just return the one we might
-- have been supplied with.
True | not (backendProducesObject $ backend dflags) ->
- return (MustCompile, maybe_iface)
+ return (NeedsRecompile MustCompile, maybe_iface)
-- Try and read the old interface for the current module
-- from the .hi file left from the last time we compiled it
True -> do
maybe_iface' <- liftIO $ getIface
- return (MustCompile, maybe_iface')
+ return (NeedsRecompile MustCompile, maybe_iface')
False -> do
maybe_iface' <- liftIO $ getIface
case maybe_iface' of
-- We can't retrieve the iface
- Nothing -> return (MustCompile, Nothing)
+ Nothing -> return (NeedsRecompile MustCompile, Nothing)
-- We have got the old iface; check its versions
-- even in the SourceUnmodifiedAndStable case we
@@ -329,9 +351,9 @@ checkVersions hsc_env mod_summary iface
-- test case bkpcabal04!
; hsc_env <- getTopEnv
; if mi_src_hash iface /= ms_hs_hash mod_summary
- then return (RecompBecause SourceFileChanged, Nothing) else do {
+ then return (needsRecompileBecause SourceFileChanged, Nothing) else do {
; if not (isHomeModule home_unit (mi_module iface))
- then return (RecompBecause ThisUnitIdChanged, Nothing) else do {
+ then return (needsRecompileBecause ThisUnitIdChanged, Nothing) else do {
; recomp <- liftIO $ checkFlagHash hsc_env iface
`recompThen` checkOptimHash hsc_env iface
`recompThen` checkHpcHash hsc_env iface
@@ -407,7 +429,7 @@ pluginRecompileToRecompileRequired old_fp new_fp pr
-- when we have an impure plugin in the stack we have to unconditionally
-- recompile since it might integrate all sorts of crazy IO results into
-- its compilation output.
- ForceRecompile -> RecompBecause ImpurePlugin
+ ForceRecompile -> needsRecompileBecause ImpurePlugin
| old_fp `elem` magic_fingerprints ||
new_fp `elem` magic_fingerprints
@@ -419,16 +441,16 @@ pluginRecompileToRecompileRequired old_fp new_fp pr
-- For example when we go from ForceRecomp to NoForceRecomp
-- recompilation is triggered since the old impure plugins could have
-- changed the build output which is now back to normal.
- = RecompBecause PluginsChanged
+ = needsRecompileBecause PluginsChanged
| otherwise =
case pr of
-- even though a plugin is forcing recompilation the fingerprint changed
-- which would cause recompilation anyways so we report the fingerprint
-- change instead.
- ForceRecompile -> RecompBecause PluginFingerprintChanged
+ ForceRecompile -> needsRecompileBecause PluginFingerprintChanged
- _ -> RecompBecause PluginFingerprintChanged
+ _ -> needsRecompileBecause PluginFingerprintChanged
where
magic_fingerprints =
@@ -446,7 +468,7 @@ checkHsig logger home_unit mod_summary iface = do
massert (isHomeModule home_unit outer_mod)
case inner_mod == mi_semantic_module iface of
True -> up_to_date logger (text "implementing module unchanged")
- False -> return (RecompBecause ModuleInstChanged)
+ False -> return $ needsRecompileBecause ModuleInstChanged
-- | Check if @.hie@ file is out of date or missing.
checkHie :: DynFlags -> ModSummary -> RecompileRequired
@@ -456,10 +478,10 @@ checkHie dflags mod_summary =
in if not (gopt Opt_WriteHie dflags)
then UpToDate
else case (hie_date_opt, hi_date) of
- (Nothing, _) -> RecompBecause HieMissing
+ (Nothing, _) -> needsRecompileBecause HieMissing
(Just hie_date, Just hi_date)
| hie_date < hi_date
- -> RecompBecause HieOutdated
+ -> needsRecompileBecause HieOutdated
_ -> UpToDate
-- | Check the flags haven't changed
@@ -519,7 +541,7 @@ checkMergedSignatures hsc_env mod_summary iface = do
Just r -> sort $ map (instModuleToModule unit_state) r
if old_merged == new_merged
then up_to_date logger (text "signatures to merge in unchanged" $$ ppr new_merged)
- else return (RecompBecause SigsMergeChanged)
+ else return $ needsRecompileBecause SigsMergeChanged
-- If the direct imports of this module are resolved to targets that
-- are not among the dependencies of the previous interface file,
@@ -537,21 +559,21 @@ checkDependencies hsc_env summary iface
res_normal <- classify_import (findImportedModule hsc_env) (ms_textual_imps summary ++ ms_srcimps summary)
res_plugin <- classify_import (\mod _ -> findPluginModule fc fopts units mhome_unit mod) (ms_plugin_imps summary)
case sequence (res_normal ++ res_plugin ++ [Right (fake_ghc_prim_import)| ms_ghc_prim_import summary]) of
- Left recomp -> return recomp
+ Left recomp -> return $ NeedsRecompile recomp
Right es -> do
let (hs, ps) = partitionEithers es
- res1 <- liftIO $ check_mods (sort hs) prev_dep_mods
-
- let allPkgDeps = sortBy (comparing snd) $ nubOrdOn snd (ps ++ implicit_deps ++ bkpk_units)
- res2 <- liftIO $ check_packages allPkgDeps prev_dep_pkgs
- return (res1 `mappend` res2)
+ liftIO $
+ check_mods (sort hs) prev_dep_mods
+ `recompThen`
+ let allPkgDeps = sortBy (comparing snd) $ nubOrdOn snd (ps ++ implicit_deps ++ bkpk_units)
+ in check_packages allPkgDeps prev_dep_pkgs
where
classify_import :: (ModuleName -> t -> IO FindResult)
-> [(t, GenLocated l ModuleName)]
-> IfG
[Either
- RecompileRequired (Either (UnitId, ModuleName) (String, UnitId))]
+ CompileReason (Either (UnitId, ModuleName) (String, UnitId))]
classify_import find_import imports =
liftIO $ traverse (\(mb_pkg, L _ mod) ->
let reason = ModuleChanged mod
@@ -594,7 +616,7 @@ checkDependencies hsc_env summary iface
text "module no longer" <+> quotes (ppr old) <+>
text "in dependencies"
- return (RecompBecause (ModuleRemoved old))
+ return $ needsRecompileBecause $ ModuleRemoved old
check_mods (new:news) olds
| Just (old, olds') <- uncons olds
, new == old = check_mods (dropWhile (== new) news) olds'
@@ -602,7 +624,7 @@ checkDependencies hsc_env summary iface
trace_hi_diffs logger $
text "imported module " <> quotes (ppr new) <>
text " not among previous dependencies"
- return (RecompBecause (ModuleAdded new))
+ return $ needsRecompileBecause $ ModuleAdded new
check_packages :: [(String, UnitId)] -> [UnitId] -> IO RecompileRequired
check_packages [] [] = return UpToDate
@@ -610,7 +632,7 @@ checkDependencies hsc_env summary iface
trace_hi_diffs logger $
text "package " <> quotes (ppr old) <>
text "no longer in dependencies"
- return (RecompBecause (UnitDepRemoved old))
+ return $ needsRecompileBecause $ UnitDepRemoved old
check_packages (new:news) olds
| Just (old, olds') <- uncons olds
, snd new == old = check_packages (dropWhile ((== (snd new)) . snd) news) olds'
@@ -618,24 +640,22 @@ checkDependencies hsc_env summary iface
trace_hi_diffs logger $
text "imported package " <> quotes (ppr new) <>
text " not among previous dependencies"
- return (RecompBecause (ModulePackageChanged (fst new)))
+ return $ needsRecompileBecause $ ModulePackageChanged $ fst new
needInterface :: Module -> (ModIface -> IO RecompileRequired)
-> IfG RecompileRequired
needInterface mod continue
= do
- mb_recomp <- getFromModIface
+ mb_recomp <- tryGetModIface
"need version info for"
mod
- continue
case mb_recomp of
- Nothing -> return MustCompile
- Just recomp -> return recomp
+ Nothing -> return $ NeedsRecompile MustCompile
+ Just iface -> liftIO $ continue iface
-getFromModIface :: String -> Module -> (ModIface -> IO a)
- -> IfG (Maybe a)
-getFromModIface doc_msg mod getter
+tryGetModIface :: String -> Module -> IfG (Maybe ModIface)
+tryGetModIface doc_msg mod
= do -- Load the imported interface if possible
logger <- getLogger
let doc_str = sep [text doc_msg, ppr mod]
@@ -653,7 +673,7 @@ getFromModIface doc_msg mod getter
-- old interface file. Don't complain: it might
-- just be that the current module doesn't need that
-- import and it's been deleted
- Succeeded iface -> Just <$> liftIO (getter iface)
+ Succeeded iface -> pure $ Just iface
-- | Given the usage information extracted from the old
-- M.hi file for the module being compiled, figure out
@@ -704,17 +724,15 @@ checkModUsage _ this_pkg UsageHomeModule{
recompile <- checkModuleFingerprint logger reason old_mod_hash new_mod_hash
if not (recompileRequired recompile)
then return UpToDate
- else
- -- CHECK EXPORT LIST
- checkMaybeHash logger reason maybe_old_export_hash new_export_hash
- (text " Export list changed") $ do
-
- -- CHECK ITEMS ONE BY ONE
- !recompile <- checkList [ checkEntityUsage logger reason new_decl_hash u
- | u <- old_decl_hash]
- if recompileRequired recompile
- then return recompile -- This one failed, so just bail out now
- else up_to_date logger (text " Great! The bits I use are up to date")
+ else checkList
+ [ -- CHECK EXPORT LIST
+ checkMaybeHash logger reason maybe_old_export_hash new_export_hash
+ (text " Export list changed")
+ , -- CHECK ITEMS ONE BY ONE
+ checkList [ checkEntityUsage logger reason new_decl_hash u
+ | u <- old_decl_hash]
+ , up_to_date logger (text " Great! The bits I use are up to date")
+ ]
checkModUsage fc _this_pkg UsageFile{ usg_file_path = file,
usg_file_hash = old_hash,
@@ -727,7 +745,7 @@ checkModUsage fc _this_pkg UsageFile{ usg_file_path = file,
else return UpToDate
where
reason = FileChanged file
- recomp = RecompBecause (fromMaybe reason (fmap CustomReason mlabel))
+ recomp = needsRecompileBecause $ fromMaybe reason $ fmap CustomReason mlabel
handler = if debugIsOn
then \e -> pprTrace "UsageFile" (text (show e)) $ return recomp
else \_ -> return recomp -- if we can't find the file, just recompile, don't fail
@@ -769,12 +787,11 @@ checkMaybeHash
-> Fingerprint
-> SDoc
-> IO RecompileRequired
- -> IO RecompileRequired
-checkMaybeHash logger reason maybe_old_hash new_hash doc continue
+checkMaybeHash logger reason maybe_old_hash new_hash doc
| Just hash <- maybe_old_hash, hash /= new_hash
= out_of_date_hash logger reason doc hash new_hash
| otherwise
- = continue
+ = return UpToDate
------------------------
checkEntityUsage :: Logger
@@ -798,22 +815,12 @@ up_to_date :: Logger -> SDoc -> IO RecompileRequired
up_to_date logger msg = trace_hi_diffs logger msg >> return UpToDate
out_of_date :: Logger -> RecompReason -> SDoc -> IO RecompileRequired
-out_of_date logger reason msg = trace_hi_diffs logger msg >> return (RecompBecause reason)
+out_of_date logger reason msg = trace_hi_diffs logger msg >> return (needsRecompileBecause reason)
out_of_date_hash :: Logger -> RecompReason -> SDoc -> Fingerprint -> Fingerprint -> IO RecompileRequired
out_of_date_hash logger reason msg old_hash new_hash
= out_of_date logger reason (hsep [msg, ppr old_hash, text "->", ppr new_hash])
-----------------------
-checkList :: Monad m => [m RecompileRequired] -> m RecompileRequired
--- This helper is used in two places
-checkList [] = return UpToDate
-checkList (check:checks) = do recompile <- check
- if recompileRequired recompile
- then return recompile
- else checkList checks
-
-
-- ---------------------------------------------------------------------------
-- Compute fingerprints for the interface