module Rules (buildRules, oracleRules, packageTargets, topLevelTargets , toolArgsTarget ) where import qualified Hadrian.Oracles.ArgsHash import qualified Hadrian.Oracles.Cabal.Rules import qualified Hadrian.Oracles.DirectoryContents import qualified Hadrian.Oracles.Path import qualified Hadrian.Oracles.TextFile import Expression import qualified Oracles.Flavour import qualified Oracles.ModuleFiles import Packages import qualified Rules.BinaryDist import qualified Rules.Compile import qualified Rules.Configure import qualified Rules.Dependencies import qualified Rules.Documentation import qualified Rules.Generate import qualified Rules.Gmp import qualified Rules.Libffi import qualified Rules.Library import qualified Rules.Program import qualified Rules.Register import qualified Rules.Rts import qualified Rules.SimpleTargets import Settings import Settings.Program (programContext) import Target import UserSettings -- | @tool-args@ is used by tooling in order to get the arguments necessary -- to set up a GHC API session which can compile modules from GHC. When -- run, the target prints out the arguments that would be passed to @ghc@ -- during normal compilation to @stdout@. -- -- This target is called by the `ghci.sh` script in order to load all of GHC's -- modules into GHCi. toolArgsTarget :: Rules () toolArgsTarget = do "tool-args" ~> do -- We can't build DLLs on Windows (yet). Actually we should only -- include the dynamic way when we have a dynamic host GHC, but just -- checking for Windows seems simpler for now. let fake_target = target (Context Stage0 compiler (if windowsHost then vanilla else dynamic)) (Ghc ToolArgs Stage0) [] ["ignored"] -- need the autogenerated files so that they are precompiled includesDependencies Stage0 >>= need interpret fake_target Rules.Generate.compilerDependencies >>= need root <- buildRoot let dir = buildDir (vanillaContext Stage0 compiler) need [ root -/- dir -/- "Config.hs" ] need [ root -/- dir -/- "Parser.hs" ] need [ root -/- dir -/- "Lexer.hs" ] need [ root -/- dir -/- "GHC" -/- "Cmm" -/- "Parser.hs" ] need [ root -/- dir -/- "GHC" -/- "Cmm" -/- "Lexer.hs" ] -- Find out the arguments that are needed to load a module into the -- session arg_list <- interpret fake_target getArgs liftIO $ putStrLn (intercalate "\n" arg_list) allStages :: [Stage] allStages = [minBound .. maxBound] -- | This rule calls 'need' on all top-level build targets that Hadrian builds -- by default, respecting the 'finalStage' flag. topLevelTargets :: Rules () topLevelTargets = action $ do verbosity <- getVerbosity forM_ [ Stage1 ..] $ \stage -> do when (verbosity >= Loud) $ do (libraries, programs) <- partition isLibrary <$> stagePackages stage libNames <- mapM (name stage) libraries pgmNames <- mapM (name stage) programs let stageHeader t ps = "| Building " ++ show stage ++ " " ++ t ++ ": " ++ intercalate ", " ps putNormal . unlines $ [ stageHeader "libraries" libNames , stageHeader "programs" pgmNames ] let buildStages = [ s | s <- [Stage0 ..], s < finalStage ] targets <- concatForM buildStages $ \stage -> do packages <- stagePackages stage mapM (path stage) packages -- Why we need wrappers: https://gitlab.haskell.org/ghc/ghc/issues/16534. root <- buildRoot let wrappers = [ root -/- ("ghc-" ++ stageString s) | s <- [Stage1 ..] , s < finalStage ] need (targets ++ wrappers) where -- either the package database config file for libraries or -- the programPath for programs. However this still does -- not support multiple targets, where a cabal package has -- a library /and/ a program. path :: Stage -> Package -> Action FilePath path stage pkg | isLibrary pkg = pkgConfFile (vanillaContext stage pkg) | otherwise = programPath =<< programContext stage pkg name :: Stage -> Package -> Action String name stage pkg | isLibrary pkg = return (pkgName pkg) | otherwise = programName (vanillaContext stage pkg) -- TODO: Get rid of the @includeGhciLib@ hack. -- | Return the list of targets associated with a given 'Stage' and 'Package'. -- By setting the Boolean parameter to False it is possible to exclude the GHCi -- library from the targets, and avoid configuring the package to determine -- whether GHCi library needs to be built for it. We typically want to set -- this parameter to True, however it is important to set it to False when -- computing 'topLevelTargets', as otherwise the whole build gets sequentialised -- because packages are configured in the order respecting their dependencies. packageTargets :: Bool -> Stage -> Package -> Action [FilePath] packageTargets includeGhciLib stage pkg = do let context = vanillaContext stage pkg activePackages <- stagePackages stage if pkg `notElem` activePackages then return [] -- Skip inactive packages. else if isLibrary pkg then do -- Collect all targets of a library package. let pkgWays = if pkg == rts then getRtsWays else getLibraryWays ways <- interpretInContext context pkgWays libs <- mapM (pkgLibraryFile . Context stage pkg) ways more <- Rules.Library.libraryTargets includeGhciLib context setupConfig <- pkgSetupConfigFile context return $ [setupConfig] ++ libs ++ more else do -- The only target of a program package is the executable. prgContext <- programContext stage pkg prgPath <- programPath prgContext return [prgPath] packageRules :: Rules () packageRules = do -- We cannot register multiple GHC packages in parallel. Also we cannot run -- GHC when the package database is being mutated by "ghc-pkg". This is a -- classic concurrent read exclusive write (CREW) conflict. let maxConcurrentReaders = 1000 packageDb <- newResource "package-db" maxConcurrentReaders let readPackageDb = [(packageDb, 1)] writePackageDb = [(packageDb, maxConcurrentReaders)] Rules.Compile.compilePackage readPackageDb Rules.Dependencies.buildPackageDependencies readPackageDb Rules.Documentation.buildPackageDocumentation Rules.Program.buildProgramRules readPackageDb Rules.Register.configurePackageRules forM_ [Stage0 ..] (Rules.Register.registerPackageRules writePackageDb) -- TODO: Can we get rid of this enumeration of contexts? Since we iterate -- over it to generate all 4 types of rules below, all the time, we -- might want to see whether the parse-and-extract approach of -- Rules.Compile and Rules.Library could save us some time there. let vanillaContexts = liftM2 vanillaContext allStages knownPackages forM_ vanillaContexts Rules.Generate.generatePackageCode Rules.SimpleTargets.simplePackageTargets Rules.SimpleTargets.completionRule buildRules :: Rules () buildRules = do Rules.BinaryDist.bindistRules Rules.Configure.configureRules Rules.Generate.copyRules Rules.Generate.generateRules Rules.Gmp.gmpRules Rules.Libffi.libffiRules Rules.Library.libraryRules Rules.Rts.rtsRules packageRules oracleRules :: Rules () oracleRules = do Hadrian.Oracles.ArgsHash.argsHashOracle trackArgument getArgs Hadrian.Oracles.Cabal.Rules.cabalOracle Hadrian.Oracles.DirectoryContents.directoryContentsOracle Hadrian.Oracles.Path.pathOracle Hadrian.Oracles.TextFile.textFileOracle Oracles.Flavour.oracles Oracles.ModuleFiles.moduleFilesOracle