Support parsedResultAction of GHC plugins (#795)

* add failing test

* add fix (disable hasrecord due to linker error on my local machine)

* re-enable record-hasfield

* Allow CPP in Preprocessor module

* Revert "Allow CPP in Preprocessor module"

This reverts commit c392150421.

* apply pr 801

* move all the CPP to D.I.GHC.Compat

* fix hlint complaint

* unconditionally import MonadIO

* refactor, address PR comments

* isolate the two plugin tests

* minimize diff

* Fix test timeout

* Disable record pre processor test in 8.4

* Fix compiler warning on 8.4

* Fix yet another warning in 8.4

* Explicitly import for 8.4

* 8.4 again

* Don't apply this plugin in 8.4

The Plugins import is unavailable in 8.4

* CPP at it again
This commit is contained in:
Sridhar Ratnakumar 2020-09-16 03:57:44 -04:00 committed by GitHub
parent 91cb1a96e0
commit 8d478394af
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 101 additions and 44 deletions

View File

@ -350,6 +350,10 @@ test-suite ghcide-tests
tasty-quickcheck, tasty-quickcheck,
tasty-rerun, tasty-rerun,
text text
if (impl(ghc >= 8.6))
build-depends:
record-dot-preprocessor,
record-hasfield
hs-source-dirs: test/cabal test/exe test/src bench/lib hs-source-dirs: test/cabal test/exe test/src bench/lib
include-dirs: include include-dirs: include
ghc-options: -threaded -Wall -Wno-name-shadowing -O0 ghc-options: -threaded -Wall -Wno-name-shadowing -O0

View File

@ -43,7 +43,6 @@ import Development.IDE.Types.Options
import Development.IDE.Types.Location import Development.IDE.Types.Location
#if MIN_GHC_API_VERSION(8,6,0) #if MIN_GHC_API_VERSION(8,6,0)
import DynamicLoading (initializePlugins)
import LoadIface (loadModuleInterface) import LoadIface (loadModuleInterface)
#endif #endif
@ -101,8 +100,8 @@ parseModule
parseModule IdeOptions{..} env comp_pkgs filename modTime mbContents = parseModule IdeOptions{..} env comp_pkgs filename modTime mbContents =
fmap (either (, Nothing) id) $ fmap (either (, Nothing) id) $
evalGhcEnv env $ runExceptT $ do evalGhcEnv env $ runExceptT $ do
(contents, dflags) <- preprocessor filename mbContents (contents, dflags) <- preprocessor env filename mbContents
(diag, modu) <- parseFileContents optPreprocessor dflags comp_pkgs filename modTime contents (diag, modu) <- parseFileContents env optPreprocessor dflags comp_pkgs filename modTime contents
return (diag, Just (contents, modu)) return (diag, Just (contents, modu))
@ -456,12 +455,13 @@ getModSummaryFromBuffer fp modTime dflags parsed contents = do
-- Runs preprocessors as needed. -- Runs preprocessors as needed.
getModSummaryFromImports getModSummaryFromImports
:: (HasDynFlags m, ExceptionMonad m, MonadIO m) :: (HasDynFlags m, ExceptionMonad m, MonadIO m)
=> FilePath => HscEnv
-> FilePath
-> UTCTime -> UTCTime
-> Maybe SB.StringBuffer -> Maybe SB.StringBuffer
-> ExceptT [FileDiagnostic] m ModSummary -> ExceptT [FileDiagnostic] m ModSummary
getModSummaryFromImports fp modTime contents = do getModSummaryFromImports env fp modTime contents = do
(contents, dflags) <- preprocessor fp contents (contents, dflags) <- preprocessor env fp contents
(srcImports, textualImports, L _ moduleName) <- (srcImports, textualImports, L _ moduleName) <-
ExceptT $ liftIO $ first (diagFromErrMsgs "parser" dflags) <$> GHC.getHeaderImports dflags contents fp fp ExceptT $ liftIO $ first (diagFromErrMsgs "parser" dflags) <$> GHC.getHeaderImports dflags contents fp fp
@ -532,14 +532,15 @@ parseHeader dflags filename contents = do
-- parsed module (or errors) and any parse warnings. Does not run any preprocessors -- parsed module (or errors) and any parse warnings. Does not run any preprocessors
parseFileContents parseFileContents
:: GhcMonad m :: GhcMonad m
=> (GHC.ParsedSource -> IdePreprocessedSource) => HscEnv
-> (GHC.ParsedSource -> IdePreprocessedSource)
-> DynFlags -- ^ flags to use -> DynFlags -- ^ flags to use
-> [PackageName] -- ^ The package imports to ignore -> [PackageName] -- ^ The package imports to ignore
-> FilePath -- ^ the filename (for source locations) -> FilePath -- ^ the filename (for source locations)
-> UTCTime -- ^ the modification timestamp -> UTCTime -- ^ the modification timestamp
-> SB.StringBuffer -- ^ Haskell module source text (full Unicode is supported) -> SB.StringBuffer -- ^ Haskell module source text (full Unicode is supported)
-> ExceptT [FileDiagnostic] m ([FileDiagnostic], ParsedModule) -> ExceptT [FileDiagnostic] m ([FileDiagnostic], ParsedModule)
parseFileContents customPreprocessor dflags comp_pkgs filename modTime contents = do parseFileContents env customPreprocessor dflags comp_pkgs filename modTime contents = do
let loc = mkRealSrcLoc (mkFastString filename) 1 1 let loc = mkRealSrcLoc (mkFastString filename) 1 1
case unP Parser.parseModule (mkPState dflags contents loc) of case unP Parser.parseModule (mkPState dflags contents loc) of
#if MIN_GHC_API_VERSION(8,10,0) #if MIN_GHC_API_VERSION(8,10,0)
@ -574,17 +575,17 @@ parseFileContents customPreprocessor dflags comp_pkgs filename modTime contents
let parsed' = removePackageImports comp_pkgs parsed let parsed' = removePackageImports comp_pkgs parsed
let preproc_warnings = diagFromStrings "parser" DsWarning preproc_warns let preproc_warnings = diagFromStrings "parser" DsWarning preproc_warns
ms <- getModSummaryFromBuffer filename modTime dflags parsed' contents ms <- getModSummaryFromBuffer filename modTime dflags parsed' contents
parsed'' <- liftIO $ applyPluginsParsedResultAction env dflags ms hpm_annotations parsed
let pm = let pm =
ParsedModule { ParsedModule {
pm_mod_summary = ms pm_mod_summary = ms
, pm_parsed_source = parsed' , pm_parsed_source = parsed''
, pm_extra_src_files=[] -- src imports not allowed , pm_extra_src_files=[] -- src imports not allowed
, pm_annotations = hpm_annotations , pm_annotations = hpm_annotations
} }
warnings = diagFromErrMsgs "parser" dflags warns warnings = diagFromErrMsgs "parser" dflags warns
pure (warnings ++ preproc_warnings, pm) pure (warnings ++ preproc_warnings, pm)
-- | After parsing the module remove all package imports referring to -- | After parsing the module remove all package imports referring to
-- these packages as we have already dealt with what they map to. -- these packages as we have already dealt with what they map to.
removePackageImports :: [PackageName] -> GHC.ParsedSource -> GHC.ParsedSource removePackageImports :: [PackageName] -> GHC.ParsedSource -> GHC.ParsedSource

View File

@ -37,8 +37,8 @@ import Exception (ExceptionMonad)
-- | Given a file and some contents, apply any necessary preprocessors, -- | Given a file and some contents, apply any necessary preprocessors,
-- e.g. unlit/cpp. Return the resulting buffer and the DynFlags it implies. -- e.g. unlit/cpp. Return the resulting buffer and the DynFlags it implies.
preprocessor :: (ExceptionMonad m, HasDynFlags m, MonadIO m) => FilePath -> Maybe StringBuffer -> ExceptT [FileDiagnostic] m (StringBuffer, DynFlags) preprocessor :: (ExceptionMonad m, HasDynFlags m, MonadIO m) => HscEnv -> FilePath -> Maybe StringBuffer -> ExceptT [FileDiagnostic] m (StringBuffer, DynFlags)
preprocessor filename mbContents = do preprocessor env filename mbContents = do
-- Perform unlit -- Perform unlit
(isOnDisk, contents) <- (isOnDisk, contents) <-
if isLiterate filename then do if isLiterate filename then do
@ -51,7 +51,7 @@ preprocessor filename mbContents = do
return (isOnDisk, contents) return (isOnDisk, contents)
-- Perform cpp -- Perform cpp
dflags <- ExceptT $ parsePragmasIntoDynFlags filename contents dflags <- ExceptT $ parsePragmasIntoDynFlags env filename contents
(isOnDisk, contents, dflags) <- (isOnDisk, contents, dflags) <-
if not $ xopt LangExt.Cpp dflags then if not $ xopt LangExt.Cpp dflags then
return (isOnDisk, contents, dflags) return (isOnDisk, contents, dflags)
@ -68,7 +68,7 @@ preprocessor filename mbContents = do
[] -> throw e [] -> throw e
diags -> return $ Left diags diags -> return $ Left diags
) )
dflags <- ExceptT $ parsePragmasIntoDynFlags filename contents dflags <- ExceptT $ parsePragmasIntoDynFlags env filename contents
return (False, contents, dflags) return (False, contents, dflags)
-- Perform preprocessor -- Perform preprocessor
@ -76,7 +76,7 @@ preprocessor filename mbContents = do
return (contents, dflags) return (contents, dflags)
else do else do
contents <- liftIO $ runPreprocessor dflags filename $ if isOnDisk then Nothing else Just contents contents <- liftIO $ runPreprocessor dflags filename $ if isOnDisk then Nothing else Just contents
dflags <- ExceptT $ parsePragmasIntoDynFlags filename contents dflags <- ExceptT $ parsePragmasIntoDynFlags env filename contents
return (contents, dflags) return (contents, dflags)
where where
logAction :: IORef [CPPLog] -> LogAction logAction :: IORef [CPPLog] -> LogAction
@ -134,10 +134,11 @@ isLiterate x = takeExtension x `elem` [".lhs",".lhs-boot"]
-- | This reads the pragma information directly from the provided buffer. -- | This reads the pragma information directly from the provided buffer.
parsePragmasIntoDynFlags parsePragmasIntoDynFlags
:: (ExceptionMonad m, HasDynFlags m, MonadIO m) :: (ExceptionMonad m, HasDynFlags m, MonadIO m)
=> FilePath => HscEnv
-> FilePath
-> SB.StringBuffer -> SB.StringBuffer
-> m (Either [FileDiagnostic] DynFlags) -> m (Either [FileDiagnostic] DynFlags)
parsePragmasIntoDynFlags fp contents = catchSrcErrors "pragmas" $ do parsePragmasIntoDynFlags env fp contents = catchSrcErrors "pragmas" $ do
dflags0 <- getDynFlags dflags0 <- getDynFlags
let opts = Hdr.getOptions dflags0 contents fp let opts = Hdr.getOptions dflags0 contents fp
@ -145,8 +146,8 @@ parsePragmasIntoDynFlags fp contents = catchSrcErrors "pragmas" $ do
liftIO $ evaluate $ rnf opts liftIO $ evaluate $ rnf opts
(dflags, _, _) <- parseDynamicFilePragma dflags0 opts (dflags, _, _) <- parseDynamicFilePragma dflags0 opts
return $ disableWarningsAsErrors dflags dflags' <- liftIO $ initializePlugins env dflags
return $ disableWarningsAsErrors dflags'
-- | Run (unlit) literate haskell preprocessor on a file, or buffer if set -- | Run (unlit) literate haskell preprocessor on a file, or buffer if set
runLhs :: DynFlags -> FilePath -> Maybe SB.StringBuffer -> IO SB.StringBuffer runLhs :: DynFlags -> FilePath -> Maybe SB.StringBuffer -> IO SB.StringBuffer

