| // 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" |
| "path/filepath" |
| "sort" |
| "strings" |
| "text/scanner" |
| |
| "github.com/google/blueprint" |
| "github.com/google/blueprint/pathtools" |
| "github.com/google/blueprint/proptools" |
| ) |
| |
| var ( |
| DeviceSharedLibrary = "shared_library" |
| DeviceStaticLibrary = "static_library" |
| DeviceExecutable = "executable" |
| HostSharedLibrary = "host_shared_library" |
| HostStaticLibrary = "host_static_library" |
| HostExecutable = "host_executable" |
| ) |
| |
| type BuildParams 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 ModuleBuildParams BuildParams |
| |
| type androidBaseContext interface { |
| Target() Target |
| TargetPrimary() bool |
| MultiTargets() []Target |
| Arch() Arch |
| Os() OsType |
| Host() bool |
| Device() bool |
| Darwin() bool |
| Fuchsia() bool |
| Windows() bool |
| Debug() bool |
| PrimaryArch() bool |
| Platform() bool |
| DeviceSpecific() bool |
| SocSpecific() bool |
| ProductSpecific() bool |
| ProductServicesSpecific() bool |
| AConfig() Config |
| DeviceConfig() DeviceConfig |
| } |
| |
| type BaseContext interface { |
| BaseModuleContext |
| androidBaseContext |
| } |
| |
| // BaseModuleContext is the same as blueprint.BaseModuleContext except that Config() returns |
| // a Config instead of an interface{}. |
| type BaseModuleContext interface { |
| ModuleName() string |
| ModuleDir() string |
| ModuleType() string |
| Config() Config |
| |
| ContainsProperty(name string) bool |
| Errorf(pos scanner.Position, fmt string, args ...interface{}) |
| ModuleErrorf(fmt string, args ...interface{}) |
| PropertyErrorf(property, fmt string, args ...interface{}) |
| Failed() bool |
| |
| // GlobWithDeps returns a list of files that match the specified pattern but do not match any |
| // of the patterns in excludes. It also adds efficient dependencies to rerun the primary |
| // builder whenever a file matching the pattern as added or removed, without rerunning if a |
| // file that does not match the pattern is added to a searched directory. |
| GlobWithDeps(pattern string, excludes []string) ([]string, error) |
| |
| Fs() pathtools.FileSystem |
| AddNinjaFileDeps(deps ...string) |
| } |
| |
| type ModuleContext interface { |
| androidBaseContext |
| BaseModuleContext |
| |
| // Deprecated: use ModuleContext.Build instead. |
| ModuleBuild(pctx PackageContext, params ModuleBuildParams) |
| |
| ExpandSources(srcFiles, excludes []string) Paths |
| ExpandSource(srcFile, prop string) Path |
| ExpandOptionalSource(srcFile *string, prop string) OptionalPath |
| Glob(globPattern string, excludes []string) Paths |
| GlobFiles(globPattern string, excludes []string) Paths |
| |
| InstallExecutable(installPath OutputPath, name string, srcPath Path, deps ...Path) OutputPath |
| InstallFile(installPath OutputPath, name string, srcPath Path, deps ...Path) OutputPath |
| InstallSymlink(installPath OutputPath, name string, srcPath OutputPath) OutputPath |
| InstallAbsoluteSymlink(installPath OutputPath, name string, absPath string) OutputPath |
| CheckbuildFile(srcPath Path) |
| |
| AddMissingDependencies(deps []string) |
| |
| InstallInData() bool |
| InstallInSanitizerDir() bool |
| InstallInRecovery() bool |
| |
| RequiredModuleNames() []string |
| HostRequiredModuleNames() []string |
| TargetRequiredModuleNames() []string |
| |
| // android.ModuleContext methods |
| // These are duplicated instead of embedded so that can eventually be wrapped to take an |
| // android.Module instead of a blueprint.Module |
| OtherModuleName(m blueprint.Module) string |
| OtherModuleErrorf(m blueprint.Module, fmt string, args ...interface{}) |
| OtherModuleDependencyTag(m blueprint.Module) blueprint.DependencyTag |
| |
| GetDirectDepsWithTag(tag blueprint.DependencyTag) []Module |
| GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module |
| GetDirectDep(name string) (blueprint.Module, blueprint.DependencyTag) |
| |
| ModuleSubDir() string |
| |
| VisitDirectDepsBlueprint(visit func(blueprint.Module)) |
| VisitDirectDeps(visit func(Module)) |
| VisitDirectDepsWithTag(tag blueprint.DependencyTag, visit func(Module)) |
| VisitDirectDepsIf(pred func(Module) bool, visit func(Module)) |
| // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module |
| VisitDepsDepthFirst(visit func(Module)) |
| // Deprecated: use WalkDeps instead to support multiple dependency tags on the same module |
| VisitDepsDepthFirstIf(pred func(Module) bool, visit func(Module)) |
| WalkDeps(visit func(Module, Module) bool) |
| WalkDepsBlueprint(visit func(blueprint.Module, blueprint.Module) bool) |
| |
| Variable(pctx PackageContext, name, value string) |
| Rule(pctx PackageContext, name string, params blueprint.RuleParams, argNames ...string) blueprint.Rule |
| // Similar to blueprint.ModuleContext.Build, but takes Paths instead of []string, |
| // and performs more verification. |
| Build(pctx PackageContext, params BuildParams) |
| |
| PrimaryModule() Module |
| FinalModule() Module |
| VisitAllModuleVariants(visit func(Module)) |
| |
| GetMissingDependencies() []string |
| Namespace() blueprint.Namespace |
| } |
| |
| type Module interface { |
| blueprint.Module |
| |
| // GenerateAndroidBuildActions is analogous to Blueprints' GenerateBuildActions, |
| // but GenerateAndroidBuildActions also has access to Android-specific information. |
| // For more information, see Module.GenerateBuildActions within Blueprint's module_ctx.go |
| GenerateAndroidBuildActions(ModuleContext) |
| |
| DepsMutator(BottomUpMutatorContext) |
| |
| base() *ModuleBase |
| Enabled() bool |
| Target() Target |
| InstallInData() bool |
| InstallInSanitizerDir() bool |
| InstallInRecovery() bool |
| SkipInstall() |
| ExportedToMake() bool |
| NoticeFile() OptionalPath |
| |
| AddProperties(props ...interface{}) |
| GetProperties() []interface{} |
| |
| BuildParamsForTests() []BuildParams |
| RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams |
| VariablesForTests() map[string]string |
| } |
| |
| type nameProperties struct { |
| // The name of the module. Must be unique across all modules. |
| Name *string |
| } |
| |
| type commonProperties struct { |
| // emit build rules for this module |
| Enabled *bool `android:"arch_variant"` |
| |
| // Controls the visibility of this module to other modules. Allowable values are one or more of |
| // these formats: |
| // |
| // ["//visibility:public"]: Anyone can use this module. |
| // ["//visibility:private"]: Only rules in the module's package (not its subpackages) can use |
| // this module. |
| // ["//some/package:__pkg__", "//other/package:__pkg__"]: Only modules in some/package and |
| // other/package (defined in some/package/*.bp and other/package/*.bp) have access to |
| // this module. Note that sub-packages do not have access to the rule; for example, |
| // //some/package/foo:bar or //other/package/testing:bla wouldn't have access. __pkg__ |
| // is a special module and must be used verbatim. It represents all of the modules in the |
| // package. |
| // ["//project:__subpackages__", "//other:__subpackages__"]: Only modules in packages project |
| // or other or in one of their sub-packages have access to this module. For example, |
| // //project:rule, //project/library:lib or //other/testing/internal:munge are allowed |
| // to depend on this rule (but not //independent:evil) |
| // ["//project"]: This is shorthand for ["//project:__pkg__"] |
| // [":__subpackages__"]: This is shorthand for ["//project:__subpackages__"] where |
| // //project is the module's package. e.g. using [":__subpackages__"] in |
| // packages/apps/Settings/Android.bp is equivalent to |
| // //packages/apps/Settings:__subpackages__. |
| // ["//visibility:legacy_public"]: The default visibility, behaves as //visibility:public |
| // for now. It is an error if it is used in a module. |
| // See https://android.googlesource.com/platform/build/soong/+/master/README.md#visibility for |
| // more details. |
| Visibility []string |
| |
| // 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 |
| } |
| } |
| |
| UseTargetVariants bool `blueprint:"mutated"` |
| 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 specific to an SoC (System-On-a-Chip). When set to true, |
| // it is installed into /vendor (or /system/vendor if vendor partition does not exist). |
| // Use `soc_specific` instead for better meaning. |
| Vendor *bool |
| |
| // whether this module is specific to an SoC (System-On-a-Chip). When set to true, |
| // it is installed into /vendor (or /system/vendor if vendor partition does not exist). |
| Soc_specific *bool |
| |
| // whether this module is specific to a device, not only for SoC, but also for off-chip |
| // peripherals. When set to true, it is installed into /odm (or /vendor/odm if odm partition |
| // does not exist, or /system/vendor/odm if both odm and vendor partitions do not exist). |
| // This implies `soc_specific:true`. |
| Device_specific *bool |
| |
| // whether this module is specific to a software configuration of a product (e.g. country, |
| // network operator, etc). When set to true, it is installed into /product (or |
| // /system/product if product partition does not exist). |
| Product_specific *bool |
| |
| // whether this module provides services owned by the OS provider to the core platform. When set |
| // to true, it is installed into /product_services (or /system/product_services if |
| // product_services partition does not exist). |
| Product_services_specific *bool |
| |
| // Whether this module is installed to recovery partition |
| Recovery *bool |
| |
| // init.rc files to be installed if this module is installed |
| Init_rc []string `android:"path"` |
| |
| // VINTF manifest fragments to be installed if this module is installed |
| Vintf_fragments []string `android:"path"` |
| |
| // names of other modules to install if this module is installed |
| Required []string `android:"arch_variant"` |
| |
| // names of other modules to install on host if this module is installed |
| Host_required []string `android:"arch_variant"` |
| |
| // names of other modules to install on target if this module is installed |
| Target_required []string `android:"arch_variant"` |
| |
| // relative path to a file to include in the list of notices for the device |
| Notice *string `android:"path"` |
| |
| Dist struct { |
| // copy the output of this module to the $DIST_DIR when `dist` is specified on the |
| // command line and any of these targets are also on the command line, or otherwise |
| // built |
| Targets []string `android:"arch_variant"` |
| |
| // The name of the output artifact. This defaults to the basename of the output of |
| // the module. |
| Dest *string `android:"arch_variant"` |
| |
| // The directory within the dist directory to store the artifact. Defaults to the |
| // top level directory (""). |
| Dir *string `android:"arch_variant"` |
| |
| // A suffix to add to the artifact file name (before any extension). |
| Suffix *string `android:"arch_variant"` |
| } `android:"arch_variant"` |
| |
| // Set by TargetMutator |
| CompileTarget Target `blueprint:"mutated"` |
| CompileMultiTargets []Target `blueprint:"mutated"` |
| CompilePrimary bool `blueprint:"mutated"` |
| |
| // Set by InitAndroidModule |
| HostOrDeviceSupported HostOrDeviceSupported `blueprint:"mutated"` |
| ArchSpecific bool `blueprint:"mutated"` |
| |
| SkipInstall bool `blueprint:"mutated"` |
| |
| NamespaceExportedToMake bool `blueprint:"mutated"` |
| } |
| |
| type hostAndDeviceProperties struct { |
| // If set to true, build a variant of the module for the host. Defaults to false. |
| Host_supported *bool |
| |
| // If set to true, build a variant of the module for the device. Defaults to true. |
| Device_supported *bool |
| } |
| |
| type Multilib string |
| |
| const ( |
| MultilibBoth Multilib = "both" |
| MultilibFirst Multilib = "first" |
| MultilibCommon Multilib = "common" |
| MultilibCommonFirst Multilib = "common_first" |
| MultilibDefault Multilib = "" |
| ) |
| |
| type HostOrDeviceSupported int |
| |
| const ( |
| _ HostOrDeviceSupported = iota |
| |
| // Host and HostCross are built by default. Device is not supported. |
| HostSupported |
| |
| // Host is built by default. HostCross and Device are not supported. |
| HostSupportedNoCross |
| |
| // Device is built by default. Host and HostCross are not supported. |
| DeviceSupported |
| |
| // Device is built by default. Host and HostCross are supported. |
| HostAndDeviceSupported |
| |
| // Host, HostCross, and Device are built by default. |
| HostAndDeviceDefault |
| |
| // Nothing is supported. This is not exposed to the user, but used to mark a |
| // host only module as unsupported when the module type is not supported on |
| // the host OS. E.g. benchmarks are supported on Linux but not Darwin. |
| NeitherHostNorDeviceSupported |
| ) |
| |
| type moduleKind int |
| |
| const ( |
| platformModule moduleKind = iota |
| deviceSpecificModule |
| socSpecificModule |
| productSpecificModule |
| productServicesSpecificModule |
| ) |
| |
| func (k moduleKind) String() string { |
| switch k { |
| case platformModule: |
| return "platform" |
| case deviceSpecificModule: |
| return "device-specific" |
| case socSpecificModule: |
| return "soc-specific" |
| case productSpecificModule: |
| return "product-specific" |
| case productServicesSpecificModule: |
| return "productservices-specific" |
| default: |
| panic(fmt.Errorf("unknown module kind %d", k)) |
| } |
| } |
| |
| func InitAndroidModule(m Module) { |
| base := m.base() |
| base.module = m |
| |
| m.AddProperties( |
| &base.nameProperties, |
| &base.commonProperties, |
| &base.variableProperties) |
| base.generalProperties = m.GetProperties() |
| base.customizableProperties = m.GetProperties() |
| } |
| |
| 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 |
| base.commonProperties.UseTargetVariants = true |
| |
| switch hod { |
| case HostAndDeviceSupported, HostAndDeviceDefault: |
| m.AddProperties(&base.hostAndDeviceProperties) |
| } |
| |
| InitArchModule(m) |
| } |
| |
| func InitAndroidMultiTargetsArchModule(m Module, hod HostOrDeviceSupported, defaultMultilib Multilib) { |
| InitAndroidArchModule(m, hod, defaultMultilib) |
| m.base().commonProperties.UseTargetVariants = false |
| } |
| |
| // 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 |
| noticeFile OptionalPath |
| |
| // Used by buildTargetSingleton to create checkbuild and per-directory build targets |
| // Only set on the final variant of each module |
| installTarget WritablePath |
| checkbuildTarget WritablePath |
| blueprintDir string |
| |
| hooks hooks |
| |
| registerProps []interface{} |
| |
| // For tests |
| buildParams []BuildParams |
| ruleParams map[blueprint.Rule]blueprint.RuleParams |
| variables map[string]string |
| |
| prefer32 func(ctx BaseModuleContext, base *ModuleBase, class OsClass) bool |
| } |
| |
| func (a *ModuleBase) DepsMutator(BottomUpMutatorContext) {} |
| |
| func (a *ModuleBase) AddProperties(props ...interface{}) { |
| a.registerProps = append(a.registerProps, props...) |
| } |
| |
| func (a *ModuleBase) GetProperties() []interface{} { |
| return a.registerProps |
| } |
| |
| func (a *ModuleBase) BuildParamsForTests() []BuildParams { |
| return a.buildParams |
| } |
| |
| func (a *ModuleBase) RuleParamsForTests() map[blueprint.Rule]blueprint.RuleParams { |
| return a.ruleParams |
| } |
| |
| func (a *ModuleBase) VariablesForTests() map[string]string { |
| return a.variables |
| } |
| |
| func (a *ModuleBase) Prefer32(prefer32 func(ctx BaseModuleContext, base *ModuleBase, class OsClass) bool) { |
| a.prefer32 = prefer32 |
| } |
| |
| // 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 String(a.nameProperties.Name) |
| } |
| |
| // BaseModuleName returns the name of the module as specified in the blueprints file. |
| func (a *ModuleBase) BaseModuleName() string { |
| return String(a.nameProperties.Name) |
| } |
| |
| func (a *ModuleBase) base() *ModuleBase { |
| return a |
| } |
| |
| func (a *ModuleBase) SetTarget(target Target, multiTargets []Target, primary bool) { |
| a.commonProperties.CompileTarget = target |
| a.commonProperties.CompileMultiTargets = multiTargets |
| 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) MultiTargets() []Target { |
| return a.commonProperties.CompileMultiTargets |
| } |
| |
| 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, HostAndDeviceDefault: |
| var supported []OsClass |
| if Bool(a.hostAndDeviceProperties.Host_supported) || |
| (a.commonProperties.HostOrDeviceSupported == HostAndDeviceDefault && |
| a.hostAndDeviceProperties.Host_supported == nil) { |
| 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) Platform() bool { |
| return !a.DeviceSpecific() && !a.SocSpecific() && !a.ProductSpecific() && !a.ProductServicesSpecific() |
| } |
| |
| func (a *ModuleBase) DeviceSpecific() bool { |
| return Bool(a.commonProperties.Device_specific) |
| } |
| |
| func (a *ModuleBase) SocSpecific() bool { |
| return Bool(a.commonProperties.Vendor) || Bool(a.commonProperties.Proprietary) || Bool(a.commonProperties.Soc_specific) |
| } |
| |
| func (a *ModuleBase) ProductSpecific() bool { |
| return Bool(a.commonProperties.Product_specific) |
| } |
| |
| func (a *ModuleBase) ProductServicesSpecific() bool { |
| return Bool(a.commonProperties.Product_services_specific) |
| } |
| |
| 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) ExportedToMake() bool { |
| return a.commonProperties.NamespaceExportedToMake |
| } |
| |
| func (a *ModuleBase) computeInstallDeps( |
| ctx blueprint.ModuleContext) Paths { |
| |
| result := Paths{} |
| // TODO(ccross): we need to use WalkDeps and have some way to know which dependencies require installation |
| 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 (p *ModuleBase) InstallInRecovery() bool { |
| return Bool(p.commonProperties.Recovery) |
| } |
| |
| func (a *ModuleBase) Owner() string { |
| return String(a.commonProperties.Owner) |
| } |
| |
| func (a *ModuleBase) NoticeFile() OptionalPath { |
| return a.noticeFile |
| } |
| |
| func (a *ModuleBase) generateModuleTarget(ctx ModuleContext) { |
| allInstalledFiles := Paths{} |
| allCheckbuildFiles := Paths{} |
| ctx.VisitAllModuleVariants(func(module Module) { |
| a := module.base() |
| allInstalledFiles = append(allInstalledFiles, a.installFiles...) |
| allCheckbuildFiles = append(allCheckbuildFiles, a.checkbuildFiles...) |
| }) |
| |
| var deps Paths |
| |
| namespacePrefix := ctx.Namespace().(*Namespace).id |
| if namespacePrefix != "" { |
| namespacePrefix = namespacePrefix + "-" |
| } |
| |
| if len(allInstalledFiles) > 0 { |
| name := PathForPhony(ctx, namespacePrefix+ctx.ModuleName()+"-install") |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Output: name, |
| Implicits: allInstalledFiles, |
| Default: !ctx.Config().EmbeddedInMake(), |
| }) |
| deps = append(deps, name) |
| a.installTarget = name |
| } |
| |
| if len(allCheckbuildFiles) > 0 { |
| name := PathForPhony(ctx, namespacePrefix+ctx.ModuleName()+"-checkbuild") |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Output: name, |
| Implicits: allCheckbuildFiles, |
| }) |
| deps = append(deps, name) |
| a.checkbuildTarget = name |
| } |
| |
| if len(deps) > 0 { |
| suffix := "" |
| if ctx.Config().EmbeddedInMake() { |
| suffix = "-soong" |
| } |
| |
| name := PathForPhony(ctx, namespacePrefix+ctx.ModuleName()+suffix) |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Outputs: []WritablePath{name}, |
| Implicits: deps, |
| }) |
| |
| a.blueprintDir = ctx.ModuleDir() |
| } |
| } |
| |
| func determineModuleKind(a *ModuleBase, ctx blueprint.BaseModuleContext) moduleKind { |
| var socSpecific = Bool(a.commonProperties.Vendor) || Bool(a.commonProperties.Proprietary) || Bool(a.commonProperties.Soc_specific) |
| var deviceSpecific = Bool(a.commonProperties.Device_specific) |
| var productSpecific = Bool(a.commonProperties.Product_specific) |
| var productServicesSpecific = Bool(a.commonProperties.Product_services_specific) |
| |
| msg := "conflicting value set here" |
| if socSpecific && deviceSpecific { |
| ctx.PropertyErrorf("device_specific", "a module cannot be specific to SoC and device at the same time.") |
| if Bool(a.commonProperties.Vendor) { |
| ctx.PropertyErrorf("vendor", msg) |
| } |
| if Bool(a.commonProperties.Proprietary) { |
| ctx.PropertyErrorf("proprietary", msg) |
| } |
| if Bool(a.commonProperties.Soc_specific) { |
| ctx.PropertyErrorf("soc_specific", msg) |
| } |
| } |
| |
| if productSpecific && productServicesSpecific { |
| ctx.PropertyErrorf("product_specific", "a module cannot be specific to product and product_services at the same time.") |
| ctx.PropertyErrorf("product_services_specific", msg) |
| } |
| |
| if (socSpecific || deviceSpecific) && (productSpecific || productServicesSpecific) { |
| if productSpecific { |
| ctx.PropertyErrorf("product_specific", "a module cannot be specific to SoC or device and product at the same time.") |
| } else { |
| ctx.PropertyErrorf("product_services_specific", "a module cannot be specific to SoC or device and product_services at the same time.") |
| } |
| if deviceSpecific { |
| ctx.PropertyErrorf("device_specific", msg) |
| } else { |
| if Bool(a.commonProperties.Vendor) { |
| ctx.PropertyErrorf("vendor", msg) |
| } |
| if Bool(a.commonProperties.Proprietary) { |
| ctx.PropertyErrorf("proprietary", msg) |
| } |
| if Bool(a.commonProperties.Soc_specific) { |
| ctx.PropertyErrorf("soc_specific", msg) |
| } |
| } |
| } |
| |
| if productSpecific { |
| return productSpecificModule |
| } else if productServicesSpecific { |
| return productServicesSpecificModule |
| } else if deviceSpecific { |
| return deviceSpecificModule |
| } else if socSpecific { |
| return socSpecificModule |
| } else { |
| return platformModule |
| } |
| } |
| |
| func (a *ModuleBase) androidBaseContextFactory(ctx blueprint.BaseModuleContext) androidBaseContextImpl { |
| return androidBaseContextImpl{ |
| target: a.commonProperties.CompileTarget, |
| targetPrimary: a.commonProperties.CompilePrimary, |
| multiTargets: a.commonProperties.CompileMultiTargets, |
| kind: determineModuleKind(a, ctx), |
| config: ctx.Config().(Config), |
| } |
| } |
| |
| func (a *ModuleBase) GenerateBuildActions(blueprintCtx blueprint.ModuleContext) { |
| ctx := &androidModuleContext{ |
| module: a.module, |
| ModuleContext: blueprintCtx, |
| androidBaseContextImpl: a.androidBaseContextFactory(blueprintCtx), |
| installDeps: a.computeInstallDeps(blueprintCtx), |
| installFiles: a.installFiles, |
| missingDeps: blueprintCtx.GetMissingDependencies(), |
| variables: make(map[string]string), |
| } |
| |
| if ctx.config.captureBuild { |
| ctx.ruleParams = make(map[blueprint.Rule]blueprint.RuleParams) |
| } |
| |
| desc := "//" + ctx.ModuleDir() + ":" + ctx.ModuleName() + " " |
| var suffix []string |
| if ctx.Os().Class != Device && ctx.Os().Class != Generic { |
| suffix = append(suffix, ctx.Os().String()) |
| } |
| if !ctx.PrimaryArch() { |
| suffix = append(suffix, ctx.Arch().ArchType.String()) |
| } |
| |
| ctx.Variable(pctx, "moduleDesc", desc) |
| |
| s := "" |
| if len(suffix) > 0 { |
| s = " [" + strings.Join(suffix, " ") + "]" |
| } |
| ctx.Variable(pctx, "moduleDescSuffix", s) |
| |
| // Some common property checks for properties that will be used later in androidmk.go |
| if a.commonProperties.Dist.Dest != nil { |
| _, err := validateSafePath(*a.commonProperties.Dist.Dest) |
| if err != nil { |
| ctx.PropertyErrorf("dist.dest", "%s", err.Error()) |
| } |
| } |
| if a.commonProperties.Dist.Dir != nil { |
| _, err := validateSafePath(*a.commonProperties.Dist.Dir) |
| if err != nil { |
| ctx.PropertyErrorf("dist.dir", "%s", err.Error()) |
| } |
| } |
| if a.commonProperties.Dist.Suffix != nil { |
| if strings.Contains(*a.commonProperties.Dist.Suffix, "/") { |
| ctx.PropertyErrorf("dist.suffix", "Suffix may not contain a '/' character.") |
| } |
| } |
| |
| if a.Enabled() { |
| a.module.GenerateAndroidBuildActions(ctx) |
| if ctx.Failed() { |
| return |
| } |
| |
| a.installFiles = append(a.installFiles, ctx.installFiles...) |
| a.checkbuildFiles = append(a.checkbuildFiles, ctx.checkbuildFiles...) |
| |
| notice := proptools.StringDefault(a.commonProperties.Notice, "NOTICE") |
| if m := SrcIsModule(notice); m != "" { |
| a.noticeFile = ctx.ExpandOptionalSource(¬ice, "notice") |
| } else { |
| noticePath := filepath.Join(ctx.ModuleDir(), notice) |
| a.noticeFile = ExistentPathForSource(ctx, noticePath) |
| } |
| } |
| |
| if a == ctx.FinalModule().(Module).base() { |
| a.generateModuleTarget(ctx) |
| if ctx.Failed() { |
| return |
| } |
| } |
| |
| a.buildParams = ctx.buildParams |
| a.ruleParams = ctx.ruleParams |
| a.variables = ctx.variables |
| } |
| |
| type androidBaseContextImpl struct { |
| target Target |
| multiTargets []Target |
| targetPrimary bool |
| debug bool |
| kind moduleKind |
| config Config |
| } |
| |
| type androidModuleContext struct { |
| blueprint.ModuleContext |
| androidBaseContextImpl |
| installDeps Paths |
| installFiles Paths |
| checkbuildFiles Paths |
| missingDeps []string |
| module Module |
| |
| // For tests |
| buildParams []BuildParams |
| ruleParams map[blueprint.Rule]blueprint.RuleParams |
| variables map[string]string |
| } |
| |
| func (a *androidModuleContext) ninjaError(desc string, outputs []string, err error) { |
| a.ModuleContext.Build(pctx.PackageContext, blueprint.BuildParams{ |
| Rule: ErrorRule, |
| Description: desc, |
| Outputs: outputs, |
| Optional: true, |
| Args: map[string]string{ |
| "error": err.Error(), |
| }, |
| }) |
| return |
| } |
| |
| func (a *androidModuleContext) Config() Config { |
| return a.ModuleContext.Config().(Config) |
| } |
| |
| func (a *androidModuleContext) ModuleBuild(pctx PackageContext, params ModuleBuildParams) { |
| a.Build(pctx, BuildParams(params)) |
| } |
| |
| func convertBuildParams(params BuildParams) blueprint.BuildParams { |
| bparams := blueprint.BuildParams{ |
| Rule: params.Rule, |
| Description: params.Description, |
| 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.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()) |
| } |
| |
| bparams.Outputs = proptools.NinjaEscapeList(bparams.Outputs) |
| bparams.ImplicitOutputs = proptools.NinjaEscapeList(bparams.ImplicitOutputs) |
| bparams.Inputs = proptools.NinjaEscapeList(bparams.Inputs) |
| bparams.Implicits = proptools.NinjaEscapeList(bparams.Implicits) |
| bparams.OrderOnly = proptools.NinjaEscapeList(bparams.OrderOnly) |
| bparams.Depfile = proptools.NinjaEscapeList([]string{bparams.Depfile})[0] |
| |
| return bparams |
| } |
| |
| func (a *androidModuleContext) Variable(pctx PackageContext, name, value string) { |
| if a.config.captureBuild { |
| a.variables[name] = value |
| } |
| |
| a.ModuleContext.Variable(pctx.PackageContext, name, value) |
| } |
| |
| func (a *androidModuleContext) Rule(pctx PackageContext, name string, params blueprint.RuleParams, |
| argNames ...string) blueprint.Rule { |
| |
| rule := a.ModuleContext.Rule(pctx.PackageContext, name, params, argNames...) |
| |
| if a.config.captureBuild { |
| a.ruleParams[rule] = params |
| } |
| |
| return rule |
| } |
| |
| func (a *androidModuleContext) Build(pctx PackageContext, params BuildParams) { |
| if a.config.captureBuild { |
| a.buildParams = append(a.buildParams, params) |
| } |
| |
| bparams := convertBuildParams(params) |
| |
| if bparams.Description != "" { |
| bparams.Description = "${moduleDesc}" + params.Description + "${moduleDescSuffix}" |
| } |
| |
| 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.PackageContext, bparams) |
| } |
| |
| func (a *androidModuleContext) GetMissingDependencies() []string { |
| return a.missingDeps |
| } |
| |
| func (a *androidModuleContext) AddMissingDependencies(deps []string) { |
| if deps != nil { |
| a.missingDeps = append(a.missingDeps, deps...) |
| a.missingDeps = FirstUniqueStrings(a.missingDeps) |
| } |
| } |
| |
| func (a *androidModuleContext) validateAndroidModule(module blueprint.Module) Module { |
| aModule, _ := module.(Module) |
| if aModule == nil { |
| a.ModuleErrorf("module %q not an android module", a.OtherModuleName(aModule)) |
| return nil |
| } |
| |
| if !aModule.Enabled() { |
| if a.Config().AllowMissingDependencies() { |
| a.AddMissingDependencies([]string{a.OtherModuleName(aModule)}) |
| } else { |
| a.ModuleErrorf("depends on disabled module %q", a.OtherModuleName(aModule)) |
| } |
| return nil |
| } |
| |
| return aModule |
| } |
| |
| func (a *androidModuleContext) getDirectDepInternal(name string, tag blueprint.DependencyTag) (blueprint.Module, blueprint.DependencyTag) { |
| type dep struct { |
| mod blueprint.Module |
| tag blueprint.DependencyTag |
| } |
| var deps []dep |
| a.VisitDirectDepsBlueprint(func(m blueprint.Module) { |
| if aModule, _ := m.(Module); aModule != nil && aModule.base().BaseModuleName() == name { |
| returnedTag := a.ModuleContext.OtherModuleDependencyTag(aModule) |
| if tag == nil || returnedTag == tag { |
| deps = append(deps, dep{aModule, returnedTag}) |
| } |
| } |
| }) |
| if len(deps) == 1 { |
| return deps[0].mod, deps[0].tag |
| } else if len(deps) >= 2 { |
| panic(fmt.Errorf("Multiple dependencies having same BaseModuleName() %q found from %q", |
| name, a.ModuleName())) |
| } else { |
| return nil, nil |
| } |
| } |
| |
| func (a *androidModuleContext) GetDirectDepsWithTag(tag blueprint.DependencyTag) []Module { |
| var deps []Module |
| a.VisitDirectDepsBlueprint(func(m blueprint.Module) { |
| if aModule, _ := m.(Module); aModule != nil { |
| if a.ModuleContext.OtherModuleDependencyTag(aModule) == tag { |
| deps = append(deps, aModule) |
| } |
| } |
| }) |
| return deps |
| } |
| |
| func (a *androidModuleContext) GetDirectDepWithTag(name string, tag blueprint.DependencyTag) blueprint.Module { |
| m, _ := a.getDirectDepInternal(name, tag) |
| return m |
| } |
| |
| func (a *androidModuleContext) GetDirectDep(name string) (blueprint.Module, blueprint.DependencyTag) { |
| return a.getDirectDepInternal(name, nil) |
| } |
| |
| func (a *androidModuleContext) VisitDirectDepsBlueprint(visit func(blueprint.Module)) { |
| a.ModuleContext.VisitDirectDeps(visit) |
| } |
| |
| func (a *androidModuleContext) VisitDirectDeps(visit func(Module)) { |
| a.ModuleContext.VisitDirectDeps(func(module blueprint.Module) { |
| if aModule := a.validateAndroidModule(module); aModule != nil { |
| visit(aModule) |
| } |
| }) |
| } |
| |
| func (a *androidModuleContext) VisitDirectDepsWithTag(tag blueprint.DependencyTag, visit func(Module)) { |
| a.ModuleContext.VisitDirectDeps(func(module blueprint.Module) { |
| if aModule := a.validateAndroidModule(module); aModule != nil { |
| if a.ModuleContext.OtherModuleDependencyTag(aModule) == tag { |
| visit(aModule) |
| } |
| } |
| }) |
| } |
| |
| func (a *androidModuleContext) VisitDirectDepsIf(pred func(Module) bool, visit func(Module)) { |
| a.ModuleContext.VisitDirectDepsIf( |
| // pred |
| func(module blueprint.Module) bool { |
| if aModule := a.validateAndroidModule(module); aModule != nil { |
| return pred(aModule) |
| } else { |
| return false |
| } |
| }, |
| // visit |
| func(module blueprint.Module) { |
| visit(module.(Module)) |
| }) |
| } |
| |
| func (a *androidModuleContext) VisitDepsDepthFirst(visit func(Module)) { |
| a.ModuleContext.VisitDepsDepthFirst(func(module blueprint.Module) { |
| if aModule := a.validateAndroidModule(module); aModule != nil { |
| visit(aModule) |
| } |
| }) |
| } |
| |
| func (a *androidModuleContext) VisitDepsDepthFirstIf(pred func(Module) bool, visit func(Module)) { |
| a.ModuleContext.VisitDepsDepthFirstIf( |
| // pred |
| func(module blueprint.Module) bool { |
| if aModule := a.validateAndroidModule(module); aModule != nil { |
| return pred(aModule) |
| } else { |
| return false |
| } |
| }, |
| // visit |
| func(module blueprint.Module) { |
| visit(module.(Module)) |
| }) |
| } |
| |
| func (a *androidModuleContext) WalkDepsBlueprint(visit func(blueprint.Module, blueprint.Module) bool) { |
| a.ModuleContext.WalkDeps(visit) |
| } |
| |
| func (a *androidModuleContext) WalkDeps(visit func(Module, Module) bool) { |
| a.ModuleContext.WalkDeps(func(child, parent blueprint.Module) bool { |
| childAndroidModule := a.validateAndroidModule(child) |
| parentAndroidModule := a.validateAndroidModule(parent) |
| if childAndroidModule != nil && parentAndroidModule != nil { |
| return visit(childAndroidModule, parentAndroidModule) |
| } else { |
| return false |
| } |
| }) |
| } |
| |
| func (a *androidModuleContext) VisitAllModuleVariants(visit func(Module)) { |
| a.ModuleContext.VisitAllModuleVariants(func(module blueprint.Module) { |
| visit(module.(Module)) |
| }) |
| } |
| |
| func (a *androidModuleContext) PrimaryModule() Module { |
| return a.ModuleContext.PrimaryModule().(Module) |
| } |
| |
| func (a *androidModuleContext) FinalModule() Module { |
| return a.ModuleContext.FinalModule().(Module) |
| } |
| |
| func (a *androidBaseContextImpl) Target() Target { |
| return a.target |
| } |
| |
| func (a *androidBaseContextImpl) TargetPrimary() bool { |
| return a.targetPrimary |
| } |
| |
| func (a *androidBaseContextImpl) MultiTargets() []Target { |
| return a.multiTargets |
| } |
| |
| 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) Fuchsia() bool { |
| return a.target.Os == Fuchsia |
| } |
| |
| 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]) <= 1 { |
| return true |
| } |
| return a.target.Arch.ArchType == a.config.Targets[a.target.Os][0].Arch.ArchType |
| } |
| |
| func (a *androidBaseContextImpl) AConfig() Config { |
| return a.config |
| } |
| |
| func (a *androidBaseContextImpl) DeviceConfig() DeviceConfig { |
| return DeviceConfig{a.config.deviceConfig} |
| } |
| |
| func (a *androidBaseContextImpl) Platform() bool { |
| return a.kind == platformModule |
| } |
| |
| func (a *androidBaseContextImpl) DeviceSpecific() bool { |
| return a.kind == deviceSpecificModule |
| } |
| |
| func (a *androidBaseContextImpl) SocSpecific() bool { |
| return a.kind == socSpecificModule |
| } |
| |
| func (a *androidBaseContextImpl) ProductSpecific() bool { |
| return a.kind == productSpecificModule |
| } |
| |
| func (a *androidBaseContextImpl) ProductServicesSpecific() bool { |
| return a.kind == productServicesSpecificModule |
| } |
| |
| // Makes this module a platform module, i.e. not specific to soc, device, |
| // product, or product_services. |
| func (a *ModuleBase) MakeAsPlatform() { |
| a.commonProperties.Vendor = boolPtr(false) |
| a.commonProperties.Proprietary = boolPtr(false) |
| a.commonProperties.Soc_specific = boolPtr(false) |
| a.commonProperties.Product_specific = boolPtr(false) |
| a.commonProperties.Product_services_specific = boolPtr(false) |
| } |
| |
| func (a *androidModuleContext) InstallInData() bool { |
| return a.module.InstallInData() |
| } |
| |
| func (a *androidModuleContext) InstallInSanitizerDir() bool { |
| return a.module.InstallInSanitizerDir() |
| } |
| |
| func (a *androidModuleContext) InstallInRecovery() bool { |
| return a.module.InstallInRecovery() |
| } |
| |
| func (a *androidModuleContext) skipInstall(fullInstallPath OutputPath) bool { |
| if a.module.base().commonProperties.SkipInstall { |
| return true |
| } |
| |
| // We'll need a solution for choosing which of modules with the same name in different |
| // namespaces to install. For now, reuse the list of namespaces exported to Make as the |
| // list of namespaces to install in a Soong-only build. |
| if !a.module.base().commonProperties.NamespaceExportedToMake { |
| return true |
| } |
| |
| if a.Device() { |
| if a.Config().SkipDeviceInstall() { |
| return true |
| } |
| |
| if a.Config().SkipMegaDeviceInstall(fullInstallPath.String()) { |
| return true |
| } |
| } |
| |
| return false |
| } |
| |
| func (a *androidModuleContext) InstallFile(installPath OutputPath, name string, srcPath Path, |
| deps ...Path) OutputPath { |
| return a.installFile(installPath, name, srcPath, Cp, deps) |
| } |
| |
| func (a *androidModuleContext) InstallExecutable(installPath OutputPath, name string, srcPath Path, |
| deps ...Path) OutputPath { |
| return a.installFile(installPath, name, srcPath, CpExecutable, deps) |
| } |
| |
| func (a *androidModuleContext) installFile(installPath OutputPath, name string, srcPath Path, |
| rule blueprint.Rule, 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.Build(pctx, BuildParams{ |
| Rule: rule, |
| Description: "install " + fullInstallPath.Base(), |
| Output: fullInstallPath, |
| Input: srcPath, |
| Implicits: implicitDeps, |
| OrderOnly: orderOnlyDeps, |
| Default: !a.Config().EmbeddedInMake(), |
| }) |
| |
| a.installFiles = append(a.installFiles, fullInstallPath) |
| } |
| a.checkbuildFiles = append(a.checkbuildFiles, srcPath) |
| return fullInstallPath |
| } |
| |
| 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) { |
| |
| relPath, err := filepath.Rel(path.Dir(fullInstallPath.String()), srcPath.String()) |
| if err != nil { |
| panic(fmt.Sprintf("Unable to generate symlink between %q and %q: %s", fullInstallPath.Base(), srcPath.Base(), err)) |
| } |
| a.Build(pctx, BuildParams{ |
| Rule: Symlink, |
| Description: "install symlink " + fullInstallPath.Base(), |
| Output: fullInstallPath, |
| OrderOnly: Paths{srcPath}, |
| Default: !a.Config().EmbeddedInMake(), |
| Args: map[string]string{ |
| "fromPath": relPath, |
| }, |
| }) |
| |
| a.installFiles = append(a.installFiles, fullInstallPath) |
| a.checkbuildFiles = append(a.checkbuildFiles, srcPath) |
| } |
| return fullInstallPath |
| } |
| |
| // installPath/name -> absPath where absPath might be a path that is available only at runtime |
| // (e.g. /apex/...) |
| func (a *androidModuleContext) InstallAbsoluteSymlink(installPath OutputPath, name string, absPath string) OutputPath { |
| fullInstallPath := installPath.Join(a, name) |
| a.module.base().hooks.runInstallHooks(a, fullInstallPath, true) |
| |
| if !a.skipInstall(fullInstallPath) { |
| a.Build(pctx, BuildParams{ |
| Rule: Symlink, |
| Description: "install symlink " + fullInstallPath.Base() + " -> " + absPath, |
| Output: fullInstallPath, |
| Default: !a.Config().EmbeddedInMake(), |
| Args: map[string]string{ |
| "fromPath": absPath, |
| }, |
| }) |
| |
| a.installFiles = append(a.installFiles, fullInstallPath) |
| } |
| 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 |
| |
| // Adds necessary dependencies to satisfy filegroup or generated sources modules listed in srcFiles |
| // using ":module" syntax, if any. |
| // |
| // Deprecated: tag the property with `android:"path"` instead. |
| 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...) |
| } |
| |
| // Adds necessary dependencies to satisfy filegroup or generated sources modules specified in s |
| // using ":module" syntax, if any. |
| // |
| // Deprecated: tag the property with `android:"path"` instead. |
| func ExtractSourceDeps(ctx BottomUpMutatorContext, s *string) { |
| if s != nil { |
| if m := SrcIsModule(*s); m != "" { |
| ctx.AddDependency(ctx.Module(), SourceDepTag, m) |
| } |
| } |
| } |
| |
| type SourceFileProducer interface { |
| Srcs() Paths |
| } |
| |
| type HostToolProvider interface { |
| HostToolPath() OptionalPath |
| } |
| |
| // Returns a list of paths expanded from globs and modules referenced using ":module" syntax. The property must |
| // be tagged with `android:"path" to support automatic source module dependency resolution. |
| // |
| // Deprecated: use PathsForModuleSrc or PathsForModuleSrcExcludes instead. |
| func (ctx *androidModuleContext) ExpandSources(srcFiles, excludes []string) Paths { |
| return PathsForModuleSrcExcludes(ctx, srcFiles, excludes) |
| } |
| |
| // Returns a single path expanded from globs and modules referenced using ":module" syntax. The property must |
| // be tagged with `android:"path" to support automatic source module dependency resolution. |
| // |
| // Deprecated: use PathForModuleSrc instead. |
| func (ctx *androidModuleContext) ExpandSource(srcFile, prop string) Path { |
| return PathForModuleSrc(ctx, srcFile) |
| } |
| |
| // Returns an optional single path expanded from globs and modules referenced using ":module" syntax if |
| // the srcFile is non-nil. The property must be tagged with `android:"path" to support automatic source module |
| // dependency resolution. |
| func (ctx *androidModuleContext) ExpandOptionalSource(srcFile *string, prop string) OptionalPath { |
| if srcFile != nil { |
| return OptionalPathForPath(PathForModuleSrc(ctx, *srcFile)) |
| } |
| return OptionalPath{} |
| } |
| |
| func (ctx *androidModuleContext) RequiredModuleNames() []string { |
| return ctx.module.base().commonProperties.Required |
| } |
| |
| func (ctx *androidModuleContext) HostRequiredModuleNames() []string { |
| return ctx.module.base().commonProperties.Host_required |
| } |
| |
| func (ctx *androidModuleContext) TargetRequiredModuleNames() []string { |
| return ctx.module.base().commonProperties.Target_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, true) |
| } |
| |
| func (ctx *androidModuleContext) GlobFiles(globPattern string, excludes []string) Paths { |
| ret, err := ctx.GlobWithDeps(globPattern, excludes) |
| if err != nil { |
| ctx.ModuleErrorf("glob: %s", err.Error()) |
| } |
| return pathsForModuleSrcFromFullPath(ctx, ret, false) |
| } |
| |
| func init() { |
| RegisterSingletonType("buildtarget", BuildTargetSingleton) |
| } |
| |
| func BuildTargetSingleton() Singleton { |
| return &buildTargetSingleton{} |
| } |
| |
| func parentDir(dir string) string { |
| dir, _ = filepath.Split(dir) |
| return filepath.Clean(dir) |
| } |
| |
| type buildTargetSingleton struct{} |
| |
| func (c *buildTargetSingleton) GenerateBuildActions(ctx SingletonContext) { |
| var checkbuildDeps Paths |
| |
| mmTarget := func(dir string) WritablePath { |
| return PathForPhony(ctx, |
| "MODULES-IN-"+strings.Replace(filepath.Clean(dir), "/", "-", -1)) |
| } |
| |
| modulesInDir := make(map[string]Paths) |
| |
| ctx.VisitAllModules(func(module Module) { |
| blueprintDir := module.base().blueprintDir |
| installTarget := module.base().installTarget |
| checkbuildTarget := module.base().checkbuildTarget |
| |
| if checkbuildTarget != nil { |
| checkbuildDeps = append(checkbuildDeps, checkbuildTarget) |
| modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], checkbuildTarget) |
| } |
| |
| if installTarget != nil { |
| modulesInDir[blueprintDir] = append(modulesInDir[blueprintDir], installTarget) |
| } |
| }) |
| |
| suffix := "" |
| if ctx.Config().EmbeddedInMake() { |
| suffix = "-soong" |
| } |
| |
| // Create a top-level checkbuild target that depends on all modules |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Output: PathForPhony(ctx, "checkbuild"+suffix), |
| Implicits: checkbuildDeps, |
| }) |
| |
| // Make will generate the MODULES-IN-* targets |
| if ctx.Config().EmbeddedInMake() { |
| return |
| } |
| |
| sortedKeys := func(m map[string]Paths) []string { |
| s := make([]string, 0, len(m)) |
| for k := range m { |
| s = append(s, k) |
| } |
| sort.Strings(s) |
| return s |
| } |
| |
| // 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 MODULES-IN-<directory> target that depends on all modules in a directory, and |
| // depends on the MODULES-IN-* targets of all of its subdirectories that contain Android.bp |
| // files. |
| for _, dir := range dirs { |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Output: mmTarget(dir), |
| Implicits: modulesInDir[dir], |
| // HACK: checkbuild should be an optional build, but force it |
| // enabled for now in standalone builds |
| Default: !ctx.Config().EmbeddedInMake(), |
| }) |
| } |
| |
| // Create (host|host-cross|target)-<OS> phony rules to build a reduced checkbuild. |
| osDeps := map[OsType]Paths{} |
| ctx.VisitAllModules(func(module Module) { |
| if module.Enabled() { |
| os := module.Target().Os |
| osDeps[os] = append(osDeps[os], module.base().checkbuildFiles...) |
| } |
| }) |
| |
| osClass := make(map[string]Paths) |
| for os, deps := range osDeps { |
| var className string |
| |
| switch os.Class { |
| case Host: |
| className = "host" |
| case HostCross: |
| className = "host-cross" |
| case Device: |
| className = "target" |
| default: |
| continue |
| } |
| |
| name := PathForPhony(ctx, className+"-"+os.Name) |
| osClass[className] = append(osClass[className], name) |
| |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Output: name, |
| Implicits: deps, |
| }) |
| } |
| |
| // Wrap those into host|host-cross|target phony rules |
| osClasses := sortedKeys(osClass) |
| for _, class := range osClasses { |
| ctx.Build(pctx, BuildParams{ |
| Rule: blueprint.Phony, |
| Output: PathForPhony(ctx, class), |
| Implicits: osClass[class], |
| }) |
| } |
| } |
| |
| // Collect information for opening IDE project files in java/jdeps.go. |
| type IDEInfo interface { |
| IDEInfo(ideInfo *IdeInfo) |
| BaseModuleName() string |
| } |
| |
| // Extract the base module name from the Import name. |
| // Often the Import name has a prefix "prebuilt_". |
| // Remove the prefix explicitly if needed |
| // until we find a better solution to get the Import name. |
| type IDECustomizedModuleName interface { |
| IDECustomizedModuleName() string |
| } |
| |
| type IdeInfo struct { |
| Deps []string `json:"dependencies,omitempty"` |
| Srcs []string `json:"srcs,omitempty"` |
| Aidl_include_dirs []string `json:"aidl_include_dirs,omitempty"` |
| Jarjar_rules []string `json:"jarjar_rules,omitempty"` |
| Jars []string `json:"jars,omitempty"` |
| Classes []string `json:"class,omitempty"` |
| Installed_paths []string `json:"installed,omitempty"` |
| } |