a866713ddb
Some minimal branches have v/g_s/build/release (where apex_contibutions have been set to mainline prebuilts), but not v/g/b (where the apex_contributions for prebuilts have been defined). These minimal branches are unsuitable for building a product that consume mainline prebuilts, but they would still like to do aosp product builds. aosp products should not use the mainline prebuilts anyways, but this has been implemented as - always create the dependency edge to the selected apex contributions - do not visit the dependency edge subsequently if IgnoreApexContributions is set set to true To support aosp product builds in minimal branches, this CL updates the implementation to skip creating the dependency edge when IgnoreApexContributions is set to true Test: go test ./android Change-Id: Iaa0971760e64f9b7a03542f179231ce2268b6616
194 lines
6.2 KiB
Go
194 lines
6.2 KiB
Go
// Copyright 2023 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 (
|
|
"github.com/google/blueprint"
|
|
"github.com/google/blueprint/proptools"
|
|
)
|
|
|
|
func init() {
|
|
RegisterApexContributionsBuildComponents(InitRegistrationContext)
|
|
}
|
|
|
|
func RegisterApexContributionsBuildComponents(ctx RegistrationContext) {
|
|
ctx.RegisterModuleType("apex_contributions", apexContributionsFactory)
|
|
ctx.RegisterModuleType("apex_contributions_defaults", apexContributionsDefaultsFactory)
|
|
ctx.RegisterSingletonModuleType("all_apex_contributions", allApexContributionsFactory)
|
|
}
|
|
|
|
type apexContributions struct {
|
|
ModuleBase
|
|
DefaultableModuleBase
|
|
properties contributionProps
|
|
}
|
|
|
|
type contributionProps struct {
|
|
// Name of the mainline module
|
|
Api_domain *string
|
|
// A list of module names that should be used when this contribution
|
|
// is selected via product_config
|
|
// The name should be explicit (foo or prebuilt_foo)
|
|
Contents []string
|
|
}
|
|
|
|
func (m *apexContributions) ApiDomain() string {
|
|
return proptools.String(m.properties.Api_domain)
|
|
}
|
|
|
|
func (m *apexContributions) Contents() []string {
|
|
return m.properties.Contents
|
|
}
|
|
|
|
// apex_contributions contains a list of module names (source or
|
|
// prebuilt) belonging to the mainline module
|
|
// An apex can have multiple apex_contributions modules
|
|
// with different combinations of source or prebuilts, but only one can be
|
|
// selected via product_config.
|
|
func apexContributionsFactory() Module {
|
|
module := &apexContributions{}
|
|
module.AddProperties(&module.properties)
|
|
InitAndroidModule(module)
|
|
InitDefaultableModule(module)
|
|
return module
|
|
}
|
|
|
|
// This module type does not have any build actions.
|
|
// It provides metadata that is used in post-deps mutator phase for source vs
|
|
// prebuilts selection.
|
|
func (m *apexContributions) GenerateAndroidBuildActions(ctx ModuleContext) {
|
|
}
|
|
|
|
type apexContributionsDefaults struct {
|
|
ModuleBase
|
|
DefaultsModuleBase
|
|
}
|
|
|
|
func apexContributionsDefaultsFactory() Module {
|
|
module := &apexContributionsDefaults{}
|
|
module.AddProperties(&contributionProps{})
|
|
InitDefaultsModule(module)
|
|
return module
|
|
}
|
|
|
|
// A container for apex_contributions.
|
|
// Based on product_config, it will create a dependency on the selected
|
|
// apex_contributions per mainline module
|
|
type allApexContributions struct {
|
|
SingletonModuleBase
|
|
}
|
|
|
|
func allApexContributionsFactory() SingletonModule {
|
|
module := &allApexContributions{}
|
|
InitAndroidModule(module)
|
|
return module
|
|
}
|
|
|
|
type apexContributionsDepTag struct {
|
|
blueprint.BaseDependencyTag
|
|
}
|
|
|
|
var (
|
|
AcDepTag = apexContributionsDepTag{}
|
|
)
|
|
|
|
// Creates a dep to each selected apex_contributions
|
|
func (a *allApexContributions) DepsMutator(ctx BottomUpMutatorContext) {
|
|
// Skip apex_contributions if BuildApexContributionContents is true
|
|
// This product config var allows some products in the same family to use mainline modules from source
|
|
// (e.g. shiba and shiba_fullmte)
|
|
// Eventually these product variants will have their own release config maps.
|
|
if !proptools.Bool(ctx.Config().BuildIgnoreApexContributionContents()) {
|
|
ctx.AddDependency(ctx.Module(), AcDepTag, ctx.Config().AllApexContributions()...)
|
|
}
|
|
}
|
|
|
|
// Set PrebuiltSelectionInfoProvider in post deps phase
|
|
func (a *allApexContributions) SetPrebuiltSelectionInfoProvider(ctx BaseModuleContext) {
|
|
addContentsToProvider := func(p *PrebuiltSelectionInfoMap, m *apexContributions) {
|
|
for _, content := range m.Contents() {
|
|
if !ctx.OtherModuleExists(content) && !ctx.Config().AllowMissingDependencies() {
|
|
ctx.ModuleErrorf("%s listed in apex_contributions %s does not exist\n", content, m.Name())
|
|
}
|
|
pi := &PrebuiltSelectionInfo{
|
|
selectedModuleName: content,
|
|
metadataModuleName: m.Name(),
|
|
apiDomain: m.ApiDomain(),
|
|
}
|
|
p.Add(ctx, pi)
|
|
}
|
|
}
|
|
|
|
p := PrebuiltSelectionInfoMap{}
|
|
ctx.VisitDirectDepsWithTag(AcDepTag, func(child Module) {
|
|
if m, ok := child.(*apexContributions); ok {
|
|
addContentsToProvider(&p, m)
|
|
} else {
|
|
ctx.ModuleErrorf("%s is not an apex_contributions module\n", child.Name())
|
|
}
|
|
})
|
|
SetProvider(ctx, PrebuiltSelectionInfoProvider, p)
|
|
}
|
|
|
|
// A provider containing metadata about whether source or prebuilt should be used
|
|
// This provider will be used in prebuilt_select mutator to redirect deps
|
|
var PrebuiltSelectionInfoProvider = blueprint.NewMutatorProvider[PrebuiltSelectionInfoMap]("prebuilt_select")
|
|
|
|
// Map of selected module names to a metadata object
|
|
// The metadata contains information about the api_domain of the selected module
|
|
type PrebuiltSelectionInfoMap map[string]PrebuiltSelectionInfo
|
|
|
|
// Add a new entry to the map with some validations
|
|
func (pm *PrebuiltSelectionInfoMap) Add(ctx BaseModuleContext, p *PrebuiltSelectionInfo) {
|
|
if p == nil {
|
|
return
|
|
}
|
|
(*pm)[p.selectedModuleName] = *p
|
|
}
|
|
|
|
type PrebuiltSelectionInfo struct {
|
|
// e.g. (libc|prebuilt_libc)
|
|
selectedModuleName string
|
|
// Name of the apex_contributions module
|
|
metadataModuleName string
|
|
// e.g. com.android.runtime
|
|
apiDomain string
|
|
}
|
|
|
|
// Returns true if `name` is explicitly requested using one of the selected
|
|
// apex_contributions metadata modules.
|
|
func (p *PrebuiltSelectionInfoMap) IsSelected(name string) bool {
|
|
_, exists := (*p)[name]
|
|
return exists
|
|
}
|
|
|
|
// Return the list of soong modules selected for this api domain
|
|
// In the case of apexes, it is the canonical name of the apex on device (/apex/<apex_name>)
|
|
func (p *PrebuiltSelectionInfoMap) GetSelectedModulesForApiDomain(apiDomain string) []string {
|
|
selected := []string{}
|
|
for _, entry := range *p {
|
|
if entry.apiDomain == apiDomain {
|
|
selected = append(selected, entry.selectedModuleName)
|
|
}
|
|
}
|
|
return selected
|
|
}
|
|
|
|
// This module type does not have any build actions.
|
|
func (a *allApexContributions) GenerateAndroidBuildActions(ctx ModuleContext) {
|
|
}
|
|
|
|
func (a *allApexContributions) GenerateSingletonBuildActions(ctx SingletonContext) {
|
|
}
|