View File

@ -735,11 +735,12 @@ isHiFileStableRule = define $ \IsHiFileStable f -> do
getModSummaryRule :: Rules () getModSummaryRule :: Rules ()
getModSummaryRule = do getModSummaryRule = do
defineEarlyCutoff $ \GetModSummary f -> do defineEarlyCutoff $ \GetModSummary f -> do
dflags <- hsc_dflags . hscEnv <$> use_ GhcSession f session <- hscEnv <$> use_ GhcSession f
let dflags = hsc_dflags session
(modTime, mFileContent) <- getFileContents f (modTime, mFileContent) <- getFileContents f
let fp = fromNormalizedFilePath f let fp = fromNormalizedFilePath f
modS <- liftIO $ evalWithDynFlags dflags $ runExceptT $ modS <- liftIO $ evalWithDynFlags dflags $ runExceptT $
getModSummaryFromImports fp modTime (textToStringBuffer <$> mFileContent) getModSummaryFromImports session fp modTime (textToStringBuffer <$> mFileContent)
case modS of case modS of
Right ms -> do Right ms -> do
let fingerPrint = hash (computeFingerprint f dflags ms, hashUTC modTime) let fingerPrint = hash (computeFingerprint f dflags ms, hashUTC modTime)

View File

@ -56,6 +56,8 @@ module Development.IDE.GHC.Compat(
disableWarningsAsErrors, disableWarningsAsErrors,
module GHC, module GHC,
initializePlugins,
applyPluginsParsedResultAction,
#if MIN_GHC_API_VERSION(8,6,0) #if MIN_GHC_API_VERSION(8,6,0)
#if MIN_GHC_API_VERSION(8,8,0) #if MIN_GHC_API_VERSION(8,8,0)
@ -112,6 +114,8 @@ import FastString (FastString)
#if MIN_GHC_API_VERSION(8,6,0) #if MIN_GHC_API_VERSION(8,6,0)
import Development.IDE.GHC.HieAst (mkHieFile) import Development.IDE.GHC.HieAst (mkHieFile)
import Development.IDE.GHC.HieBin import Development.IDE.GHC.HieBin
import qualified DynamicLoading
import Plugins (Plugin(parsedResultAction), withPlugins)
#if MIN_GHC_API_VERSION(8,8,0) #if MIN_GHC_API_VERSION(8,8,0)
import HieUtils import HieUtils
@ -467,3 +471,27 @@ wopt_unset_fatal :: DynFlags -> WarningFlag -> DynFlags
wopt_unset_fatal dfs f wopt_unset_fatal dfs f
= dfs { fatalWarningFlags = EnumSet.delete f (fatalWarningFlags dfs) } = dfs { fatalWarningFlags = EnumSet.delete f (fatalWarningFlags dfs) }
#endif #endif
#if MIN_GHC_API_VERSION(8,6,0)
initializePlugins :: HscEnv -> DynFlags -> IO DynFlags
initializePlugins env dflags = do
DynamicLoading.initializePlugins env dflags
applyPluginsParsedResultAction :: HscEnv -> DynFlags -> ModSummary -> ApiAnns -> ParsedSource -> IO ParsedSource
applyPluginsParsedResultAction env dflags ms hpm_annotations parsed = do
-- Apply parsedResultAction of plugins
let applyPluginAction p opts = parsedResultAction p opts ms
fmap hpm_module $
runHsc env $ withPlugins dflags applyPluginAction
(HsParsedModule parsed [] hpm_annotations)
#else
initializePlugins :: HscEnv -> DynFlags -> IO DynFlags
initializePlugins _env dflags = do
return dflags
applyPluginsParsedResultAction :: HscEnv -> DynFlags -> ModSummary -> ApiAnns -> ParsedSource -> IO ParsedSource
applyPluginsParsedResultAction _env _dflags _ms _hpm_annotations parsed =
return parsed
#endif

View File

@ -77,7 +77,8 @@ main = do
, codeLensesTests , codeLensesTests
, outlineTests , outlineTests
, findDefinitionAndHoverTests , findDefinitionAndHoverTests
, pluginTests , pluginSimpleTests
, pluginParsedResultTests
, preprocessorTests , preprocessorTests
, thTests , thTests
, safeTests , safeTests
@ -2250,29 +2251,43 @@ checkFileCompiles fp =
void (openTestDataDoc (dir </> fp)) void (openTestDataDoc (dir </> fp))
expectNoMoreDiagnostics 0.5 expectNoMoreDiagnostics 0.5
pluginSimpleTests :: TestTree
pluginSimpleTests =
testSessionWait "simple plugin" $ do
let content =
T.unlines
[ "{-# OPTIONS_GHC -fplugin GHC.TypeLits.KnownNat.Solver #-}"
, "{-# LANGUAGE DataKinds, ScopedTypeVariables, TypeOperators #-}"
, "module Testing where"
, "import Data.Proxy"
, "import GHC.TypeLits"
-- This function fails without plugins being initialized.
, "f :: forall n. KnownNat n => Proxy n -> Integer"
, "f _ = natVal (Proxy :: Proxy n) + natVal (Proxy :: Proxy (n+2))"
, "foo :: Int -> Int -> Int"
, "foo a b = a + c"
]
_ <- createDoc "Testing.hs" "haskell" content
expectDiagnostics
[ ( "Testing.hs",
[(DsError, (8, 14), "Variable not in scope: c")]
)
]
pluginParsedResultTests :: TestTree
pluginTests :: TestTree pluginParsedResultTests =
pluginTests = testSessionWait "plugins" $ do (`xfail84` "record-dot-preprocessor unsupported on 8.4") $ testSessionWait "parsedResultAction plugin" $ do
let content = let content =
T.unlines T.unlines
[ "{-# OPTIONS_GHC -fplugin GHC.TypeLits.KnownNat.Solver #-}" [ "{-# LANGUAGE DuplicateRecordFields, TypeApplications, FlexibleContexts, DataKinds, MultiParamTypeClasses, TypeSynonymInstances, FlexibleInstances #-}"
, "{-# LANGUAGE DataKinds, ScopedTypeVariables, TypeOperators #-}" , "{-# OPTIONS_GHC -fplugin=RecordDotPreprocessor #-}"
, "module Testing where" , "module Testing (Company(..), display) where"
, "import Data.Proxy" , "data Company = Company {name :: String}"
, "import GHC.TypeLits" , "display :: Company -> String"
-- This function fails without plugins being initialized. , "display c = c.name"
, "f :: forall n. KnownNat n => Proxy n -> Integer" ]
, "f _ = natVal (Proxy :: Proxy n) + natVal (Proxy :: Proxy (n+2))" _ <- createDoc "Testing.hs" "haskell" content
, "foo :: Int -> Int -> Int" expectNoMoreDiagnostics 1
, "foo a b = a + c"
]
_ <- createDoc "Testing.hs" "haskell" content
expectDiagnostics
[ ( "Testing.hs",
[(DsError, (8, 14), "Variable not in scope: c")]
)
]
cppTests :: TestTree cppTests :: TestTree
cppTests = cppTests =
@ -2734,6 +2749,13 @@ pattern R x y x' y' = Range (Position x y) (Position x' y')
xfail :: TestTree -> String -> TestTree xfail :: TestTree -> String -> TestTree
xfail = flip expectFailBecause xfail = flip expectFailBecause
xfail84 :: TestTree -> String -> TestTree
#if MIN_GHC_API_VERSION(8,6,0)
xfail84 t _ = t
#else
xfail84 = flip expectFailBecause
#endif
expectFailCabal :: String -> TestTree -> TestTree expectFailCabal :: String -> TestTree -> TestTree
#ifdef STACK #ifdef STACK
expectFailCabal _ = id expectFailCabal _ = id