forked from ucsd-progsys/liquidhaskell
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Plugin.hs
607 lines (509 loc) · 27.1 KB
/
Plugin.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
-- | This module provides a GHC 'Plugin' that allows LiquidHaskell to be hooked directly into GHC's
-- compilation pipeline, facilitating its usage and adoption.
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE ViewPatterns #-}
module Language.Haskell.Liquid.GHC.Plugin (
plugin
) where
import qualified Liquid.GHC.API as O
import Liquid.GHC.API as GHC hiding (Target, Type)
import qualified Text.PrettyPrint.HughesPJ as PJ
import qualified Language.Fixpoint.Types as F
import qualified Liquid.GHC.Misc as LH
import qualified Language.Haskell.Liquid.UX.CmdLine as LH
import qualified Liquid.GHC.Interface as LH
import qualified Language.Haskell.Liquid.Liquid as LH
import qualified Language.Haskell.Liquid.Types.PrettyPrint as LH ( filterReportErrors
, filterReportErrorsWith
, defaultFilterReporter
, reduceFilters )
import qualified Liquid.GHC.Logging as LH (fromPJDoc)
import Language.Haskell.Liquid.GHC.Plugin.Types
import Language.Haskell.Liquid.GHC.Plugin.Util as Util
import Language.Haskell.Liquid.GHC.Plugin.SpecFinder
as SpecFinder
import Liquid.GHC.Types (MGIModGuts(..), miModGuts)
import GHC.LanguageExtensions
import Control.Monad
import qualified Control.Monad.Catch as Ex
import Data.Coerce
import Data.Kind ( Type )
import Data.List as L
hiding ( intersperse )
import Data.IORef
import qualified Data.Set as S
import Data.Set ( Set )
import qualified Data.HashSet as HS
import qualified Data.HashMap.Strict as HM
import System.IO.Unsafe ( unsafePerformIO )
import Language.Fixpoint.Types hiding ( errs
, panic
, Error
, Result
, Expr
)
import qualified Language.Haskell.Liquid.Measure as Ms
import Language.Haskell.Liquid.Parse
import Language.Haskell.Liquid.Transforms.ANF
import Language.Haskell.Liquid.Types hiding ( getConfig )
import Language.Haskell.Liquid.Bare
import Language.Haskell.Liquid.UX.CmdLine
import Optics
-- | Represents an abnormal but non-fatal state of the plugin. Because it is not
-- meant to escape the plugin, it is not thrown in IO but instead carried around
-- in an `Either`'s `Left` case and handled at the top level of the plugin
-- function.
newtype LiquidCheckException = ErrorsOccurred [Filter] -- Unmatched expected errors
deriving (Eq, Ord, Show)
---------------------------------------------------------------------------------
-- | State and configuration management -----------------------------------------
---------------------------------------------------------------------------------
-- | A reference to cache the LH's 'Config' and produce it only /once/, during the dynFlags hook.
cfgRef :: IORef Config
cfgRef = unsafePerformIO $ newIORef defConfig
{-# NOINLINE cfgRef #-}
-- | Set to 'True' to enable debug logging.
debugLogs :: Bool
debugLogs = False
---------------------------------------------------------------------------------
-- | Useful functions -----------------------------------------------------------
---------------------------------------------------------------------------------
-- | Reads the 'Config' out of a 'IORef'.
getConfig :: IO Config
getConfig = readIORef cfgRef
-- | Combinator which conditionally print on the screen based on the value of 'debugLogs'.
debugLog :: MonadIO m => String -> m ()
debugLog msg = when debugLogs $ liftIO (putStrLn msg)
---------------------------------------------------------------------------------
-- | The Plugin entrypoint ------------------------------------------------------
---------------------------------------------------------------------------------
plugin :: GHC.Plugin
plugin = GHC.defaultPlugin {
typeCheckResultAction = liquidPlugin
, driverPlugin = customDynFlags
, pluginRecompile = purePlugin
}
where
liquidPlugin :: [CommandLineOption] -> ModSummary -> TcGblEnv -> TcM TcGblEnv
liquidPlugin _ summary gblEnv = do
cfg <- liftIO getConfig
if skipModule cfg then return gblEnv
else liquidPluginGo summary gblEnv
-- Unfortunately, we can't make Haddock run the LH plugin, because the former
-- does mangle the '.hi' files, causing annotations to not be persisted in the
-- 'ExternalPackageState' and/or 'HomePackageTable'. For this reason we disable
-- the plugin altogether if the module is being compiled with Haddock.
-- See also: https://github.com/ucsd-progsys/liquidhaskell/issues/1727
-- for a post-mortem.
liquidPluginGo summary gblEnv = do
logger <- getLogger
dynFlags <- getDynFlags
withTiming logger dynFlags (text "LiquidHaskell" <+> brackets (ppr $ ms_mod_name summary)) (const ()) $ do
if gopt Opt_Haddock dynFlags
then do
-- Warn the user
let msg = PJ.vcat [ PJ.text "LH can't be run with Haddock."
, PJ.nest 4 $ PJ.text "Documentation will still be created."
]
let srcLoc = mkSrcLoc (mkFastString $ ms_hspp_file summary) 1 1
let warning = mkWarning (mkSrcSpan srcLoc srcLoc) msg
liftIO $ printWarning logger dynFlags warning
pure gblEnv
else do
newGblEnv <- typecheckHook summary gblEnv
case newGblEnv of
-- Exit with success if all expected errors were found
Left (ErrorsOccurred []) -> pure gblEnv
-- Exit with error if there were unmatched expected errors
Left (ErrorsOccurred errorFilters) -> do
defaultFilterReporter (LH.modSummaryHsFile summary) errorFilters
failM
Right newGblEnv' ->
pure newGblEnv'
--------------------------------------------------------------------------------
-- | GHC Configuration & Setup -------------------------------------------------
--------------------------------------------------------------------------------
-- | Overrides the default 'DynFlags' options. Specifically, we need the GHC
-- lexer not to throw away block comments, as this is where the LH spec comments
-- would live. This is why we set the 'Opt_KeepRawTokenStream' option.
customDynFlags :: [CommandLineOption] -> HscEnv -> IO HscEnv
customDynFlags opts hscEnv = do
cfg <- liftIO $ LH.getOpts opts
writeIORef cfgRef cfg
return (hscEnv { hsc_dflags = configureDynFlags (hsc_dflags hscEnv) })
where
configureDynFlags :: DynFlags -> DynFlags
configureDynFlags df =
df `gopt_set` Opt_ImplicitImportQualified
`gopt_set` Opt_PIC
`gopt_set` Opt_DeferTypedHoles
`gopt_set` Opt_KeepRawTokenStream
`xopt_set` MagicHash
`xopt_set` DeriveGeneric
`xopt_set` StandaloneDeriving
--------------------------------------------------------------------------------
-- | \"Unoptimising\" things ----------------------------------------------------
--------------------------------------------------------------------------------
-- | LiquidHaskell requires the unoptimised core binds in order to work correctly, but at the same time the
-- user can invoke GHC with /any/ optimisation flag turned out. This is why we grab the core binds by
-- desugaring the module during /parsing/ (before that's already too late) and we cache the core binds for
-- the rest of the program execution.
class Unoptimise a where
type UnoptimisedTarget a :: Type
unoptimise :: a -> UnoptimisedTarget a
instance Unoptimise DynFlags where
type UnoptimisedTarget DynFlags = DynFlags
unoptimise df = updOptLevel 0 df
{ debugLevel = 1
, ghcLink = LinkInMemory
, backend = Interpreter
, ghcMode = CompManager
}
instance Unoptimise ModSummary where
type UnoptimisedTarget ModSummary = ModSummary
unoptimise modSummary = modSummary { ms_hspp_opts = unoptimise (ms_hspp_opts modSummary) }
instance Unoptimise (DynFlags, HscEnv) where
type UnoptimisedTarget (DynFlags, HscEnv) = HscEnv
unoptimise (unoptimise -> df, env) = env { hsc_dflags = df }
--------------------------------------------------------------------------------
-- | Typechecking phase --------------------------------------------------------
--------------------------------------------------------------------------------
-- | We hook at this stage of the pipeline in order to call \"liquidhaskell\". This
-- might seems counterintuitive as LH works on a desugared module. However, there
-- are a bunch of reasons why we do this:
--
-- 1. Tools like \"ghcide\" works by running the compilation pipeline up until
-- this stage, which means that we won't be able to report errors and warnings
-- if we call /LH/ any later than here;
--
-- 2. Although /LH/ works on \"Core\", it requires the _unoptimised_ \"Core\" that we
-- grab from parsing (again) the module by using the GHC API, so we are really
-- independent from the \"normal\" compilation pipeline.
--
typecheckHook :: ModSummary -> TcGblEnv -> TcM (Either LiquidCheckException TcGblEnv)
typecheckHook (unoptimise -> modSummary) tcGblEnv = do
debugLog $ "We are in module: " <> show (toStableModule thisModule)
env <- env_top <$> getEnv
parsed <- liftIO $ parseModuleIO env (LH.keepRawTokenStream modSummary)
let comments = LH.extractSpecComments parsed
-- The LH plugin itself calls the type checker (see following line). This
-- would lead to a loop if we didn't remove the plugin when calling the type
-- checker.
typechecked <- liftIO $ typecheckModuleIO (dropPlugins env) (LH.ignoreInline parsed)
resolvedNames <- liftIO $ LH.lookupTyThings env modSummary tcGblEnv
availTyCons <- liftIO $ LH.availableTyCons env modSummary tcGblEnv (tcg_exports tcGblEnv)
availVars <- liftIO $ LH.availableVars env modSummary tcGblEnv (tcg_exports tcGblEnv)
unoptimisedGuts <- liftIO $ desugarModuleIO env modSummary typechecked
let tcData = mkTcData (tcg_rn_imports tcGblEnv) resolvedNames availTyCons availVars
let pipelineData = PipelineData unoptimisedGuts tcData (map mkSpecComment comments)
liquidHaskellCheck pipelineData modSummary tcGblEnv
where
thisModule :: Module
thisModule = tcg_mod tcGblEnv
dropPlugins hsc_env = hsc_env { hsc_plugins = [], hsc_static_plugins = [] }
serialiseSpec :: Module -> TcGblEnv -> LiquidLib -> TcM TcGblEnv
serialiseSpec thisModule tcGblEnv liquidLib = do
-- ---
-- -- CAN WE 'IGNORE' THE BELOW? TODO:IGNORE -- issue use `emptyLiquidLib` instead of pmrClientLib
-- ProcessModuleResult{..} <- processModule lhContext
-- liftIO $ putStrLn "liquidHaskellCheck 7"
-- -- Call into the existing Liquid interface
-- out <- liftIO $ LH.checkTargetInfo pmrTargetInfo
-- liftIO $ putStrLn "liquidHaskellCheck 8"
-- -- Report the outcome of the checking
-- LH.reportResult errorLogger cfg [giTarget (giSrc pmrTargetInfo)] out
-- case o_result out of
-- Safe _stats -> pure ()
-- _ -> failM
-- liftIO $ putStrLn "liquidHaskellCheck 9"
-- ---
let serialisedSpec = Util.serialiseLiquidLib liquidLib thisModule
debugLog $ "Serialised annotation ==> " ++ (O.showSDocUnsafe . O.ppr $ serialisedSpec)
-- liftIO $ putStrLn "liquidHaskellCheck 10"
pure $ tcGblEnv { tcg_anns = serialisedSpec : tcg_anns tcGblEnv }
processInputSpec :: Config -> PipelineData -> ModSummary -> TcGblEnv -> BareSpec -> TcM (Either LiquidCheckException TcGblEnv)
processInputSpec cfg pipelineData modSummary tcGblEnv inputSpec = do
debugLog $ " Input spec: \n" ++ show inputSpec
debugLog $ "Relevant ===> \n" ++ unlines (renderModule <$> S.toList (relevantModules modGuts))
logicMap :: LogicMap <- liftIO LH.makeLogicMap
-- debugLog $ "Logic map:\n" ++ show logicMap
let lhContext = LiquidHaskellContext {
lhGlobalCfg = cfg
, lhInputSpec = inputSpec
, lhModuleLogicMap = logicMap
, lhModuleSummary = modSummary
, lhModuleTcData = pdTcData pipelineData
, lhModuleGuts = pdUnoptimisedCore pipelineData
, lhRelevantModules = relevantModules modGuts
}
-- liftIO $ putStrLn ("liquidHaskellCheck 6: " ++ show isIg)
if isIgnore inputSpec
then pure $ Left (ErrorsOccurred [])
else do
liquidLib' <- checkLiquidHaskellContext lhContext
traverse (serialiseSpec thisModule tcGblEnv) liquidLib'
where
thisModule :: Module
thisModule = tcg_mod tcGblEnv
modGuts :: ModGuts
modGuts = pdUnoptimisedCore pipelineData
liquidHaskellCheckWithConfig :: Config -> PipelineData -> ModSummary -> TcGblEnv -> TcM (Either LiquidCheckException TcGblEnv)
liquidHaskellCheckWithConfig globalCfg pipelineData modSummary tcGblEnv = do
-- The 'specQuotes' contain stuff we need from imported modules, extracted
-- from the annotations in their interface files.
let specQuotes :: [BPspec]
specQuotes = LH.extractSpecQuotes' tcg_mod tcg_anns tcGblEnv
-- Here, we are calling Liquid Haskell's parser, acting on the unparsed
-- spec comments stored in the pipeline data, supported by the specQuotes
-- obtained from the imported modules.
inputSpec' :: Either LiquidCheckException BareSpec <-
getLiquidSpec thisFile thisModule (pdSpecComments pipelineData) specQuotes
case inputSpec' of
Left e -> pure $ Left e
Right inputSpec ->
withPragmas globalCfg thisFile (Ms.pragmas $ review bareSpecIso inputSpec) $ \moduleCfg -> do
processInputSpec moduleCfg pipelineData modSummary tcGblEnv inputSpec
`Ex.catch` (\(e :: UserError) -> reportErrs moduleCfg [e])
`Ex.catch` (\(e :: Error) -> reportErrs moduleCfg [e])
`Ex.catch` (\(es :: [Error]) -> reportErrs moduleCfg es)
where
thisFile :: FilePath
thisFile = LH.modSummaryHsFile modSummary
continue :: TcM (Either LiquidCheckException TcGblEnv)
continue = pure $ Left (ErrorsOccurred [])
reportErrs :: (Show e, F.PPrint e) => Config -> [TError e] -> TcM (Either LiquidCheckException TcGblEnv)
reportErrs cfg = LH.filterReportErrors thisFile GHC.failM continue (getFilters cfg) Full
thisModule :: Module
thisModule = tcg_mod tcGblEnv
-- | Partially calls into LiquidHaskell's GHC API.
liquidHaskellCheck :: PipelineData -> ModSummary -> TcGblEnv -> TcM (Either LiquidCheckException TcGblEnv)
liquidHaskellCheck pipelineData modSummary tcGblEnv = do
cfg <- liftIO getConfig
liquidHaskellCheckWithConfig cfg pipelineData modSummary tcGblEnv
checkLiquidHaskellContext :: LiquidHaskellContext -> TcM (Either LiquidCheckException LiquidLib)
checkLiquidHaskellContext lhContext = do
pmr <- processModule lhContext
case pmr of
Left e -> pure $ Left e
Right ProcessModuleResult{..} -> do
-- Call into the existing Liquid interface
out <- liftIO $ LH.checkTargetInfo pmrTargetInfo
let bareSpec = lhInputSpec lhContext
file = LH.modSummaryHsFile $ lhModuleSummary lhContext
withPragmas (lhGlobalCfg lhContext) file (Ms.pragmas $ review bareSpecIso bareSpec) $ \moduleCfg -> do
let filters = getFilters moduleCfg
-- Report the outcome of the checking
LH.reportResult (errorLogger file filters) moduleCfg [giTarget (giSrc pmrTargetInfo)] out
-- If there are unmatched filters or errors, and we are not reporting with
-- json, we don't make it to this part of the code because errorLogger
-- will throw an exception.
--
-- F.Crash is also handled by reportResult and errorLogger
case o_result out of
F.Safe _ -> return $ Right pmrClientLib
_ | json moduleCfg -> failM
| otherwise -> return $ Left $ ErrorsOccurred []
errorLogger :: FilePath -> [Filter] -> OutputResult -> TcM ()
errorLogger file filters outputResult = do
LH.filterReportErrorsWith
FilterReportErrorsArgs { msgReporter = GHC.reportErrors
, filterReporter = LH.defaultFilterReporter file
, failure = GHC.failM
, continue = pure ()
, pprinter = \(spn, e) -> mkLongErrAt spn (LH.fromPJDoc e) O.empty
, matchingFilters = LH.reduceFilters (\(src, doc) -> PJ.render doc ++ " at " ++ LH.showPpr src) filters
, filters = filters
}
(LH.orMessages outputResult)
isIgnore :: BareSpec -> Bool
isIgnore (MkBareSpec sp) = any ((== "--skip-module") . F.val) (pragmas sp)
--------------------------------------------------------------------------------
-- | Working with bare & lifted specs ------------------------------------------
--------------------------------------------------------------------------------
loadDependencies :: Config
-- ^ The 'Config' associated to the /current/ module being compiled.
-> Module
-> [Module]
-> TcM TargetDependencies
loadDependencies currentModuleConfig thisModule mods = do
hscEnv <- env_top <$> getEnv
results <- SpecFinder.findRelevantSpecs
(excludeAutomaticAssumptionsFor currentModuleConfig) hscEnv mods
deps <- foldlM processResult mempty (reverse results)
redundant <- liftIO $ configToRedundantDependencies hscEnv currentModuleConfig
debugLog $ "Redundant dependencies ==> " ++ show redundant
pure $ foldl' (flip dropDependency) deps redundant
where
processResult :: TargetDependencies -> SpecFinderResult -> TcM TargetDependencies
processResult !acc (SpecNotFound mdl) = do
debugLog $ "[T:" ++ renderModule thisModule
++ "] Spec not found for " ++ renderModule mdl
pure acc
processResult _ (SpecFound originalModule location _) = do
dynFlags <- getDynFlags
debugLog $ "[T:" ++ show (moduleName thisModule)
++ "] Spec found for " ++ renderModule originalModule ++ ", at location " ++ show location
Util.pluginAbort (O.showSDoc dynFlags $ O.text "A BareSpec was returned as a dependency, this is not allowed, in " O.<+> O.ppr thisModule)
processResult !acc (LibFound originalModule location lib) = do
debugLog $ "[T:" ++ show (moduleName thisModule)
++ "] Lib found for " ++ renderModule originalModule ++ ", at location " ++ show location
pure $ TargetDependencies {
getDependencies = HM.insert (toStableModule originalModule) (libTarget lib) (getDependencies $ acc <> libDeps lib)
}
data LiquidHaskellContext = LiquidHaskellContext {
lhGlobalCfg :: Config
, lhInputSpec :: BareSpec
, lhModuleLogicMap :: LogicMap
, lhModuleSummary :: ModSummary
, lhModuleTcData :: TcData
, lhModuleGuts :: ModGuts
, lhRelevantModules :: Set Module
}
--------------------------------------------------------------------------------
-- | Per-Module Pipeline -------------------------------------------------------
--------------------------------------------------------------------------------
data ProcessModuleResult = ProcessModuleResult {
pmrClientLib :: LiquidLib
-- ^ The \"client library\" we will serialise on disk into an interface's 'Annotation'.
, pmrTargetInfo :: TargetInfo
-- ^ The 'GhcInfo' for the current 'Module' that LiquidHaskell will process.
}
-- | Parse the spec comments from one module, supported by the
-- spec quotes from the imported module. Also looks for
-- "companion specs" for the current module and merges them in
-- if it finds one.
getLiquidSpec :: FilePath -> Module -> [SpecComment] -> [BPspec] -> TcM (Either LiquidCheckException BareSpec)
getLiquidSpec thisFile thisModule specComments specQuotes = do
globalCfg <- liftIO getConfig
let commSpecE :: Either [Error] (ModName, Spec LocBareType LocSymbol)
commSpecE = hsSpecificationP (moduleName thisModule) (coerce specComments) specQuotes
case commSpecE of
Left errors ->
LH.filterReportErrors thisFile GHC.failM continue (getFilters globalCfg) Full errors
Right (view bareSpecIso . snd -> commSpec) -> do
env <- env_top <$> getEnv
res <- liftIO $ SpecFinder.findCompanionSpec env thisModule
case res of
SpecFound _ _ companionSpec -> do
debugLog $ "Companion spec found for " ++ renderModule thisModule
pure $ Right $ commSpec <> companionSpec
_ -> pure $ Right commSpec
where
continue = pure $ Left (ErrorsOccurred [])
processModule :: LiquidHaskellContext -> TcM (Either LiquidCheckException ProcessModuleResult)
processModule LiquidHaskellContext{..} = do
debugLog ("Module ==> " ++ renderModule thisModule)
hscEnv <- env_top <$> getEnv
let bareSpec = lhInputSpec
-- /NOTE/: For the Plugin to work correctly, we shouldn't call 'canonicalizePath', because otherwise
-- this won't trigger the \"external name resolution\" as part of 'Language.Haskell.Liquid.Bare.Resolve'
-- (cfr. 'allowExtResolution').
let file = LH.modSummaryHsFile lhModuleSummary
_ <- liftIO $ LH.checkFilePragmas $ Ms.pragmas (review bareSpecIso bareSpec)
withPragmas lhGlobalCfg file (Ms.pragmas $ review bareSpecIso bareSpec) $ \moduleCfg -> do
dependencies <- loadDependencies moduleCfg
thisModule
(S.toList lhRelevantModules)
debugLog $ "Found " <> show (HM.size $ getDependencies dependencies) <> " dependencies:"
when debugLogs $
forM_ (HM.keys . getDependencies $ dependencies) $ debugLog . moduleStableString . unStableModule
debugLog $ "mg_exports => " ++ O.showSDocUnsafe (O.ppr $ mg_exports modGuts)
debugLog $ "mg_tcs => " ++ O.showSDocUnsafe (O.ppr $ mg_tcs modGuts)
targetSrc <- liftIO $ makeTargetSrc moduleCfg file lhModuleTcData modGuts hscEnv
logger <- getLogger
dynFlags <- getDynFlags
-- See https://github.com/ucsd-progsys/liquidhaskell/issues/1711
-- Due to the fact the internals can throw exceptions from pure code at any point, we need to
-- call 'evaluate' to force any exception and catch it, if we can.
result <-
makeTargetSpec moduleCfg lhModuleLogicMap targetSrc bareSpec dependencies
let continue = pure $ Left (ErrorsOccurred [])
reportErrs :: (Show e, F.PPrint e) => [TError e] -> TcRn (Either LiquidCheckException ProcessModuleResult)
reportErrs = LH.filterReportErrors file GHC.failM continue (getFilters moduleCfg) Full
(case result of
-- Print warnings and errors, aborting the compilation.
Left diagnostics -> do
liftIO $ mapM_ (printWarning logger dynFlags) (allWarnings diagnostics)
reportErrs $ allErrors diagnostics
Right (warnings, targetSpec, liftedSpec) -> do
liftIO $ mapM_ (printWarning logger dynFlags) warnings
let targetInfo = TargetInfo targetSrc targetSpec
debugLog $ "bareSpec ==> " ++ show bareSpec
debugLog $ "liftedSpec ==> " ++ show liftedSpec
let clientLib = mkLiquidLib liftedSpec & addLibDependencies dependencies
let result' = ProcessModuleResult {
pmrClientLib = clientLib
, pmrTargetInfo = targetInfo
}
pure $ Right result')
`Ex.catch` (\(e :: UserError) -> reportErrs [e])
`Ex.catch` (\(e :: Error) -> reportErrs [e])
`Ex.catch` (\(es :: [Error]) -> reportErrs es)
where
modGuts = lhModuleGuts
thisModule = mg_module modGuts
makeTargetSrc :: Config
-> FilePath
-> TcData
-> ModGuts
-> HscEnv
-> IO TargetSrc
makeTargetSrc cfg file tcData modGuts hscEnv = do
coreBinds <- anormalize cfg hscEnv modGuts
-- The type constructors for a module are the (nubbed) union of the ones defined and
-- the ones exported. This covers the case of \"wrapper modules\" that simply re-exports
-- everything from the imported modules.
let availTcs = tcAvailableTyCons tcData
let allTcs = L.nub (mgi_tcs mgiModGuts ++ availTcs)
let dataCons = concatMap (map dataConWorkId . tyConDataCons) allTcs
let (fiTcs, fiDcs) = LH.makeFamInstEnv (getFamInstances modGuts)
let things = tcResolvedNames tcData
let impVars = LH.importVars coreBinds ++ LH.classCons (mgi_cls_inst mgiModGuts)
debugLog $ "_gsTcs => " ++ show allTcs
debugLog $ "_gsFiTcs => " ++ show fiTcs
debugLog $ "_gsFiDcs => " ++ show fiDcs
debugLog $ "dataCons => " ++ show dataCons
debugLog $ "coreBinds => " ++ (O.showSDocUnsafe . O.ppr $ coreBinds)
debugLog $ "impVars => " ++ (O.showSDocUnsafe . O.ppr $ impVars)
debugLog $ "defVars => " ++ show (L.nub $ dataCons ++ letVars coreBinds ++ tcAvailableVars tcData)
debugLog $ "useVars => " ++ (O.showSDocUnsafe . O.ppr $ readVars coreBinds)
debugLog $ "derVars => " ++ (O.showSDocUnsafe . O.ppr $ HS.fromList (LH.derivedVars cfg mgiModGuts))
debugLog $ "gsExports => " ++ show (mgi_exports mgiModGuts)
debugLog $ "gsTcs => " ++ (O.showSDocUnsafe . O.ppr $ allTcs)
debugLog $ "gsCls => " ++ (O.showSDocUnsafe . O.ppr $ mgi_cls_inst mgiModGuts)
debugLog $ "gsFiTcs => " ++ (O.showSDocUnsafe . O.ppr $ fiTcs)
debugLog $ "gsFiDcs => " ++ show fiDcs
debugLog $ "gsPrimTcs => " ++ (O.showSDocUnsafe . O.ppr $ GHC.primTyCons)
debugLog $ "things => " ++ (O.showSDocUnsafe . O.vcat . map O.ppr $ things)
debugLog $ "allImports => " ++ show (tcAllImports tcData)
debugLog $ "qualImports => " ++ show (tcQualifiedImports tcData)
return $ TargetSrc
{ giTarget = file
, giTargetMod = ModName Target (moduleName (mg_module modGuts))
, giCbs = coreBinds
, giImpVars = impVars
, giDefVars = L.nub $ dataCons ++ letVars coreBinds ++ tcAvailableVars tcData
, giUseVars = readVars coreBinds
, giDerVars = HS.fromList (LH.derivedVars cfg mgiModGuts)
, gsExports = mgi_exports mgiModGuts
, gsTcs = allTcs
, gsCls = mgi_cls_inst mgiModGuts
, gsFiTcs = fiTcs
, gsFiDcs = fiDcs
, gsPrimTcs = GHC.primTyCons
, gsQualImps = tcQualifiedImports tcData
, gsAllImps = tcAllImports tcData
, gsTyThings = [ t | (_, Just t) <- things ]
}
where
mgiModGuts :: MGIModGuts
mgiModGuts = miModGuts deriv modGuts
where
deriv = Just $ instEnvElts $ mg_inst_env modGuts
getFamInstances :: ModGuts -> [FamInst]
getFamInstances guts = famInstEnvElts (mg_fam_inst_env guts)