%
% (c) The GRASP/AQUA Project, Glasgow University, 19931998
%
\section{Code output phase}
\begin{code}
module CodeOutput( codeOutput, outputForeignStubs ) where
#include "HsVersions.h"
#ifndef OMIT_NATIVE_CODEGEN
import UniqSupply ( mkSplitUniqSupply )
import AsmCodeGen ( nativeCodeGen )
#endif
#ifdef JAVA
import JavaGen ( javaGen )
import qualified PrintJava
import OccurAnal ( occurAnalyseBinds )
#endif
import Finder ( mkStubPaths )
import PprC ( writeCs )
import CmmLint ( cmmLint )
import Packages
import Util
import Cmm ( RawCmm )
import HscTypes
import DynFlags
import ErrUtils ( dumpIfSet_dyn, showPass, ghcExit )
import Outputable
import Module
import Maybes ( firstJust )
import Control.Exception
import Control.Monad
import System.Directory
import System.FilePath
import System.IO
\end{code}
%************************************************************************
%* *
\subsection{Steering}
%* *
%************************************************************************
\begin{code}
codeOutput :: DynFlags
-> Module
-> ModLocation
-> ForeignStubs
-> [PackageId]
-> [RawCmm]
-> IO (Bool, Bool)
codeOutput dflags this_mod location foreign_stubs pkg_deps flat_abstractC
=
do { when (dopt Opt_DoCmmLinting dflags) $ do
{ showPass dflags "CmmLint"
; let lints = map cmmLint flat_abstractC
; case firstJust lints of
Just err -> do { printDump err
; ghcExit dflags 1
}
Nothing -> return ()
}
; showPass dflags "CodeOutput"
; let filenm = hscOutName dflags
; stubs_exist <- outputForeignStubs dflags this_mod location foreign_stubs
; case hscTarget dflags of {
HscInterpreted -> return ();
HscAsm -> outputAsm dflags filenm flat_abstractC;
HscC -> outputC dflags filenm flat_abstractC pkg_deps;
HscJava ->
#ifdef JAVA
outputJava dflags filenm mod_name tycons core_binds;
#else
panic "Java support not compiled into this ghc";
#endif
HscNothing -> panic "codeOutput: HscNothing"
}
; return stubs_exist
}
doOutput :: String -> (Handle -> IO ()) -> IO ()
doOutput filenm io_action = bracket (openFile filenm WriteMode) hClose io_action
\end{code}
%************************************************************************
%* *
\subsection{C}
%* *
%************************************************************************
\begin{code}
outputC :: DynFlags
-> FilePath
-> [RawCmm]
-> [PackageId]
-> IO ()
outputC dflags filenm flat_absC packages
= do
let rts = getPackageDetails (pkgState dflags) rtsPackageId
let cc_injects = unlines (map mk_include (includes rts))
mk_include h_file =
case h_file of
'"':_ -> "#include "++h_file
'<':_ -> "#include "++h_file
_ -> "#include \""++h_file++"\""
pkg_configs <- getPreloadPackagesAnd dflags packages
let pkg_names = map (display.sourcePackageId) pkg_configs
doOutput filenm $ \ h -> do
hPutStr h ("/* GHC_PACKAGES " ++ unwords pkg_names ++ "\n*/\n")
hPutStr h cc_injects
writeCs dflags h flat_absC
\end{code}
%************************************************************************
%* *
\subsection{Assembler}
%* *
%************************************************************************
\begin{code}
outputAsm :: DynFlags -> FilePath -> [RawCmm] -> IO ()
#ifndef OMIT_NATIVE_CODEGEN
outputAsm dflags filenm flat_absC
= do ncg_uniqs <- mkSplitUniqSupply 'n'
doOutput filenm $
\f ->
nativeCodeGen dflags f ncg_uniqs flat_absC
where
#else /* OMIT_NATIVE_CODEGEN */
outputAsm _ _ _
= pprPanic "This compiler was built without a native code generator"
(text "Use -fvia-C instead")
#endif
\end{code}
%************************************************************************
%* *
\subsection{Java}
%* *
%************************************************************************
\begin{code}
#ifdef JAVA
outputJava dflags filenm mod tycons core_binds
= doOutput filenm (\ f -> printForUser f alwaysQualify pp_java)
where
occ_anal_binds = occurAnalyseBinds core_binds
java_code = javaGen mod [] tycons occ_anal_binds
pp_java = PrintJava.compilationUnit java_code
#endif
\end{code}
%************************************************************************
%* *
\subsection{Foreign import/export}
%* *
%************************************************************************
\begin{code}
outputForeignStubs :: DynFlags -> Module -> ModLocation -> ForeignStubs
-> IO (Bool,
Bool)
outputForeignStubs dflags mod location stubs
= case stubs of
NoStubs -> do
stub_c_exists <- doesFileExist stub_c
stub_h_exists <- doesFileExist stub_h
return (stub_h_exists, stub_c_exists)
ForeignStubs h_code c_code -> do
let
stub_c_output_d = pprCode CStyle c_code
stub_c_output_w = showSDoc stub_c_output_d
stub_h_output_d = pprCode CStyle h_code
stub_h_output_w = showSDoc stub_h_output_d
createDirectoryHierarchy (takeDirectory stub_c)
dumpIfSet_dyn dflags Opt_D_dump_foreign
"Foreign export header file" stub_h_output_d
let rts_includes =
let rts_pkg = getPackageDetails (pkgState dflags) rtsPackageId in
concatMap mk_include (includes rts_pkg)
mk_include i = "#include \"" ++ i ++ "\"\n"
stub_h_file_exists
<- outputForeignStubs_help stub_h stub_h_output_w
("#include \"HsFFI.h\"\n" ++ cplusplus_hdr) cplusplus_ftr
dumpIfSet_dyn dflags Opt_D_dump_foreign
"Foreign export stubs" stub_c_output_d
stub_c_file_exists
<- outputForeignStubs_help stub_c stub_c_output_w
("#define IN_STG_CODE 0\n" ++
"#include \"Rts.h\"\n" ++
rts_includes ++
cplusplus_hdr)
cplusplus_ftr
return (stub_h_file_exists, stub_c_file_exists)
where
(stub_c, stub_h, _) = mkStubPaths dflags (moduleName mod) location
cplusplus_hdr = "#ifdef __cplusplus\nextern \"C\" {\n#endif\n"
cplusplus_ftr = "#ifdef __cplusplus\n}\n#endif\n"
outputForeignStubs_help :: FilePath -> String -> String -> String -> IO Bool
outputForeignStubs_help _fname "" _header _footer = return False
outputForeignStubs_help fname doc_str header footer
= do writeFile fname (header ++ doc_str ++ '\n':footer ++ "\n")
return True
\end{code}