GHC/As a library: Difference between revisions
(→Another example: The code now works with GHC 7.6.x as well) |
DaniilFrumin (talk | contribs) (Updating the example) |
||
Line 8: | Line 8: | ||
The instructions on this page concern the API of GHC 6.10.1 and above. Please note that the GHC API is still in flux and may change quite significantly between major releases while we (the GHC team) provide new features or simplify certain aspects. | The instructions on this page concern the API of GHC 6.10.1 and above. Please note that the GHC API is still in flux and may change quite significantly between major releases while we (the GHC team) provide new features or simplify certain aspects. | ||
== Getting Started == | == Getting Started == | ||
Line 23: | Line 22: | ||
The following little program essentially does what <tt>ghc --make</tt> does. | The following little program essentially does what <tt>ghc --make</tt> does. | ||
<haskell> | |||
import GHC | |||
import GHC.Paths (libdir) | |||
import DynFlags | |||
errorHandler = defaultErrorHandler defaultFatalMessager | |||
main :: IO () | |||
main = errorHandler defaultFlushOut $ | |||
runGhc (Just libdir) $ do | |||
dflags <- getSessionDynFlags | |||
setSessionDynFlags dflags | |||
target <- guessTarget "test_main.hs" Nothing | |||
setTargets [target] | |||
load LoadAllTargets | |||
return () | |||
</haskell> | |||
If you are using an older version of GHC and the code above does not compiles for you, try this snippet instead: | |||
<haskell> | <haskell> | ||
{-# LANGUAGE CPP #-} | {-# LANGUAGE CPP #-} |
Revision as of 20:33, 22 July 2013
For instructions on the GHC API with GHC 6.8 or older please refer to GHC/As a library (up to 6.8)
Introduction
GHC's functionality can be useful for more things than just compiling Haskell programs. Important use cases are programs that analyse (and perhaps transform) Haskell code. Others include loading Haskell code dynamically in a GHCi-like manner. For this reason, a lot of GHC's features can be accessed by programs which import the ghc package.
The instructions on this page concern the API of GHC 6.10.1 and above. Please note that the GHC API is still in flux and may change quite significantly between major releases while we (the GHC team) provide new features or simplify certain aspects.
Getting Started
To use the GHC API you need GHC 6.10.1 or above and import the ghc package.
ghc -package ghc my_program.hs
In most cases you probably also want to use the ghc-paths package.
Most of the common functionality is provided by the GHC module, but occasionally you may have to import other modules. See the GHC's haddock documentation for a list of these modules. One good entry point into the docs is GHC.
A Simple Example
The following little program essentially does what ghc --make does.
import GHC
import GHC.Paths (libdir)
import DynFlags
errorHandler = defaultErrorHandler defaultFatalMessager
main :: IO ()
main = errorHandler defaultFlushOut $
runGhc (Just libdir) $ do
dflags <- getSessionDynFlags
setSessionDynFlags dflags
target <- guessTarget "test_main.hs" Nothing
setTargets [target]
load LoadAllTargets
return ()
If you are using an older version of GHC and the code above does not compiles for you, try this snippet instead:
{-# LANGUAGE CPP #-}
import GHC
import GHC.Paths ( libdir )
import DynFlags
main =
#if __GLASGOW_HASKELL__ > 704
defaultErrorHandler defaultFatalMessager defaultFlushOut $ do
#else
defaultErrorHandler defaultLogAction $ do
#endif
runGhc (Just libdir) $ do
dflags <- getSessionDynFlags
setSessionDynFlags dflags
target <- guessTarget "test_main.hs" Nothing
setTargets [target]
load LoadAllTargets
The outermost function, defaultErrorHandler, sets up proper exception handlers and prints an error message and exits with exit code 1 if it encounters one of these exceptions.
Most of GHC's high-level API requires access to a current session. Therefore, these functions require to be called inside a monad that is an instance of the GhcMonad typeclass. Two default implementations of this typeclass are Ghc and GhcT. In the above example we used the Ghc monad since we don't need to track any extra state.
The argument to runGhc is a bit tricky. GHC needs this to find its libraries, so the argument must refer to the directory that is printed by ghc --print-libdir for the same version of GHC that the program is being compiled with. Above we therefore use the ghc-paths package which provides this for us.
Another example
Here we demonstrate calling parseModule, typecheckModule, desugarModule, getNamesInScope, and getModuleGraph. This works for haskell-platform, ghc-7.0.3 to ghc-7.6.x. It also demonstrates how to enable some extensions.
--A.hs
--invoke: ghci -package ghc A.hs
{-# LANGUAGE CPP #-}
import GHC
import Outputable
import GHC.Paths ( libdir )
--GHC.Paths is available via cabal install ghc-paths
import DynFlags
targetFile = "B.hs"
main = do
res <- example
#if __GLASGOW_HASKELL__ > 704
putStrLn $ showSDoc tracingDynFlags ( ppr res )
#else
putStrLn $ showSDoc ( ppr res )
#endif
example =
#if __GLASGOW_HASKELL__ > 704
defaultErrorHandler defaultFatalMessager defaultFlushOut $ do
#else
defaultErrorHandler defaultLogAction $ do
#endif
runGhc (Just libdir) $ do
dflags <- getSessionDynFlags
let dflags' = foldl xopt_set dflags
[Opt_Cpp, Opt_ImplicitPrelude, Opt_MagicHash]
setSessionDynFlags dflags'
target <- guessTarget targetFile Nothing
setTargets [target]
load LoadAllTargets
modSum <- getModSummary $ mkModuleName "B"
p <- parseModule modSum
t <- typecheckModule p
d <- desugarModule t
l <- loadModule d
n <- getNamesInScope
c <- return $ coreModule d
g <- getModuleGraph
mapM showModule g
return $ (parsedSource d,"/n-----/n", typecheckedSource d)
--B.hs
module B where
main = print "Hello, World!"
Links
- GHC API haddock
- Great blog post about using the GHC API to get "QuickInfo" in a Visual Studio binding for Haskell.