-
Notifications
You must be signed in to change notification settings - Fork 138
/
Plugin.hs
668 lines (568 loc) · 29 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
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
-- | 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 OverloadedStrings #-}
{-# LANGUAGE RecordWildCards #-}
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE TupleSections #-}
module Language.Haskell.Liquid.GHC.Plugin (
plugin
) where
import qualified Liquid.GHC.API as O
import Liquid.GHC.API as GHC hiding (Type)
import qualified Text.PrettyPrint.HughesPJ as PJ
import qualified Language.Fixpoint.Types as F
import qualified Language.Haskell.Liquid.GHC.Misc as LH
import qualified Language.Haskell.Liquid.UX.CmdLine as LH
import qualified Language.Haskell.Liquid.GHC.Interface as LH
import Language.Haskell.Liquid.LHNameResolution (resolveLHNames)
import qualified Language.Haskell.Liquid.Liquid as LH
import qualified Language.Haskell.Liquid.Types.PrettyPrint as LH ( filterReportErrors
, filterReportErrorsWith
, defaultFilterReporter
, reduceFilters )
import qualified Language.Haskell.Liquid.GHC.Logging as LH (addTcRnUnknownMessages)
import Language.Haskell.Liquid.GHC.Plugin.Types
import qualified Language.Haskell.Liquid.GHC.Plugin.Serialisation as Serialisation
import Language.Haskell.Liquid.GHC.Plugin.SpecFinder
as SpecFinder
import Language.Haskell.Liquid.GHC.Types (MGIModGuts(..), miModGuts)
import Language.Haskell.Liquid.Transforms.InlineAux (inlineAux)
import Language.Haskell.Liquid.Transforms.Rewrite (rewriteBinds)
import Control.Monad
import qualified Control.Monad.Catch as Ex
import Control.Monad.IO.Class (MonadIO)
import Data.Coerce
import Data.Function ((&))
import qualified Data.List as L
import Data.IORef
import qualified Data.Map as M
import Data.Map ( Map )
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.Errors
import Language.Haskell.Liquid.Types.PrettyPrint
import Language.Haskell.Liquid.Types.Specs
import Language.Haskell.Liquid.Types.Types
import Language.Haskell.Liquid.Types.Visitors
import Language.Haskell.Liquid.Bare
import qualified Language.Haskell.Liquid.Bare.Resolve as Resolve
import Language.Haskell.Liquid.UX.CmdLine
import Language.Haskell.Liquid.UX.Config
-- | 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 -----------------------------------------
---------------------------------------------------------------------------------
-- | Set to 'True' to enable debug logging.
debugLogs :: Bool
debugLogs = False
---------------------------------------------------------------------------------
-- | Useful functions -----------------------------------------------------------
---------------------------------------------------------------------------------
-- | 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 {
driverPlugin = lhDynFlags
, parsedResultAction = parsePlugin
, typeCheckResultAction = typecheckPlugin
, pluginRecompile = purePlugin
}
where
liquidPlugin :: (MonadIO m) => [CommandLineOption] -> a -> (Config -> m a) -> m a
liquidPlugin opts def go = do
cfg <- liftIO $ LH.getOpts opts
if skipModule cfg then return def else go cfg
parsePlugin :: [CommandLineOption] -> ModSummary -> ParsedResult -> Hsc ParsedResult
parsePlugin opts ms parsedResult = liquidPlugin opts parsedResult $ \cfg ->
parsedHook cfg ms parsedResult
typecheckPlugin :: [CommandLineOption] -> ModSummary -> TcGblEnv -> TcM TcGblEnv
typecheckPlugin opts summary gblEnv = liquidPlugin opts gblEnv $ \cfg ->
typecheckPluginGo cfg 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.
typecheckPluginGo cfg summary gblEnv = do
logger <- getLogger
dynFlags <- getDynFlags
withTiming logger (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 warning
pure gblEnv
else do
newGblEnv <- typecheckHook cfg 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'
--------------------------------------------------------------------------------
-- | Inter-phase communication -------------------------------------------------
--------------------------------------------------------------------------------
--
-- Since we have no good way of passing extra data between different
-- phases of our plugin, we resort to leaving breadcrumbs in horrible
-- global mutable state.
--
-- Each module gets a mutable breadcrumb during compilation.
--
-- The @parseResultAction@ sets the breadcumb to @Parsed mod@.
--
-- The @typeCheckResultAction@ clears the breadcrumb on entry.
--
-- If the @typeCheckResultAction@ does not find the breadcrumb, it skips the
-- module, assuming that it has been verified already. This could happen if
-- the plugin is activated more than once on the same module (by passing
-- multiple times @-fplugin=LiquidHaskell@ to GHC).
{- HLINT ignore Breadcrumb "Use newtype instead of data" -}
-- It's basically an accidental detail that we only have one type of
-- breadcrumb at the moment. We don't want to use a `newtype` to
-- avoid communicating the false intention that a breadcrumb "is" a
-- list of `SpecComment`s.
data Breadcrumb
= Parsed ![SpecComment]
breadcrumbsRef :: IORef (Map Module Breadcrumb)
breadcrumbsRef = unsafePerformIO $ newIORef mempty
{-# NOINLINE breadcrumbsRef #-}
{- HLINT ignore swapBreadcrumb "Use tuple-section" -}
swapBreadcrumb :: (MonadIO m) => Module -> Maybe Breadcrumb -> m (Maybe Breadcrumb)
swapBreadcrumb mod0 new = liftIO $ atomicModifyIORef' breadcrumbsRef $ \breadcrumbs ->
let (old, breadcrumbs') = M.alterF (\old1 -> (old1, new)) mod0 breadcrumbs
in (breadcrumbs', old)
--------------------------------------------------------------------------------
-- | GHC Configuration & Setup -------------------------------------------------
--------------------------------------------------------------------------------
lhDynFlags :: [CommandLineOption] -> HscEnv -> IO HscEnv
lhDynFlags _ hscEnv =
return hscEnv
{ hsc_dflags =
hsc_dflags hscEnv
-- Ignore-interface-pragmas need to be unset to have access to
-- the RHS unfoldings in the `Ghc.Var`s which is
-- needed as part of the reflection of foreign functions in the logic
--
-- This needs to be active before the plugin runs, so pragmas are
-- read at the time the interface files are loaded.
`gopt_unset` Opt_IgnoreInterfacePragmas
-- 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.
`gopt_set` Opt_KeepRawTokenStream
}
--------------------------------------------------------------------------------
-- | Desugarer setting tweaks --------------------------------------------------
--------------------------------------------------------------------------------
-- | LiquidHaskell requires the desugarer to keep source note ticks
-- and to export everything.
--
-- TODO: We shouldn't rely on exports to find out what's in a ModGuts
-- https://github.com/ucsd-progsys/liquidhaskell/pull/2388#issuecomment-2411418479
desugarerDynFlags :: DynFlags -> DynFlags
desugarerDynFlags df = (foldl gopt_unset df disabledOpts)
{ debugLevel = 1 -- To keep source note ticks
, backend = interpreterBackend -- To export everything
}
where
disabledOpts =
[ Opt_EnableRewriteRules -- Prevent the simple optimizer from firing rewrite rules
-- during desugaring. See tests/reflect/pos/T2405.hs for
-- a discussion of an example where this is unwanted.
]
--------------------------------------------------------------------------------
-- | Parsing phase -------------------------------------------------------------
--------------------------------------------------------------------------------
-- | We hook at this stage of the pipeline to extract the
-- specs. Comments are preserved during parsing because we turn on
-- 'Opt_KeepRawTokenStream'. So we can get the /LH/ specs from
-- there.
parsedHook :: Config -> ModSummary -> ParsedResult -> Hsc ParsedResult
parsedHook _cfg ms parsedResult = do
-- TODO: we should parse the specs here so we can fail early
-- (before going through the whole GHC typechecker) if they don't
-- parse. Alas, LH.filterReportError is TcRn-specific.
let specComments = map mkSpecComment $ LH.extractSpecComments (parsedResultModule parsedResult)
-- See 'Breadcrumb' for more information.
_oldBreadcrumb <- swapBreadcrumb thisModule $ Just (Parsed specComments)
return parsedResult
where
thisModule = ms_mod ms
--------------------------------------------------------------------------------
-- | 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 desugaring a postprocessed version of the typechecked module, so we are
-- really independent from the \"normal\" compilation pipeline.
--
typecheckHook :: Config -> ModSummary -> TcGblEnv -> TcM (Either LiquidCheckException TcGblEnv)
typecheckHook cfg0 ms tcGblEnv = swapBreadcrumb thisModule Nothing >>= \case
Just (Parsed specComments) ->
typecheckHook' cfg0 ms tcGblEnv specComments
Nothing ->
-- The module has been verified by an earlier call to the plugin.
-- This could happen if multiple @-fplugin=LiquidHaskell@ flags are passed to GHC.
-- See 'Breadcrumb' for more information.
pure $ Right tcGblEnv
where
thisModule = ms_mod ms
typecheckHook' :: Config -> ModSummary -> TcGblEnv -> [SpecComment] -> TcM (Either LiquidCheckException TcGblEnv)
typecheckHook' cfg ms tcGblEnv specComments = do
debugLog $ "We are in module: " <> show (toStableModule thisModule)
case parseSpecComments (coerce specComments) of
Left errors ->
LH.filterReportErrors thisFile GHC.failM continue (getFilters cfg) Full errors
Right specs ->
liquidCheckModule cfg ms tcGblEnv specs
where
thisModule = ms_mod ms
thisFile = LH.modSummaryHsFile ms
continue = pure $ Left (ErrorsOccurred [])
liquidCheckModule :: Config -> ModSummary -> TcGblEnv -> [BPspec] -> TcM (Either LiquidCheckException TcGblEnv)
liquidCheckModule cfg0 ms tcg specs = do
withPragmas cfg0 thisFile pragmas $ \cfg -> do
pipelineData <- do
env <- getTopEnv
session <- Session <$> liftIO (newIORef env)
liftIO $ flip reflectGhc session $ mkPipelineData ms tcg specs
liquidLib <- setGblEnv tcg $ liquidHaskellCheckWithConfig cfg pipelineData ms
traverse (serialiseSpec tcg) liquidLib
where
thisFile = LH.modSummaryHsFile ms
pragmas = [ s | Pragma s <- specs ]
mkPipelineData :: (GhcMonad m) => ModSummary -> TcGblEnv -> [BPspec] -> m PipelineData
mkPipelineData ms tcg0 specs = do
let tcg = addNoInlinePragmasToBinds tcg0
unoptimisedGuts <- withSession $ \hsc_env ->
let lcl_hsc_env = hscUpdateFlags (noWarnings . desugarerDynFlags) hsc_env in
liftIO $ hscDesugar lcl_hsc_env ms tcg
resolvedNames <- LH.lookupTyThings tcg
avails <- LH.availableTyThings tcg (tcg_exports tcg)
let availTyCons = [ tc | ATyCon tc <- avails ]
availVars = [ var | AnId var <- avails ]
let tcData = mkTcData (tcg_rn_imports tcg) resolvedNames availTyCons availVars
return $ PipelineData unoptimisedGuts tcData specs
where
noWarnings dflags = dflags { warningFlags = mempty }
serialiseSpec :: TcGblEnv -> LiquidLib -> TcM TcGblEnv
serialiseSpec 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"
-- ---
serialisedSpec <- liftIO $ Serialisation.serialiseLiquidLib liquidLib thisModule
debugLog $ "Serialised annotation ==> " ++ (O.showSDocUnsafe . O.ppr $ serialisedSpec)
-- liftIO $ putStrLn "liquidHaskellCheck 10"
pure $ tcGblEnv { tcg_anns = serialisedSpec : tcg_anns tcGblEnv }
where
thisModule = tcg_mod tcGblEnv
processInputSpec
:: Config
-> PipelineData
-> ModSummary
-> BareSpecParsed
-> TcM (Either LiquidCheckException LiquidLib)
processInputSpec cfg pipelineData modSummary inputSpec = do
tcg <- getGblEnv
debugLog $ " Input spec: \n" ++ show (fromBareSpecParsed inputSpec)
debugLog $ "Direct ===> \n" ++ unlines (renderModule <$> directImports tcg)
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 = directImports tcg
}
-- liftIO $ putStrLn ("liquidHaskellCheck 6: " ++ show isIg)
if isIgnore inputSpec
then pure $ Left (ErrorsOccurred [])
else checkLiquidHaskellContext lhContext
liquidHaskellCheckWithConfig
:: Config -> PipelineData -> ModSummary -> TcM (Either LiquidCheckException LiquidLib)
liquidHaskellCheckWithConfig cfg pipelineData modSummary = do
-- Parse the spec comments stored in the pipeline data.
let inputSpec = snd $
hsSpecificationP (moduleName thisModule) (pdSpecComments pipelineData)
processInputSpec cfg pipelineData modSummary inputSpec
`Ex.catch` (\(e :: UserError) -> reportErrs [e])
`Ex.catch` (\(e :: Error) -> reportErrs [e])
`Ex.catch` (\(es :: [Error]) -> reportErrs es)
where
thisFile = LH.modSummaryHsFile modSummary
thisModule = ms_mod modSummary
continue :: TcM (Either LiquidCheckException a)
continue = pure $ Left (ErrorsOccurred [])
reportErrs :: (Show e, F.PPrint e) => [TError e] -> TcM (Either LiquidCheckException a)
reportErrs = LH.filterReportErrors thisFile GHC.failM continue (getFilters cfg) Full
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 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 { errorReporter = \errs ->
LH.addTcRnUnknownMessages [(sp, e) | (sp, e) <- errs]
, filterReporter = LH.defaultFilterReporter file
, failure = GHC.failM
, continue = pure ()
, matchingFilters = LH.reduceFilters (\(src, doc) -> PJ.render doc ++ " at " ++ LH.showPpr src) filters
, filters = filters
}
(LH.orMessages outputResult)
isIgnore :: Spec lname ty -> Bool
isIgnore sp = any ((== "--skip-module") . F.val) (pragmas sp)
--------------------------------------------------------------------------------
-- | Working with bare & lifted specs ------------------------------------------
--------------------------------------------------------------------------------
loadDependencies :: Config -> [Module] -> TcM TargetDependencies
loadDependencies currentModuleConfig mods = do
hscEnv <- env_top <$> getEnv
results <- SpecFinder.findRelevantSpecs
(excludeAutomaticAssumptionsFor currentModuleConfig) hscEnv mods
let deps = TargetDependencies $ foldl' processResult mempty (reverse results)
redundant <- liftIO $ configToRedundantDependencies hscEnv currentModuleConfig
debugLog $ "Redundant dependencies ==> " ++ show redundant
pure $ foldl' (flip dropDependency) deps redundant
where
processResult
:: HM.HashMap StableModule LiftedSpec
-> SpecFinderResult
-> HM.HashMap StableModule LiftedSpec
processResult acc (SpecNotFound _mdl) = acc
processResult acc (LibFound originalModule lib) =
HM.insert
(toStableModule originalModule)
(libTarget lib)
(acc <> getDependencies (libDeps lib))
data LiquidHaskellContext = LiquidHaskellContext {
lhGlobalCfg :: Config
, lhInputSpec :: BareSpecParsed
, lhModuleLogicMap :: LogicMap
, lhModuleSummary :: ModSummary
, lhModuleTcData :: TcData
, lhModuleGuts :: ModGuts
, lhRelevantModules :: [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.
}
processModule :: LiquidHaskellContext -> TcM (Either LiquidCheckException ProcessModuleResult)
processModule LiquidHaskellContext{..} = do
let modGuts0 = lhModuleGuts
thisModule = mg_module modGuts0
debugLog ("Module ==> " ++ renderModule thisModule)
let bareSpec0 = 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 bareSpec0
withPragmas lhGlobalCfg file (Ms.pragmas bareSpec0) $ \moduleCfg -> do
dependencies <- loadDependencies moduleCfg 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 modGuts0)
debugLog $ "mg_tcs => " ++ O.showSDocUnsafe (O.ppr $ mg_tcs modGuts0)
hscEnv <- getTopEnv
let preNormalizedCore = preNormalizeCore moduleCfg modGuts0
modGuts = modGuts0 { mg_binds = preNormalizedCore }
targetSrc <- liftIO $ makeTargetSrc moduleCfg file lhModuleTcData modGuts hscEnv
logger <- getLogger
-- 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.
tcg <- getGblEnv
let localVars = Resolve.makeLocalVars preNormalizedCore
eBareSpec = resolveLHNames
moduleCfg
thisModule
localVars
(imp_mods $ tcg_imports tcg)
(tcg_rdr_env tcg)
lhModuleLogicMap
bareSpec0
dependencies
result <-
case eBareSpec of
Left errors -> pure $ Left $ mkDiagnostics [] errors
Right (bareSpec, lnameEnv) ->
fmap (,bareSpec) <$>
makeTargetSpec
moduleCfg
localVars
lnameEnv
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) (allWarnings diagnostics)
reportErrs $ allErrors diagnostics
Right ((warnings, targetSpec, liftedSpec), bareSpec) -> do
liftIO $ mapM_ (printWarning logger) 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)
makeTargetSrc :: Config
-> FilePath
-> TcData
-> ModGuts
-> HscEnv
-> IO TargetSrc
makeTargetSrc cfg file tcData modGuts hscEnv = do
when (dumpPreNormalizedCore cfg) $ do
putStrLn "\n*************** Pre-normalized CoreBinds *****************\n"
putStrLn $ unlines $ L.intersperse "" $ map (GHC.showPpr (GHC.hsc_dflags hscEnv)) (mg_binds modGuts)
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
preNormalizeCore :: Config -> ModGuts -> [CoreBind]
preNormalizeCore cfg modGuts = rewriteBinds cfg inl_cbs
where
inl_cbs = inlineAux cfg (mg_module modGuts) (mg_binds modGuts)
getFamInstances :: ModGuts -> [FamInst]
getFamInstances guts = famInstEnvElts (mg_fam_inst_env guts)