| // 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" |
| "sort" |
| "strings" |
| "text/scanner" |
| |
| "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 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 |
| Arch() Arch |
| Os() OsType |
| Host() bool |
| Device() bool |
| Darwin() bool |
| Windows() bool |
| Debug() bool |
| PrimaryArch() bool |
| Platform() bool |
| DeviceSpecific() bool |
| SocSpecific() bool |
| ProductSpecific() 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 |
| 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 |
| ExpandSourcesSubDir(srcFiles, excludes []string, subDir string) Paths |
| 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 |
| CheckbuildFile(srcPath Path) |
| |
| AddMissingDependencies(deps []string) |
| |
| InstallInData() bool |
| InstallInSanitizerDir() bool |
| InstallInRecovery() bool |
| |
| RequiredModuleNames() []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 |
| |
| 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)) |
| VisitDepsDepthFirst(visit func(Module)) |
| VisitDepsDepthFirstIf(pred func(Module) bool, visit func(Module)) |
| WalkDeps(visit func(Module, 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 |
| |
| AddProperties(props ...interface{}) |
| GetProperties() []interface{} |
| |
| BuildParamsForTests() []BuildParams |
| } |
| |
| 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"` |
| |
| // 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 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 is installed to recovery partition |
| Recovery *bool |
| |
| // init.rc files to be installed if this module is installed |
| Init_rc []string |
| |
| // VINTF manifest fragments to be installed if this module is installed |
| Vintf_fragments []string |
| |
| // names of other modules to install if this module is installed |
| Required []string `android:"arch_variant"` |
| |
| // relative path to a file to include in the list of notices for the device |
| Notice *string |
| |
| // 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"` |
| |
| NamespaceExportedToMake 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" |
| MultilibCommonFirst Multilib = "common_first" |
| MultilibDefault Multilib = "" |
| ) |
| |
| type HostOrDeviceSupported int |
| |
| const ( |
| _ HostOrDeviceSupported = iota |
| HostSupported |
| HostSupportedNoCross |
| DeviceSupported |
| HostAndDeviceSupported |
| HostAndDeviceDefault |
| NeitherHostNorDeviceSupported |
| ) |
| |
| type moduleKind int |
| |
| const ( |
| platformModule moduleKind = iota |
| deviceSpecificModule |
| socSpecificModule |
| productSpecificModule |
| ) |
| |
| 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" |
| 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.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 |
| |
| 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 WritablePath |
| checkbuildTarget WritablePath |
| blueprintDir string |
| |
| hooks hooks |
| |
| registerProps []interface{} |
| |
| // For tests |
| buildParams []BuildParams |
| } |
| |
| 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 |
| } |
| |
| // 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, 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) Platform() bool { |
| return !a.DeviceSpecific() && !a.SocSpecific() && !a.ProductSpecific() |
| } |
| |
| 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) 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{} |
| 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) 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) |
| |
| if ((socSpecific || deviceSpecific) && productSpecific) || (socSpecific && deviceSpecific) { |
| msg := "conflicting value set here" |
| if productSpecific { |
| ctx.PropertyErrorf("product_specific", "a module cannot be specific to SoC or device and product at the same time.") |
| if deviceSpecific { |
| ctx.PropertyErrorf("device_specific", msg) |
| } |
| } else { |
| 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 { |
| return productSpecificModule |
| } 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, |
| 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(), |
| } |
| |
| 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) |
| |
| 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...) |
| } |
| |
| if a == ctx.FinalModule().(Module).base() { |
| a.generateModuleTarget(ctx) |
| if ctx.Failed() { |
| return |
| } |
| } |
| |
| a.buildParams = ctx.buildParams |
| } |
| |
| type androidBaseContextImpl struct { |
| target 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 |
| } |
| |
| 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()) |
| } |
| |
| return bparams |
| } |
| |
| func (a *androidModuleContext) Variable(pctx PackageContext, name, value string) { |
| a.ModuleContext.Variable(pctx.PackageContext, name, value) |
| } |
| |
| func (a *androidModuleContext) Rule(pctx PackageContext, name string, params blueprint.RuleParams, |
| argNames ...string) blueprint.Rule { |
| |
| return a.ModuleContext.Rule(pctx.PackageContext, name, params, argNames...) |
| } |
| |
| 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) 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) 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) 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) 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 *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) { |
| |
| 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": 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 |
| |
| // Adds necessary dependencies to satisfy filegroup or generated sources modules listed in srcFiles |
| // using ":module" syntax, if any. |
| 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. |
| 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 |
| } |
| |
| // 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, "") |
| } |
| |
| // Returns a single path expanded from globs and modules referenced using ":module" syntax. |
| // ExtractSourceDeps must have already been called during the dependency resolution phase. |
| func (ctx *androidModuleContext) ExpandSource(srcFile, prop string) Path { |
| srcFiles := ctx.ExpandSourcesSubDir([]string{srcFile}, nil, "") |
| if len(srcFiles) == 1 { |
| return srcFiles[0] |
| } else { |
| ctx.PropertyErrorf(prop, "module providing %s must produce exactly one file", prop) |
| return nil |
| } |
| } |
| |
| // Returns an optional single path expanded from globs and modules referenced using ":module" syntax if |
| // the srcFile is non-nil. |
| // ExtractSourceDeps must have already been called during the dependency resolution phase. |
| func (ctx *androidModuleContext) ExpandOptionalSource(srcFile *string, prop string) OptionalPath { |
| if srcFile != nil { |
| return OptionalPathForPath(ctx.ExpandSource(*srcFile, prop)) |
| } |
| return OptionalPath{} |
| } |
| |
| func (ctx *androidModuleContext) ExpandSourcesSubDir(srcFiles, excludes []string, subDir string) Paths { |
| prefix := PathForModuleSrc(ctx).String() |
| |
| var expandedExcludes []string |
| if excludes != nil { |
| expandedExcludes = make([]string, 0, len(excludes)) |
| } |
| |
| for _, e := range excludes { |
| if m := SrcIsModule(e); m != "" { |
| module := ctx.GetDirectDepWithTag(m, SourceDepTag) |
| if module == nil { |
| // Error will have been handled by ExtractSourcesDeps |
| continue |
| } |
| if srcProducer, ok := module.(SourceFileProducer); ok { |
| expandedExcludes = append(expandedExcludes, srcProducer.Srcs().Strings()...) |
| } else { |
| ctx.ModuleErrorf("srcs dependency %q is not a source file producing module", m) |
| } |
| } else { |
| expandedExcludes = append(expandedExcludes, 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 module == nil { |
| // Error will have been handled by ExtractSourcesDeps |
| continue |
| } |
| if srcProducer, ok := module.(SourceFileProducer); ok { |
| moduleSrcs := srcProducer.Srcs() |
| for _, e := range expandedExcludes { |
| for j, ms := range moduleSrcs { |
| if ms.String() == e { |
| moduleSrcs = append(moduleSrcs[:j], moduleSrcs[j+1:]...) |
| } |
| } |
| } |
| expandedSrcFiles = append(expandedSrcFiles, moduleSrcs...) |
| } else { |
| ctx.ModuleErrorf("srcs dependency %q is not a source file producing module", m) |
| } |
| } else if pathtools.IsGlob(s) { |
| globbedSrcFiles := ctx.GlobFiles(filepath.Join(prefix, s), expandedExcludes) |
| for i, s := range globbedSrcFiles { |
| globbedSrcFiles[i] = s.(ModuleSrcPath).WithSubDir(ctx, subDir) |
| } |
| expandedSrcFiles = append(expandedSrcFiles, globbedSrcFiles...) |
| } else { |
| p := PathForModuleSrc(ctx, s).WithSubDir(ctx, subDir) |
| j := findStringInSlice(p.String(), expandedExcludes) |
| if j == -1 { |
| expandedSrcFiles = append(expandedSrcFiles, p) |
| } |
| |
| } |
| } |
| 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, 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], |
| }) |
| } |
| } |
| |
| 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] } |