ghc-mod/Check.hs

128 lines
3.5 KiB
Haskell
Raw Normal View History

2010-03-11 10:03:17 +00:00
module Check (checkSyntax) where
2010-04-28 12:43:32 +00:00
import Bag
2010-03-11 10:03:17 +00:00
import Control.Applicative
2010-03-11 15:20:02 +00:00
import Control.Monad
2010-04-28 12:43:32 +00:00
import Data.IORef
import DynFlags
import ErrUtils
2010-04-28 12:59:27 +00:00
import Exception
2010-04-28 12:43:32 +00:00
import FastString
import GHC
import GHC.Paths (libdir)
import HscTypes
import Outputable hiding (showSDoc)
2010-03-11 13:39:07 +00:00
import Param
2010-04-28 12:43:32 +00:00
import Pretty
2010-03-11 15:20:02 +00:00
import System.Directory
import System.FilePath
2010-03-11 10:03:17 +00:00
----------------------------------------------------------------
2010-03-11 13:39:07 +00:00
checkSyntax :: Options -> String -> IO String
checkSyntax opt file = do
let outdir = outDir opt
objfile = objectFile outdir file
makeDirectory outdir
2010-04-28 12:43:32 +00:00
removeObjFile objfile
unlines <$> check file outdir
----------------------------------------------------------------
cmdOptions :: [Located String]
cmdOptions = map noLoc ["-Wall","-fno-warn-unused-do-bind"]
check :: String -> String -> IO [String]
2010-04-28 12:59:27 +00:00
check fileName dir = ghandle ignore $ runGhc (Just libdir) $ do
2010-04-28 12:43:32 +00:00
ref <- liftIO $ newIORef []
initSession
setTargetFile fileName
loadWithLogger (refLogger ref) LoadAllTargets
liftIO $ readIORef ref
2010-03-11 10:03:17 +00:00
where
2010-04-28 12:43:32 +00:00
initSession = do
dflags <- getSessionDynFlags
(dflags',_,_) <- parseDynamicFlags dflags cmdOptions
2010-04-29 03:30:23 +00:00
setSessionDynFlags $ setGhcPackage $ setImportPath $ setOutputDir dir dflags'
2010-04-28 12:43:32 +00:00
setTargetFile file = do
target <- guessTarget file Nothing
setTargets [target]
2010-04-28 12:59:27 +00:00
ignore :: SomeException -> IO [String]
ignore _ = return []
2010-04-28 12:43:32 +00:00
----------------------------------------------------------------
refLogger :: IORef [String] -> WarnErrLogger
refLogger ref Nothing = do
warns <- map showErrMsg . bagToList <$> getWarnings
liftIO $ writeIORef ref warns
clearWarnings
refLogger ref (Just e) = do
let errs = map showErrMsg . bagToList . srcErrorMessages $ e
liftIO $ writeIORef ref errs
clearWarnings
----------------------------------------------------------------
setOutputDir :: String -> DynFlags -> DynFlags
setOutputDir f d = d {
objectDir = Just f
, hiDir = Just f
, stubDir = Just f, includePaths = f : includePaths d
}
setImportPath :: DynFlags -> DynFlags
setImportPath d = d {
importPaths = importPaths d ++ ["..","../..","../../..","../../../../.."]
}
2010-04-28 14:28:18 +00:00
setGhcPackage :: DynFlags -> DynFlags
setGhcPackage d = d {
packageFlags = ExposePackage "ghc" : packageFlags d
2010-04-29 03:30:23 +00:00
, ghcLink = NoLink
2010-04-28 14:28:18 +00:00
}
2010-04-29 03:30:23 +00:00
{-
setTarget :: DynFlags -> DynFlags
setTarget d = d {
hscTarget = HscNothing
}
-}
2010-04-28 12:43:32 +00:00
----------------------------------------------------------------
showErrMsg :: ErrMsg -> String
showErrMsg err = file ++ ":" ++ line ++ ":" ++ col ++ ":" ++ msg
where
spn = head (errMsgSpans err)
file = unpackFS (srcSpanFile spn)
line = show (srcSpanStartLine spn)
col = show (srcSpanStartCol spn)
msg = showSDoc (errMsgShortDoc err)
style :: PprStyle
style = mkUserStyle neverQualify AllTheWay
showSDoc :: SDoc -> String
showSDoc d = Pretty.showDocWith OneLineMode (d style)
2010-03-11 15:20:02 +00:00
----------------------------------------------------------------
makeDirectory :: FilePath -> IO ()
makeDirectory dir = makeDirectoryRecur $ normalise dir
where
makeDirectoryRecur "" = return ()
makeDirectoryRecur cur = do
exist <- doesDirectoryExist cur
let par = takeDirectory cur
unless exist $ do
makeDirectoryRecur par
createDirectory cur
objectFile :: FilePath -> FilePath -> FilePath
objectFile dir hsfile = dir </> replaceExtension hsfile ".o"
2010-04-28 12:43:32 +00:00
removeObjFile :: FilePath -> IO ()
removeObjFile objfile = do
exist <- doesFileExist objfile
when exist $ removeFile objfile