lean4-htt/Lake/Build/Package.lean

139 lines
5.6 KiB
Text

/-
Copyright (c) 2017 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Authors: Gabriel Ebner, Sebastian Ullrich, Mac Malone
-/
import Lean.Data.Name
import Lean.Elab.Import
import Lake.Config.Package
import Lake.Build.Module
open System
open Lean hiding SearchPath
namespace Lake
-- # Build Packages
/-- Build the `extraDepTarget` of all dependent packages into a single target. -/
protected def Package.buildExtraDepsTarget (self : Package) : SchedulerM ActiveOpaqueTarget := do
let collect pkg depTargets := do
let extraDepTarget ← pkg.extraDepTarget.activate
let depTarget ← ActiveTarget.collectOpaqueArray depTargets
extraDepTarget.mixOpaqueAsync depTarget
let build dep recurse := do
let pkg := (← getPackageByName? dep.name).get!
let depTargets ← pkg.dependencies.mapM recurse
liftM <| collect pkg depTargets
let targetsE ← RBTopT.run' <| self.dependencies.mapM fun dep =>
buildRBTop (cmp := Name.quickCmp) build Dependency.name dep
match targetsE with
| Except.ok targets => collect self targets
| Except.error _ => panic! "dependency cycle emerged after resolution"
/-- Build the `extraDepTarget` of all workspace packages into a single target. -/
def buildExtraDepsTarget : SchedulerM ActiveOpaqueTarget := do
ActiveTarget.collectOpaqueArray <| ← do
(← getWorkspace).packageArray.mapM (·.extraDepTarget.activate)
-- # Build Package Modules
/--
Build the `extraDepTarget` of a package and its (transitive) dependencies
and then build their modules recursively using the given `build` function,
constructing a `NameMap` of the results.
-/
protected def Package.buildModuleMap [Inhabited o]
(build : ActiveOpaqueTarget → RecModuleBuild o) (self : Package)
: BuildM (NameMap o) := do
let depTarget ← self.buildExtraDepsTarget
let infos := (← self.getModuleArray).map fun mod => ModuleInfo.mk self mod
buildModuleMap infos (build depTarget)
/--
Build the `extraDepTarget` of a package and its (transitive) dependencies
and then build their modules recursively using the given `build` function,
constructing a single opaque target for the whole build.
-/
def Package.buildTarget [Inhabited i]
(build : ActiveOpaqueTarget → RecModuleTargetBuild i) (self : Package)
: SchedulerM ActiveOpaqueTarget := do
let depTarget ← self.buildExtraDepsTarget
let buildMods : BuildM _ := do
let mods ← self.getModuleArray
failOnBuildCycle <| ← RBTopT.run' do
let targets ← mods.mapM fun mod => (·.withoutInfo) <$>
buildRBTop (build depTarget) ModuleInfo.name (ModuleInfo.mk self mod)
ActiveTarget.collectOpaqueArray targets
buildMods.catchFailure fun _ => pure <| ActiveTarget.opaque failure
/-- Build the `.olean` files of package and its dependencies' modules. -/
def Package.buildOleanTarget (self : Package) : SchedulerM ActiveOpaqueTarget := do
self.buildTarget <| recBuildModuleOleanTargetWithLocalImports
/-- Build the `.olean` and `.c` files of package and its dependencies' modules. -/
def Package.buildOleanAndCTarget (self : Package) : SchedulerM ActiveOpaqueTarget := do
self.buildTarget <| recBuildModuleOleanAndCTargetWithLocalImports
def Package.buildDepOleans (self : Package) : BuildM PUnit := do
let targets ← self.dependencies.mapM fun dep => do
(← getPackageForModule? dep.name).get!.buildOleanTarget
targets.forM fun target => discard <| target.materialize
def Package.oleanTarget (self : Package) : OpaqueTarget :=
Target.opaque do pure (← self.buildOleanTarget).task
def Package.build (self : Package) : BuildM PUnit := do
discard (← self.buildOleanTarget).materialize
-- # Build Specific Modules of the Package
def Package.moduleOleanTarget (mod : Name) (self : Package) : FileTarget :=
BuildTarget.mk' (self.modToOlean mod) do
let depTarget ← self.buildExtraDepsTarget
let build := recBuildModuleOleanTargetWithLocalImports depTarget
return (← buildModule ⟨self, mod⟩ build).task
def Package.moduleOleanAndCTarget (mod : Name) (self : Package) : OleanAndCTarget :=
BuildTarget.mk' ⟨self.modToOlean mod, self.modToC mod⟩ do
let depTarget ← self.buildExtraDepsTarget
let build := recBuildModuleOleanAndCTargetWithLocalImports depTarget
return (← buildModule ⟨self, mod⟩ build).task
-- # Build Imports
/--
Construct an `Array` of `ModuleInfo`s for the workspace-local modules of
a `List` of import strings.
-/
def Workspace.processImportList
(imports : List String) (self : Workspace) : Array ModuleInfo := Id.run do
let mut localImports := #[]
for imp in imports do
let mod := imp.toName
if let some pkg := self.packageForModule? mod then
localImports := localImports.push ⟨pkg, mod⟩
return localImports
/--
Build the workspace-local modules of list of imports.
Builds only module `.olean` files if the default package facet is
just `oleans`. Otherwise, builds both `.olean` and `.c` files.
-/
def Package.buildImportsAndDeps (imports : List String) (self : Package) : BuildM PUnit := do
let depTarget ← self.buildExtraDepsTarget
if imports.isEmpty then
-- wait for deps to finish building
depTarget.buildOpaque
else
-- build local imports from list
let infos := (← getWorkspace).processImportList imports
if self.defaultFacet == PackageFacet.oleans then
let build := recBuildModuleOleanTargetWithLocalImports depTarget
let targets ← buildModuleArray infos build
targets.forM (·.buildOpaque)
else
let build := recBuildModuleOleanAndCTargetWithLocalImports depTarget
let targets ← buildModuleArray infos build
targets.forM (·.buildOpaque)