| // Copyright 2015 Google Inc. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package android |
| |
| import ( |
| "fmt" |
| "path/filepath" |
| "strings" |
| |
| "github.com/google/blueprint" |
| "github.com/google/blueprint/pathtools" |
| ) |
| |
| var ( |
| DeviceSharedLibrary = "shared_library" |
| DeviceStaticLibrary = "static_library" |
| DeviceExecutable = "executable" |
| HostSharedLibrary = "host_shared_library" |
| HostStaticLibrary = "host_static_library" |
| HostExecutable = "host_executable" |
| ) |
| |
| type ModuleBuildParams struct { |
| Rule blueprint.Rule |
| Deps blueprint.Deps |
| Depfile WritablePath |
| Description string |
| Output WritablePath |
| Outputs WritablePaths |
| ImplicitOutput WritablePath |
| ImplicitOutputs WritablePaths |
| Input Path |
| Inputs Paths |
| Implicit Path |
| Implicits Paths |
| OrderOnly Paths |
| Default bool |
| Args map[string]string |
| } |
| |
| type androidBaseContext interface { |
| Target() Target |
| TargetPrimary() bool |
| Arch() Arch |
| Os() OsType |
| Host() bool |
| Device() bool |
| Darwin() bool |
| Windows() bool |
| Debug() bool |
| PrimaryArch() bool |
| Vendor() bool |
| AConfig() Config |
| DeviceConfig() DeviceConfig |
| } |
| |
| type BaseContext interface { |
| blueprint.BaseModuleContext |
| androidBaseContext |
| } |
| |
| type ModuleContext interface { |
| blueprint.ModuleContext |
| androidBaseContext |
| |
| // Similar to Build, but takes Paths instead of []string, |
| // and performs more verification. |
| ModuleBuild(pctx blueprint.PackageContext, params ModuleBuildParams) |
| |
| ExpandSources(srcFiles, excludes []string) Paths |
| ExpandSourcesSubDir(srcFiles, excludes []string, subDir string) Paths |
| Glob(globPattern string, excludes []string) Paths |
| |
| InstallFile(installPath OutputPath, srcPath Path, deps ...Path) OutputPath |
| InstallFileName(installPath OutputPath, name string, srcPath Path, deps ...Path) OutputPath |
| InstallSymlink(installPath OutputPath, name string, srcPath OutputPath) OutputPath |
| CheckbuildFile(srcPath Path) |
| |
| AddMissingDependencies(deps []string) |
| |
| InstallInData() bool |
| InstallInSanitizerDir() bool |
| |
| RequiredModuleNames() []string |
| } |
| |
| type Module interface { |
| blueprint.Module |
| |
| GenerateAndroidBuildActions(ModuleContext) |
| DepsMutator(BottomUpMutatorContext) |
| |
| base() *ModuleBase |
| Enabled() bool |
| Target() Target |
| InstallInData() bool |
| InstallInSanitizerDir() bool |
| SkipInstall() |
| |
| AddProperties(props ...interface{}) |
| GetProperties() []interface{} |
| |
| BuildParamsForTests() []ModuleBuildParams |
| } |
| |
| type nameProperties struct { |
| // The name of the module. Must be unique across all modules. |
| Name string |
| } |
| |
| type commonProperties struct { |
| Tags []string |
| |
| // emit build rules for this module |
| Enabled *bool `android:"arch_variant"` |
| |
| // control whether this module compiles for 32-bit, 64-bit, or both. Possible values |
| // are "32" (compile for 32-bit only), "64" (compile for 64-bit only), "both" (compile for both |
| // architectures), or "first" (compile for 64-bit on a 64-bit platform, and 32-bit on a 32-bit |
| // platform |
| Compile_multilib string `android:"arch_variant"` |
| |
| Target struct { |
| Host struct { |
| Compile_multilib string |
| } |
| Android struct { |
| Compile_multilib string |
| } |
| } |
| |
| Default_multilib string `blueprint:"mutated"` |
| |
| // whether this is a proprietary vendor module, and should be installed into /vendor |
| Proprietary bool |
| |
| // vendor who owns this module |
| Owner *string |
| |
| // whether this module is device specific and should be installed into /vendor |
| Vendor bool |
| |
| // *.logtags files, to combine together in order to generate the /system/etc/event-log-tags |
| // file |
| Logtags []string |
| |
| // init.rc files to be installed if this module is installed |
| Init_rc []string |
| |
| // names of other modules to install if this module is installed |
| Required []string `android:"arch_variant"` |
| |
| // Set by TargetMutator |
| CompileTarget Target `blueprint:"mutated"` |
| CompilePrimary bool `blueprint:"mutated"` |
| |
| // Set by InitAndroidModule |
| HostOrDeviceSupported HostOrDeviceSupported `blueprint:"mutated"` |
| ArchSpecific bool `blueprint:"mutated"` |
| |
| SkipInstall bool `blueprint:"mutated"` |
| } |
| |
| type hostAndDeviceProperties struct { |
| Host_supported *bool |
| Device_supported *bool |
| } |
| |
| type Multilib string |
| |
| const ( |
| MultilibBoth Multilib = "both" |
| MultilibFirst Multilib = "first" |
| MultilibCommon Multilib = "common" |
| MultilibDefault Multilib = "" |
| ) |
| |
| type HostOrDeviceSupported int |
| |
| const ( |
| _ HostOrDeviceSupported = iota |
| HostSupported |
| HostSupportedNoCross |
| DeviceSupported |
| HostAndDeviceSupported |
| HostAndDeviceDefault |
| NeitherHostNorDeviceSupported |
| ) |
| |
| func InitAndroidModule(m Module) { |
| base := m.base() |
| base.module = m |
| |
| m.AddProperties( |
| &base.nameProperties, |
| &base.commonProperties, |
| &base.variableProperties) |
| } |
| |
| func InitAndroidArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) { |
| InitAndroidModule(m) |
| |
| base := m.base() |
| base.commonProperties.HostOrDeviceSupported = hod |
| base.commonProperties.Default_multilib = string(defaultMultilib) |
| base.commonProperties.ArchSpecific = true |
| |
| switch hod { |
| case HostAndDeviceSupported, HostAndDeviceDefault: |
| m.AddProperties(&base.hostAndDeviceProperties) |
| } |
| |
| InitArchModule(m) |
| } |
| |
| // A ModuleBase object contains the properties that are common to all Android |
| // modules. It should be included as an anonymous field in every module |
| // struct definition. InitAndroidModule should then be called from the module's |
| // factory function, and the return values from InitAndroidModule should be |
| // returned from the factory function. |
| // |
| // The ModuleBase type is responsible for implementing the GenerateBuildActions |
| // method to support the blueprint.Module interface. This method will then call |
| // the module's GenerateAndroidBuildActions method once for each build variant |
| // that is to be built. GenerateAndroidBuildActions is passed a |
| // AndroidModuleContext rather than the usual blueprint.ModuleContext. |
| // AndroidModuleContext exposes extra functionality specific to the Android build |
| // system including details about the particular build variant that is to be |
| // generated. |
| // |
| // For example: |
| // |
| // import ( |
| // "android/soong/android" |
| // ) |
| // |
| // type myModule struct { |
| // android.ModuleBase |
| // properties struct { |
| // MyProperty string |
| // } |
| // } |
| // |
| // func NewMyModule() android.Module) { |
| // m := &myModule{} |
| // m.AddProperties(&m.properties) |
| // android.InitAndroidModule(m) |
| // return m |
| // } |
| // |
| // func (m *myModule) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| // // Get the CPU architecture for the current build variant. |
| // variantArch := ctx.Arch() |
| // |
| // // ... |
| // } |
| type ModuleBase struct { |
| // Putting the curiously recurring thing pointing to the thing that contains |
| // the thing pattern to good use. |
| // TODO: remove this |
| module Module |
| |
| nameProperties nameProperties |
| commonProperties commonProperties |
| variableProperties variableProperties |
| hostAndDeviceProperties hostAndDeviceProperties |
| generalProperties []interface{} |
| archProperties []interface{} |
| customizableProperties []interface{} |
| |
| noAddressSanitizer bool |
| installFiles Paths |
| checkbuildFiles Paths |
| |
| // Used by buildTargetSingleton to create checkbuild and per-directory build targets |
| // Only set on the final variant of each module |
| installTarget string |
| checkbuildTarget string |
| blueprintDir string |
| |
| hooks hooks |
| |
| registerProps []interface{} |
| |
| // For tests |
| buildParams []ModuleBuildParams |
| } |
| |
| func (a *ModuleBase) AddProperties(props ...interface{}) { |
| a.registerProps = append(a.registerProps, props...) |
| } |
| |
| func (a *ModuleBase) GetProperties() []interface{} { |
| return a.registerProps |
| } |
| |
| func (a *ModuleBase) BuildParamsForTests() []ModuleBuildParams { |
| return a.buildParams |
| } |
| |
| // Name returns the name of the module. It may be overridden by individual module types, for |
| // example prebuilts will prepend prebuilt_ to the name. |
| func (a *ModuleBase) Name() string { |
| return a.nameProperties.Name |
| } |
| |
| // BaseModuleName returns the name of the module as specified in the blueprints file. |
| func (a *ModuleBase) BaseModuleName() string { |
| return a.nameProperties.Name |
| } |
| |
| func (a *ModuleBase) base() *ModuleBase { |
| return a |
| } |
| |
| func (a *ModuleBase) SetTarget(target Target, primary bool) { |
| a.commonProperties.CompileTarget = target |
| a.commonProperties.CompilePrimary = primary |
| } |
| |
| func (a *ModuleBase) Target() Target { |
| return a.commonProperties.CompileTarget |
| } |
| |
| func (a *ModuleBase) TargetPrimary() bool { |
| return a.commonProperties.CompilePrimary |
| } |
| |
| func (a *ModuleBase) Os() OsType { |
| return a.Target().Os |
| } |
| |
| func (a *ModuleBase) Host() bool { |
| return a.Os().Class == Host || a.Os().Class == HostCross |
| } |
| |
| func (a *ModuleBase) Arch() Arch { |
| return a.Target().Arch |
| } |
| |
| func (a *ModuleBase) ArchSpecific() bool { |
| return a.commonProperties.ArchSpecific |
| } |
| |
| func (a *ModuleBase) OsClassSupported() []OsClass { |
| switch a.commonProperties.HostOrDeviceSupported { |
| case HostSupported: |
| return []OsClass{Host, HostCross} |
| case HostSupportedNoCross: |
| return []OsClass{Host} |
| case DeviceSupported: |
| return []OsClass{Device} |
| case HostAndDeviceSupported: |
| var supported []OsClass |
| if Bool(a.hostAndDeviceProperties.Host_supported) { |
| supported = append(supported, Host, HostCross) |
| } |
| if a.hostAndDeviceProperties.Device_supported == nil || |
| *a.hostAndDeviceProperties.Device_supported { |
| supported = append(supported, Device) |
| } |
| return supported |
| default: |
| return nil |
| } |
| } |
| |
| func (a *ModuleBase) DeviceSupported() bool { |
| return a.commonProperties.HostOrDeviceSupported == DeviceSupported || |
| a.commonProperties.HostOrDeviceSupported == HostAndDeviceSupported && |
| (a.hostAndDeviceProperties.Device_supported == nil || |
| *a.hostAndDeviceProperties.Device_supported) |
| } |
| |
| func (a *ModuleBase) Enabled() bool { |
| if a.commonProperties.Enabled == nil { |
| return !a.Os().DefaultDisabled |
| } |
| return *a.commonProperties.Enabled |
| } |
| |
| func (a *ModuleBase) SkipInstall() { |
| a.commonProperties.SkipInstall = true |
| } |
| |
| func (a *ModuleBase) computeInstallDeps( |
| ctx blueprint.ModuleContext) Paths { |
| |
| result := Paths{} |
| ctx.VisitDepsDepthFirstIf(isFileInstaller, |
| func(m blueprint.Module) { |
| fileInstaller := m.(fileInstaller) |
| files := fileInstaller.filesToInstall() |
| result = append(result, files...) |
| }) |
| |
| return result |
| } |
| |
| func (a *ModuleBase) filesToInstall() Paths { |
| return a.installFiles |
| } |
| |
| func (p *ModuleBase) NoAddressSanitizer() bool { |
| return p.noAddressSanitizer |
| } |
| |
| func (p *ModuleBase) InstallInData() bool { |
| return false |
| } |
| |
| func (p *ModuleBase) InstallInSanitizerDir() bool { |
| return false |
| } |
| |
| func (a *ModuleBase) generateModuleTarget(ctx blueprint.ModuleContext) { |
| allInstalledFiles := Paths{} |
| allCheckbuildFiles := Paths{} |
| ctx.VisitAllModuleVariants(func(module blueprint.Module) { |
| a := module.(Module).base() |
| allInstalledFiles = append(allInstalledFiles, a.installFiles...) |
| allCheckbuildFiles = append(allCheckbuildFiles, a.checkbuildFiles...) |
| }) |
| |
| deps := []string{} |
| |
| if len(allInstalledFiles) > 0 { |
| name := ctx.ModuleName() + "-install" |
| ctx.Build(pctx, blueprint.BuildParams{ |
| Rule: blueprint.Phony, |
| Outputs: []string{name}, |
| Implicits: allInstalledFiles.Strings(), |
| Optional: ctx.Config().(Config).EmbeddedInMake(), |
| }) |
| deps = append(deps, name) |
| a.installTarget = name |
| } |
| |
| if len(allCheckbuildFiles) > 0 { |
| name := ctx.ModuleName() + "-checkbuild" |
| ctx.Build(pctx, blueprint.BuildParams{ |
| Rule: blueprint.Phony, |
| Outputs: []string{name}, |
| Implicits: allCheckbuildFiles.Strings(), |
| Optional: true, |
| }) |
| deps = append(deps, name) |
| a.checkbuildTarget = name |
| } |
| |
| if len(deps) > 0 { |
| suffix := "" |
| if ctx.Config().(Config).EmbeddedInMake() { |
| suffix = "-soong" |
| } |
| |
| ctx.Build(pctx, blueprint.BuildParams{ |
| Rule: blueprint.Phony, |
| Outputs: []string{ctx.ModuleName() + suffix}, |
| Implicits: deps, |
| Optional: true, |
| }) |
| |
| a.blueprintDir = ctx.ModuleDir() |
| } |
| } |
| |
| func (a *ModuleBase) androidBaseContextFactory(ctx blueprint.BaseModuleContext) androidBaseContextImpl { |
| return androidBaseContextImpl{ |
| target: a.commonProperties.CompileTarget, |
| targetPrimary: a.commonProperties.CompilePrimary, |
| vendor: a.commonProperties.Proprietary || a.commonProperties.Vendor, |
| config: ctx.Config().(Config), |
| } |
| } |
| |
| func (a *ModuleBase) GenerateBuildActions(ctx blueprint.ModuleContext) { |
| androidCtx := &androidModuleContext{ |
| module: a.module, |
| ModuleContext: ctx, |
| androidBaseContextImpl: a.androidBaseContextFactory(ctx), |
| installDeps: a.computeInstallDeps(ctx), |
| installFiles: a.installFiles, |
| missingDeps: ctx.GetMissingDependencies(), |
| } |
| |
| desc := "//" + ctx.ModuleDir() + ":" + ctx.ModuleName() + " " |
| var suffix []string |
| if androidCtx.Os().Class != Device && androidCtx.Os().Class != Generic { |
| suffix = append(suffix, androidCtx.Os().String()) |
| } |
| if !androidCtx.PrimaryArch() { |
| suffix = append(suffix, androidCtx.Arch().ArchType.String()) |
| } |
| |
| ctx.Variable(pctx, "moduleDesc", desc) |
| |
| s := "" |
| if len(suffix) > 0 { |
| s = " [" + strings.Join(suffix, " ") + "]" |
| } |
| ctx.Variable(pctx, "moduleDescSuffix", s) |
| |
| if a.Enabled() { |
| a.module.GenerateAndroidBuildActions(androidCtx) |
| if ctx.Failed() { |
| return |
| } |
| |
| a.installFiles = append(a.installFiles, androidCtx.installFiles...) |
| a.checkbuildFiles = append(a.checkbuildFiles, androidCtx.checkbuildFiles...) |
| } |
| |
| if a == ctx.FinalModule().(Module).base() { |
| a.generateModuleTarget(ctx) |
| if ctx.Failed() { |
| return |
| } |
| } |
| |
| a.buildParams = androidCtx.buildParams |
| } |
| |
| type androidBaseContextImpl struct { |
| target Target |
| targetPrimary bool |
| debug bool |
| vendor bool |
| config Config |
| } |
| |
| type androidModuleContext struct { |
| blueprint.ModuleContext |
| androidBaseContextImpl |
| installDeps Paths |
| installFiles Paths |
| checkbuildFiles Paths |
| missingDeps []string |
| module Module |
| |
| // For tests |
| buildParams []ModuleBuildParams |
| } |
| |
| func (a *androidModuleContext) ninjaError(desc string, outputs []string, err error) { |
| a.ModuleContext.Build(pctx, blueprint.BuildParams{ |
| Rule: ErrorRule, |
| Description: desc, |
| Outputs: outputs, |
| Optional: true, |
| Args: map[string]string{ |
| "error": err.Error(), |
| }, |
| }) |
| return |
| } |
| |
| func (a *androidModuleContext) Build(pctx blueprint.PackageContext, params blueprint.BuildParams) { |
| if a.missingDeps != nil { |
| a.ninjaError(params.Description, params.Outputs, |
| fmt.Errorf("module %s missing dependencies: %s\n", |
| a.ModuleName(), strings.Join(a.missingDeps, ", "))) |
| return |
| } |
| |
| params.Optional = true |
| a.ModuleContext.Build(pctx, params) |
| } |
| |
| func (a *androidModuleContext) ModuleBuild(pctx blueprint.PackageContext, params ModuleBuildParams) { |
| if a.config.captureBuild { |
| a.buildParams = append(a.buildParams, params) |
| } |
| |
| bparams := blueprint.BuildParams{ |
| Rule: params.Rule, |
| Deps: params.Deps, |
| Outputs: params.Outputs.Strings(), |
| ImplicitOutputs: params.ImplicitOutputs.Strings(), |
| Inputs: params.Inputs.Strings(), |
| Implicits: params.Implicits.Strings(), |
| OrderOnly: params.OrderOnly.Strings(), |
| Args: params.Args, |
| Optional: !params.Default, |
| } |
| |
| if params.Description != "" { |
| bparams.Description = "${moduleDesc}" + params.Description + "${moduleDescSuffix}" |
| } |
| |
| if params.Depfile != nil { |
| bparams.Depfile = params.Depfile.String() |
| } |
| if params.Output != nil { |
| bparams.Outputs = append(bparams.Outputs, params.Output.String()) |
| } |
| if params.ImplicitOutput != nil { |
| bparams.ImplicitOutputs = append(bparams.ImplicitOutputs, params.ImplicitOutput.String()) |
| } |
| if params.Input != nil { |
| bparams.Inputs = append(bparams.Inputs, params.Input.String()) |
| } |
| if params.Implicit != nil { |
| bparams.Implicits = append(bparams.Implicits, params.Implicit.String()) |
| } |
| |
| if a.missingDeps != nil { |
| a.ninjaError(bparams.Description, bparams.Outputs, |
| fmt.Errorf("module %s missing dependencies: %s\n", |
| a.ModuleName(), strings.Join(a.missingDeps, ", "))) |
| return |
| } |
| |
| a.ModuleContext.Build(pctx, bparams) |
| } |
| |
| func (a *androidModuleContext) GetMissingDependencies() []string { |
| return a.missingDeps |
| } |
| |
| func (a *androidModuleContext) AddMissingDependencies(deps []string) { |
| if deps != nil { |
| a.missingDeps = append(a.missingDeps, deps...) |
| } |
| } |
| |
| func (a *androidBaseContextImpl) Target() Target { |
| return a.target |
| } |
| |
| func (a *androidBaseContextImpl) TargetPrimary() bool { |
| return a.targetPrimary |
| } |
| |
| func (a *androidBaseContextImpl) Arch() Arch { |
| return a.target.Arch |
| } |
| |
| func (a *androidBaseContextImpl) Os() OsType { |
| return a.target.Os |
| } |
| |
| func (a *androidBaseContextImpl) Host() bool { |
| return a.target.Os.Class == Host || a.target.Os.Class == HostCross |
| } |
| |
| func (a *androidBaseContextImpl) Device() bool { |
| return a.target.Os.Class == Device |
| } |
| |
| func (a *androidBaseContextImpl) Darwin() bool { |
| return a.target.Os == Darwin |
| } |
| |
| func (a *androidBaseContextImpl) Windows() bool { |
| return a.target.Os == Windows |
| } |
| |
| func (a *androidBaseContextImpl) Debug() bool { |
| return a.debug |
| } |
| |
| func (a *androidBaseContextImpl) PrimaryArch() bool { |
| if len(a.config.Targets[a.target.Os.Class]) <= 1 { |
| return true |
| } |
| return a.target.Arch.ArchType == a.config.Targets[a.target.Os.Class][0].Arch.ArchType |
| } |
| |
| func (a *androidBaseContextImpl) AConfig() Config { |
| return a.config |
| } |
| |
| func (a *androidBaseContextImpl) DeviceConfig() DeviceConfig { |
| return DeviceConfig{a.config.deviceConfig} |
| } |
| |
| func (a *androidBaseContextImpl) Vendor() bool { |
| return a.vendor |
| } |
| |
| func (a *androidModuleContext) InstallInData() bool { |
| return a.module.InstallInData() |
| } |
| |
| func (a *androidModuleContext) InstallInSanitizerDir() bool { |
| return a.module.InstallInSanitizerDir() |
| } |
| |
| func (a *androidModuleContext) skipInstall(fullInstallPath OutputPath) bool { |
| if a.module.base().commonProperties.SkipInstall { |
| return true |
| } |
| |
| if a.Device() { |
| if a.AConfig().SkipDeviceInstall() { |
| return true |
| } |
| |
| if a.AConfig().SkipMegaDeviceInstall(fullInstallPath.String()) { |
| return true |
| } |
| } |
| |
| return false |
| } |
| |
| func (a *androidModuleContext) InstallFileName(installPath OutputPath, name string, srcPath Path, |
| deps ...Path) OutputPath { |
| |
| fullInstallPath := installPath.Join(a, name) |
| a.module.base().hooks.runInstallHooks(a, fullInstallPath, false) |
| |
| if !a.skipInstall(fullInstallPath) { |
| |
| deps = append(deps, a.installDeps...) |
| |
| var implicitDeps, orderOnlyDeps Paths |
| |
| if a.Host() { |
| // Installed host modules might be used during the build, depend directly on their |
| // dependencies so their timestamp is updated whenever their dependency is updated |
| implicitDeps = deps |
| } else { |
| orderOnlyDeps = deps |
| } |
| |
| a.ModuleBuild(pctx, ModuleBuildParams{ |
| Rule: Cp, |
| Description: "install " + fullInstallPath.Base(), |
| Output: fullInstallPath, |
| Input: srcPath, |
| Implicits: implicitDeps, |
| OrderOnly: orderOnlyDeps, |
| Default: !a.AConfig().EmbeddedInMake(), |
| }) |
| |
| a.installFiles = append(a.installFiles, fullInstallPath) |
| } |
| a.checkbuildFiles = append(a.checkbuildFiles, srcPath) |
| return fullInstallPath |
| } |
| |
| func (a *androidModuleContext) InstallFile(installPath OutputPath, srcPath Path, deps ...Path) OutputPath { |
| return a.InstallFileName(installPath, filepath.Base(srcPath.String()), srcPath, deps...) |
| } |
| |
| func (a *androidModuleContext) InstallSymlink(installPath OutputPath, name string, srcPath OutputPath) OutputPath { |
| fullInstallPath := installPath.Join(a, name) |
| a.module.base().hooks.runInstallHooks(a, fullInstallPath, true) |
| |
| if !a.skipInstall(fullInstallPath) { |
| |
| a.ModuleBuild(pctx, ModuleBuildParams{ |
| Rule: Symlink, |
| Description: "install symlink " + fullInstallPath.Base(), |
| Output: fullInstallPath, |
| OrderOnly: Paths{srcPath}, |
| Default: !a.AConfig().EmbeddedInMake(), |
| Args: map[string]string{ |
| "fromPath": srcPath.String(), |
| }, |
| }) |
| |
| a.installFiles = append(a.installFiles, fullInstallPath) |
| a.checkbuildFiles = append(a.checkbuildFiles, srcPath) |
| } |
| return fullInstallPath |
| } |
| |
| func (a *androidModuleContext) CheckbuildFile(srcPath Path) { |
| a.checkbuildFiles = append(a.checkbuildFiles, srcPath) |
| } |
| |
| type fileInstaller interface { |
| filesToInstall() Paths |
| } |
| |
| func isFileInstaller(m blueprint.Module) bool { |
| _, ok := m.(fileInstaller) |
| return ok |
| } |
| |
| func isAndroidModule(m blueprint.Module) bool { |
| _, ok := m.(Module) |
| return ok |
| } |
| |
| func findStringInSlice(str string, slice []string) int { |
| for i, s := range slice { |
| if s == str { |
| return i |
| } |
| } |
| return -1 |
| } |
| |
| func SrcIsModule(s string) string { |
| if len(s) > 1 && s[0] == ':' { |
| return s[1:] |
| } |
| return "" |
| } |
| |
| type sourceDependencyTag struct { |
| blueprint.BaseDependencyTag |
| } |
| |
| var SourceDepTag sourceDependencyTag |
| |
| // Returns a list of modules that must be depended on to satisfy filegroup or generated sources |
| // modules listed in srcFiles using ":module" syntax |
| func ExtractSourcesDeps(ctx BottomUpMutatorContext, srcFiles []string) { |
| var deps []string |
| set := make(map[string]bool) |
| |
| for _, s := range srcFiles { |
| if m := SrcIsModule(s); m != "" { |
| if _, found := set[m]; found { |
| ctx.ModuleErrorf("found source dependency duplicate: %q!", m) |
| } else { |
| set[m] = true |
| deps = append(deps, m) |
| } |
| } |
| } |
| |
| ctx.AddDependency(ctx.Module(), SourceDepTag, deps...) |
| } |
| |
| type SourceFileProducer interface { |
| Srcs() Paths |
| } |
| |
| // Returns a list of paths expanded from globs and modules referenced using ":module" syntax. |
| // ExtractSourcesDeps must have already been called during the dependency resolution phase. |
| func (ctx *androidModuleContext) ExpandSources(srcFiles, excludes []string) Paths { |
| return ctx.ExpandSourcesSubDir(srcFiles, excludes, "") |
| } |
| |
| func (ctx *androidModuleContext) ExpandSourcesSubDir(srcFiles, excludes []string, subDir string) Paths { |
| prefix := PathForModuleSrc(ctx).String() |
| |
| for i, e := range excludes { |
| j := findStringInSlice(e, srcFiles) |
| if j != -1 { |
| srcFiles = append(srcFiles[:j], srcFiles[j+1:]...) |
| } |
| |
| excludes[i] = filepath.Join(prefix, e) |
| } |
| |
| expandedSrcFiles := make(Paths, 0, len(srcFiles)) |
| for _, s := range srcFiles { |
| if m := SrcIsModule(s); m != "" { |
| module := ctx.GetDirectDepWithTag(m, SourceDepTag) |
| if srcProducer, ok := module.(SourceFileProducer); ok { |
| expandedSrcFiles = append(expandedSrcFiles, srcProducer.Srcs()...) |
| } else { |
| ctx.ModuleErrorf("srcs dependency %q is not a source file producing module", m) |
| } |
| } else if pathtools.IsGlob(s) { |
| globbedSrcFiles := ctx.Glob(filepath.Join(prefix, s), excludes) |
| expandedSrcFiles = append(expandedSrcFiles, globbedSrcFiles...) |
| for i, s := range expandedSrcFiles { |
| expandedSrcFiles[i] = s.(ModuleSrcPath).WithSubDir(ctx, subDir) |
| } |
| } else { |
| s := PathForModuleSrc(ctx, s).WithSubDir(ctx, subDir) |
| expandedSrcFiles = append(expandedSrcFiles, s) |
| } |
| } |
| |
| return expandedSrcFiles |
| } |
| |
| func (ctx *androidModuleContext) RequiredModuleNames() []string { |
| return ctx.module.base().commonProperties.Required |
| } |
| |
| func (ctx *androidModuleContext) Glob(globPattern string, excludes []string) Paths { |
| ret, err := ctx.GlobWithDeps(globPattern, excludes) |
| if err != nil { |
| ctx.ModuleErrorf("glob: %s", err.Error()) |
| } |
| return pathsForModuleSrcFromFullPath(ctx, ret) |
| } |
| |
| func init() { |
| RegisterSingletonType("buildtarget", BuildTargetSingleton) |
| } |
| |
| func BuildTargetSingleton() blueprint.Singleton { |
| return &buildTargetSingleton{} |
| } |
| |
| func parentDir(dir string) string { |
| dir, _ = filepath.Split(dir) |
| return filepath.Clean(dir) |
| } |
| |
| type buildTargetSingleton struct{} |
| |
| func (c *buildTargetSingleton) GenerateBuildActions(ctx blueprint.SingletonContext) { |
| checkbuildDeps := []string{} |
| |
| mmTarget := func(dir string) string { |
| return filepath.Join("mm", dir) |
| } |
| |
| modulesInDir := make(map[string][]string) |
| |
| ctx.VisitAllModules(func(module blueprint.Module) { |
| if a, ok := module.(Module); ok { |
| blueprintDir := a.base().blueprintDir |
| installTarget := a.base().installTarget |
| checkbuildTarget := a.base().checkbuildTarget |
| |
| if checkbuildTarget != "" { |
| checkbuildDeps = append(checkbuildDeps, checkbuildTarget) |
| modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], checkbuildTarget) |
| } |
| |
| if installTarget != "" { |
| modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], installTarget) |
| } |
| } |
| }) |
| |
| suffix := "" |
| if ctx.Config().(Config).EmbeddedInMake() { |
| suffix = "-soong" |
| } |
| |
| // Create a top-level checkbuild target that depends on all modules |
| ctx.Build(pctx, blueprint.BuildParams{ |
| Rule: blueprint.Phony, |
| Outputs: []string{"checkbuild" + suffix}, |
| Implicits: checkbuildDeps, |
| Optional: true, |
| }) |
| |
| // Ensure ancestor directories are in modulesInDir |
| dirs := sortedKeys(modulesInDir) |
| for _, dir := range dirs { |
| dir := parentDir(dir) |
| for dir != "." && dir != "/" { |
| if _, exists := modulesInDir[dir]; exists { |
| break |
| } |
| modulesInDir[dir] = nil |
| dir = parentDir(dir) |
| } |
| } |
| |
| // Make directories build their direct subdirectories |
| dirs = sortedKeys(modulesInDir) |
| for _, dir := range dirs { |
| p := parentDir(dir) |
| if p != "." && p != "/" { |
| modulesInDir[p] = append(modulesInDir[p], mmTarget(dir)) |
| } |
| } |
| |
| // Create a mm/<directory> target that depends on all modules in a directory, and depends |
| // on the mm/* targets of all of its subdirectories that contain Android.bp files. |
| for _, dir := range dirs { |
| ctx.Build(pctx, blueprint.BuildParams{ |
| Rule: blueprint.Phony, |
| Outputs: []string{mmTarget(dir)}, |
| Implicits: modulesInDir[dir], |
| // HACK: checkbuild should be an optional build, but force it |
| // enabled for now in standalone builds |
| Optional: ctx.Config().(Config).EmbeddedInMake(), |
| }) |
| } |
| } |
| |
| type AndroidModulesByName struct { |
| slice []Module |
| ctx interface { |
| ModuleName(blueprint.Module) string |
| ModuleSubDir(blueprint.Module) string |
| } |
| } |
| |
| func (s AndroidModulesByName) Len() int { return len(s.slice) } |
| func (s AndroidModulesByName) Less(i, j int) bool { |
| mi, mj := s.slice[i], s.slice[j] |
| ni, nj := s.ctx.ModuleName(mi), s.ctx.ModuleName(mj) |
| |
| if ni != nj { |
| return ni < nj |
| } else { |
| return s.ctx.ModuleSubDir(mi) < s.ctx.ModuleSubDir(mj) |
| } |
| } |
| func (s AndroidModulesByName) Swap(i, j int) { s.slice[i], s.slice[j] = s.slice[j], s.slice[i] } |