Merge "Add support to sdk/module_exports to specify required traits" am: 95a1d1672f

Original change: https://android-review.googlesource.com/c/platform/build/soong/+/1827876

Change-Id: I459f5843376c4ec281b6bbced07e6b60657c4070
This commit is contained in:
Paul Duffin
2021-09-23 13:03:13 +00:00
committed by Automerger Merge Worker
6 changed files with 707 additions and 3 deletions

View File

@@ -15,6 +15,7 @@
package android
import (
"fmt"
"sort"
"strings"
@@ -376,6 +377,175 @@ func (b BpPrintableBase) bpPrintable() {
var _ BpPrintable = BpPrintableBase{}
// SdkMemberTrait represents a trait that members of an sdk module can contribute to the sdk
// snapshot.
//
// A trait is simply a characteristic of sdk member that is not required by default which may be
// required for some members but not others. Traits can cause additional information to be output
// to the sdk snapshot or replace the default information exported for a member with something else.
// e.g.
// * By default cc libraries only export the default image variants to the SDK. However, for some
// members it may be necessary to export specific image variants, e.g. vendor, or recovery.
// * By default cc libraries export all the configured architecture variants except for the native
// bridge architecture variants. However, for some members it may be necessary to export the
// native bridge architecture variants as well.
// * By default cc libraries export the platform variant (i.e. sdk:). However, for some members it
// may be necessary to export the sdk variant (i.e. sdk:sdk).
//
// A sdk can request a module to provide no traits, one trait or a collection of traits. The exact
// behavior of a trait is determined by how SdkMemberType implementations handle the traits. A trait
// could be specific to one SdkMemberType or many. Some trait combinations could be incompatible.
//
// The sdk module type will create a special traits structure that contains a property for each
// trait registered with RegisterSdkMemberTrait(). The property names are those returned from
// SdkPropertyName(). Each property contains a list of modules that are required to have that trait.
// e.g. something like this:
//
// sdk {
// name: "sdk",
// ...
// traits: {
// recovery_image: ["module1", "module4", "module5"],
// native_bridge: ["module1", "module2"],
// native_sdk: ["module1", "module3"],
// ...
// },
// ...
// }
type SdkMemberTrait interface {
// SdkPropertyName returns the name of the traits property on an sdk module.
SdkPropertyName() string
}
// SdkMemberTraitBase is the base struct that must be embedded within any type that implements
// SdkMemberTrait.
type SdkMemberTraitBase struct {
// PropertyName is the name of the property
PropertyName string
}
func (b *SdkMemberTraitBase) SdkPropertyName() string {
return b.PropertyName
}
// SdkMemberTraitSet is a set of SdkMemberTrait instances.
type SdkMemberTraitSet interface {
// Empty returns true if this set is empty.
Empty() bool
// Contains returns true if this set contains the specified trait.
Contains(trait SdkMemberTrait) bool
// Subtract returns a new set containing all elements of this set except for those in the
// other set.
Subtract(other SdkMemberTraitSet) SdkMemberTraitSet
// String returns a string representation of the set and its contents.
String() string
}
func NewSdkMemberTraitSet(traits []SdkMemberTrait) SdkMemberTraitSet {
if len(traits) == 0 {
return EmptySdkMemberTraitSet()
}
m := sdkMemberTraitSet{}
for _, trait := range traits {
m[trait] = true
}
return m
}
func EmptySdkMemberTraitSet() SdkMemberTraitSet {
return (sdkMemberTraitSet)(nil)
}
type sdkMemberTraitSet map[SdkMemberTrait]bool
var _ SdkMemberTraitSet = (sdkMemberTraitSet{})
func (s sdkMemberTraitSet) Empty() bool {
return len(s) == 0
}
func (s sdkMemberTraitSet) Contains(trait SdkMemberTrait) bool {
return s[trait]
}
func (s sdkMemberTraitSet) Subtract(other SdkMemberTraitSet) SdkMemberTraitSet {
if other.Empty() {
return s
}
var remainder []SdkMemberTrait
for trait, _ := range s {
if !other.Contains(trait) {
remainder = append(remainder, trait)
}
}
return NewSdkMemberTraitSet(remainder)
}
func (s sdkMemberTraitSet) String() string {
list := []string{}
for trait, _ := range s {
list = append(list, trait.SdkPropertyName())
}
sort.Strings(list)
return fmt.Sprintf("[%s]", strings.Join(list, ","))
}
// SdkMemberTraitsRegistry is a registry of SdkMemberTrait instances.
type SdkMemberTraitsRegistry struct {
// The list of traits sorted by property name.
list []SdkMemberTrait
}
// copyAndAppend creates a new SdkMemberTraitsRegistry that includes all the traits registered in
// this registry plus the supplied trait.
func (r *SdkMemberTraitsRegistry) copyAndAppend(trait SdkMemberTrait) *SdkMemberTraitsRegistry {
oldList := r.list
// Copy the slice just in case this is being read while being modified, e.g. when testing.
list := make([]SdkMemberTrait, 0, len(oldList)+1)
list = append(list, oldList...)
list = append(list, trait)
// Sort the member types by their property name to ensure that registry order has no effect
// on behavior.
sort.Slice(list, func(i1, i2 int) bool {
t1 := list[i1]
t2 := list[i2]
return t1.SdkPropertyName() < t2.SdkPropertyName()
})
// Create a new registry so the pointer uniquely identifies the set of registered types.
return &SdkMemberTraitsRegistry{
list: list,
}
}
// RegisteredTraits returns the list of registered SdkMemberTrait instances.
func (r *SdkMemberTraitsRegistry) RegisteredTraits() []SdkMemberTrait {
return r.list
}
// UniqueOnceKey returns a key to use with Config.Once that uniquely identifies this instance.
func (r *SdkMemberTraitsRegistry) UniqueOnceKey() OnceKey {
// Use the pointer to the registry as the unique key.
return NewCustomOnceKey(r)
}
var RegisteredSdkMemberTraits = &SdkMemberTraitsRegistry{}
// RegisterSdkMemberTrait registers an SdkMemberTrait object to allow them to be used in the
// module_exports, module_exports_snapshot, sdk and sdk_snapshot module types.
func RegisterSdkMemberTrait(trait SdkMemberTrait) {
RegisteredSdkMemberTraits = RegisteredSdkMemberTraits.copyAndAppend(trait)
}
// SdkMember is an individual member of the SDK.
//
// It includes all of the variants that the SDK depends upon.
@@ -541,12 +711,23 @@ type SdkMemberType interface {
// CreateVariantPropertiesStruct creates a structure into which variant specific properties can be
// added.
CreateVariantPropertiesStruct() SdkMemberProperties
// SupportedTraits returns the set of traits supported by this member type.
SupportedTraits() SdkMemberTraitSet
}
// SdkDependencyContext provides access to information needed by the SdkMemberType.AddDependencies()
// implementations.
type SdkDependencyContext interface {
BottomUpMutatorContext
// RequiredTraits returns the set of SdkMemberTrait instances that the sdk requires the named
// member to provide.
RequiredTraits(name string) SdkMemberTraitSet
// RequiresTrait returns true if the sdk requires the member with the supplied name to provide the
// supplied trait.
RequiresTrait(name string, trait SdkMemberTrait) bool
}
// SdkMemberTypeBase is the base type for SdkMemberType implementations and must be embedded in any
@@ -565,6 +746,9 @@ type SdkMemberTypeBase struct {
// module type in its SdkMemberType.AddPrebuiltModule() method. That prevents the sdk snapshot
// code from automatically adding a prefer: true flag.
UseSourceModuleTypeInSnapshot bool
// The list of supported traits.
Traits []SdkMemberTrait
}
func (b *SdkMemberTypeBase) SdkPropertyName() string {
@@ -587,6 +771,10 @@ func (b *SdkMemberTypeBase) UsesSourceModuleTypeInSnapshot() bool {
return b.UseSourceModuleTypeInSnapshot
}
func (b *SdkMemberTypeBase) SupportedTraits() SdkMemberTraitSet {
return NewSdkMemberTraitSet(b.Traits)
}
// SdkMemberTypesRegistry encapsulates the information about registered SdkMemberTypes.
type SdkMemberTypesRegistry struct {
// The list of types sorted by property name.
@@ -733,6 +921,9 @@ type SdkMemberContext interface {
// Provided for use by sdk members to create a member specific location within the snapshot
// into which to copy the prebuilt files.
Name() string
// RequiresTrait returns true if this member is expected to provide the specified trait.
RequiresTrait(trait SdkMemberTrait) bool
}
// ExportedComponentsInfo contains information about the components that this module exports to an

View File

@@ -16,6 +16,7 @@ bootstrap_go_package {
srcs: [
"bp.go",
"exports.go",
"member_trait.go",
"member_type.go",
"sdk.go",
"update.go",
@@ -28,6 +29,7 @@ bootstrap_go_package {
"exports_test.go",
"java_sdk_test.go",
"license_sdk_test.go",
"member_trait_test.go",
"sdk_test.go",
"testing.go",
],

133
sdk/member_trait.go Normal file
View File

@@ -0,0 +1,133 @@
// Copyright (C) 2021 The Android Open Source Project
//
// 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 sdk
import (
"reflect"
"android/soong/android"
"github.com/google/blueprint/proptools"
)
// Contains information about the sdk properties that list sdk members by trait, e.g.
// native_bridge.
type sdkMemberTraitListProperty struct {
// getter for the list of member names
getter func(properties interface{}) []string
// the trait of member referenced in the list
memberTrait android.SdkMemberTrait
}
// Cache of dynamically generated dynamicSdkMemberTraits objects. The key is the pointer
// to a slice of SdkMemberTrait instances held in android.RegisteredSdkMemberTraits.
var dynamicSdkMemberTraitsMap android.OncePer
// A dynamically generated set of member list properties and associated structure type.
//
// Instances of this are created by createDynamicSdkMemberTraits.
type dynamicSdkMemberTraits struct {
// The dynamically generated structure type.
//
// Contains one []string exported field for each android.RegisteredSdkMemberTraits. The name of
// the field is the exported form of the value returned by SdkMemberTrait.SdkPropertyName().
propertiesStructType reflect.Type
// Information about each of the member trait specific list properties.
memberTraitListProperties []*sdkMemberTraitListProperty
}
func (d *dynamicSdkMemberTraits) createMemberTraitListProperties() interface{} {
return reflect.New(d.propertiesStructType).Interface()
}
func getDynamicSdkMemberTraits(registry *android.SdkMemberTraitsRegistry) *dynamicSdkMemberTraits {
// Get a key that uniquely identifies the registry contents.
key := registry.UniqueOnceKey()
// Get the registered traits.
registeredTraits := registry.RegisteredTraits()
// Get the cached value, creating new instance if necessary.
return dynamicSdkMemberTraitsMap.Once(key, func() interface{} {
return createDynamicSdkMemberTraits(registeredTraits)
}).(*dynamicSdkMemberTraits)
}
// Create the dynamicSdkMemberTraits from the list of registered member traits.
//
// A struct is created which contains one exported field per member trait corresponding to
// the SdkMemberTrait.SdkPropertyName() value.
//
// A list of sdkMemberTraitListProperty instances is created, one per member trait that provides:
// * a reference to the member trait.
// * a getter for the corresponding field in the properties struct.
//
func createDynamicSdkMemberTraits(sdkMemberTraits []android.SdkMemberTrait) *dynamicSdkMemberTraits {
var listProperties []*sdkMemberTraitListProperty
memberTraitToProperty := map[android.SdkMemberTrait]*sdkMemberTraitListProperty{}
var fields []reflect.StructField
// Iterate over the member traits creating StructField and sdkMemberTraitListProperty objects.
nextFieldIndex := 0
for _, memberTrait := range sdkMemberTraits {
p := memberTrait.SdkPropertyName()
var getter func(properties interface{}) []string
// Create a dynamic exported field for the member trait's property.
fields = append(fields, reflect.StructField{
Name: proptools.FieldNameForProperty(p),
Type: reflect.TypeOf([]string{}),
})
// Copy the field index for use in the getter func as using the loop variable directly will
// cause all funcs to use the last value.
fieldIndex := nextFieldIndex
nextFieldIndex += 1
getter = func(properties interface{}) []string {
// The properties is expected to be of the following form (where
// <Module_traits> is the name of an SdkMemberTrait.SdkPropertyName().
// properties *struct {<Module_traits> []string, ....}
//
// Although it accesses the field by index the following reflection code is equivalent to:
// *properties.<Module_traits>
//
list := reflect.ValueOf(properties).Elem().Field(fieldIndex).Interface().([]string)
return list
}
// Create an sdkMemberTraitListProperty for the member trait.
memberListProperty := &sdkMemberTraitListProperty{
getter: getter,
memberTrait: memberTrait,
}
memberTraitToProperty[memberTrait] = memberListProperty
listProperties = append(listProperties, memberListProperty)
}
// Create a dynamic struct from the collated fields.
propertiesStructType := reflect.StructOf(fields)
return &dynamicSdkMemberTraits{
memberTraitListProperties: listProperties,
propertiesStructType: propertiesStructType,
}
}

287
sdk/member_trait_test.go Normal file
View File

@@ -0,0 +1,287 @@
// Copyright (C) 2021 The Android Open Source Project
//
// 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 sdk
import (
"fmt"
"path/filepath"
"testing"
"android/soong/android"
"android/soong/java"
"github.com/google/blueprint"
)
type fakeMemberTrait struct {
android.SdkMemberTraitBase
}
type fakeMemberType struct {
android.SdkMemberTypeBase
}
func (t *fakeMemberType) AddDependencies(ctx android.SdkDependencyContext, dependencyTag blueprint.DependencyTag, names []string) {
for _, name := range names {
ctx.AddVariationDependencies(nil, dependencyTag, name)
if ctx.RequiresTrait(name, extraTrait) {
ctx.AddVariationDependencies(nil, dependencyTag, name+"_extra")
}
if ctx.RequiresTrait(name, specialTrait) {
ctx.AddVariationDependencies(nil, dependencyTag, name+"_special")
}
}
}
func (t *fakeMemberType) IsInstance(module android.Module) bool {
return true
}
func (t *fakeMemberType) AddPrebuiltModule(ctx android.SdkMemberContext, member android.SdkMember) android.BpModule {
moduleType := "java_import"
if ctx.RequiresTrait(extraTrait) {
moduleType = "java_test_import"
}
return ctx.SnapshotBuilder().AddPrebuiltModule(member, moduleType)
}
func (t *fakeMemberType) CreateVariantPropertiesStruct() android.SdkMemberProperties {
return &fakeMemberTypeProperties{}
}
type fakeMemberTypeProperties struct {
android.SdkMemberPropertiesBase
path android.Path
}
func (t *fakeMemberTypeProperties) PopulateFromVariant(ctx android.SdkMemberContext, variant android.Module) {
headerJars := variant.(java.ApexDependency).HeaderJars()
if len(headerJars) != 1 {
panic(fmt.Errorf("there must be only one header jar from %q", variant.Name()))
}
t.path = headerJars[0]
}
func (t *fakeMemberTypeProperties) AddToPropertySet(ctx android.SdkMemberContext, propertySet android.BpPropertySet) {
if t.path != nil {
relative := filepath.Join("javalibs", t.path.Base())
ctx.SnapshotBuilder().CopyToSnapshot(t.path, relative)
propertySet.AddProperty("jars", []string{relative})
}
}
var (
extraTrait = &fakeMemberTrait{
SdkMemberTraitBase: android.SdkMemberTraitBase{
PropertyName: "extra",
},
}
specialTrait = &fakeMemberTrait{
SdkMemberTraitBase: android.SdkMemberTraitBase{
PropertyName: "special",
},
}
fakeType = &fakeMemberType{
SdkMemberTypeBase: android.SdkMemberTypeBase{
PropertyName: "fake_members",
SupportsSdk: true,
Traits: []android.SdkMemberTrait{
extraTrait,
specialTrait,
},
},
}
)
func init() {
android.RegisterSdkMemberTrait(extraTrait)
android.RegisterSdkMemberTrait(specialTrait)
android.RegisterSdkMemberType(fakeType)
}
func TestBasicTrait_WithoutTrait(t *testing.T) {
result := android.GroupFixturePreparers(
prepareForSdkTestWithJava,
android.FixtureWithRootAndroidBp(`
sdk {
name: "mysdk",
fake_members: ["myjavalib"],
}
java_library {
name: "myjavalib",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
`),
).RunTest(t)
CheckSnapshot(t, result, "mysdk", "",
checkUnversionedAndroidBpContents(`
// This is auto-generated. DO NOT EDIT.
java_import {
name: "myjavalib",
prefer: false,
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/myjavalib.jar"],
}
`),
checkVersionedAndroidBpContents(`
// This is auto-generated. DO NOT EDIT.
java_import {
name: "mysdk_myjavalib@current",
sdk_member_name: "myjavalib",
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/myjavalib.jar"],
}
sdk_snapshot {
name: "mysdk@current",
visibility: ["//visibility:public"],
fake_members: ["mysdk_myjavalib@current"],
}
`),
)
}
func TestBasicTrait_MultipleTraits(t *testing.T) {
result := android.GroupFixturePreparers(
prepareForSdkTestWithJava,
android.FixtureWithRootAndroidBp(`
sdk {
name: "mysdk",
fake_members: ["myjavalib", "anotherjavalib"],
traits: {
extra: ["myjavalib"],
special: ["myjavalib", "anotherjavalib"],
},
}
java_library {
name: "myjavalib",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
java_library {
name: "myjavalib_extra",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
java_library {
name: "myjavalib_special",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
java_library {
name: "anotherjavalib",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
java_library {
name: "anotherjavalib_special",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
`),
).RunTest(t)
CheckSnapshot(t, result, "mysdk", "",
checkUnversionedAndroidBpContents(`
// This is auto-generated. DO NOT EDIT.
java_test_import {
name: "myjavalib",
prefer: false,
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/myjavalib.jar"],
}
java_import {
name: "myjavalib_extra",
prefer: false,
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/myjavalib_extra.jar"],
}
java_import {
name: "myjavalib_special",
prefer: false,
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/myjavalib_special.jar"],
}
java_import {
name: "anotherjavalib",
prefer: false,
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/anotherjavalib.jar"],
}
java_import {
name: "anotherjavalib_special",
prefer: false,
visibility: ["//visibility:public"],
apex_available: ["//apex_available:platform"],
jars: ["javalibs/anotherjavalib_special.jar"],
}
`),
)
}
func TestTraitUnsupportedByMemberType(t *testing.T) {
android.GroupFixturePreparers(
prepareForSdkTestWithJava,
android.FixtureWithRootAndroidBp(`
sdk {
name: "mysdk",
java_header_libs: ["myjavalib"],
traits: {
extra: ["myjavalib"],
},
}
java_library {
name: "myjavalib",
srcs: ["Test.java"],
system_modules: "none",
sdk_version: "none",
}
`),
).ExtendWithErrorHandler(android.FixtureExpectsAtLeastOneErrorMatchingPattern(
`\Qsdk member "myjavalib" has traits [extra] that are unsupported by its member type "java_header_libs"\E`)).
RunTest(t)
}

View File

@@ -53,6 +53,13 @@ type sdk struct {
// list properties, e.g. java_libs.
dynamicMemberTypeListProperties interface{}
// The dynamically generated information about the registered SdkMemberTrait
dynamicSdkMemberTraits *dynamicSdkMemberTraits
// The dynamically created instance of the properties struct containing the sdk member trait
// list properties.
dynamicMemberTraitListProperties interface{}
// Information about the OsType specific member variants depended upon by this variant.
//
// Set by OsType specific variants in the collectMembers() method and used by the
@@ -114,7 +121,20 @@ func newSdkModule(moduleExports bool) *sdk {
// Create an instance of the dynamically created struct that contains all the
// properties for the member type specific list properties.
s.dynamicMemberTypeListProperties = s.dynamicSdkMemberTypes.createMemberTypeListProperties()
s.AddProperties(&s.properties, s.dynamicMemberTypeListProperties)
traitRegistry := android.RegisteredSdkMemberTraits
s.dynamicSdkMemberTraits = getDynamicSdkMemberTraits(traitRegistry)
// Create an instance of the dynamically created struct that contains all the properties for the
// member trait specific list properties.
s.dynamicMemberTraitListProperties = s.dynamicSdkMemberTraits.createMemberTraitListProperties()
// Create a wrapper around the dynamic trait specific properties so that they have to be
// specified within a traits:{} section in the .bp file.
traitsWrapper := struct {
Traits interface{}
}{s.dynamicMemberTraitListProperties}
s.AddProperties(&s.properties, s.dynamicMemberTypeListProperties, &traitsWrapper)
// Make sure that the prebuilt visibility property is verified for errors.
android.AddVisibilityProperty(s, "prebuilt_visibility", &s.properties.Prebuilt_visibility)
@@ -145,6 +165,11 @@ func (s *sdk) memberTypeListProperty(memberType android.SdkMemberType) *sdkMembe
return s.dynamicSdkMemberTypes.memberTypeToProperty[memberType]
}
// memberTraitListProperties returns the list of *sdkMemberTraitListProperty instances for this sdk.
func (s *sdk) memberTraitListProperties() []*sdkMemberTraitListProperty {
return s.dynamicSdkMemberTraits.memberTraitListProperties
}
func (s *sdk) snapshot() bool {
return s.properties.Snapshot
}
@@ -198,15 +223,53 @@ func (s *sdk) AndroidMkEntries() []android.AndroidMkEntries {
}}
}
// gatherTraits gathers the traits from the dynamically generated trait specific properties.
//
// Returns a map from member name to the set of required traits.
func (s *sdk) gatherTraits() map[string]android.SdkMemberTraitSet {
traitListByMember := map[string][]android.SdkMemberTrait{}
for _, memberListProperty := range s.memberTraitListProperties() {
names := memberListProperty.getter(s.dynamicMemberTraitListProperties)
for _, name := range names {
traitListByMember[name] = append(traitListByMember[name], memberListProperty.memberTrait)
}
}
traitSetByMember := map[string]android.SdkMemberTraitSet{}
for name, list := range traitListByMember {
traitSetByMember[name] = android.NewSdkMemberTraitSet(list)
}
return traitSetByMember
}
// newDependencyContext creates a new SdkDependencyContext for this sdk.
func (s *sdk) newDependencyContext(mctx android.BottomUpMutatorContext) android.SdkDependencyContext {
traits := s.gatherTraits()
return &dependencyContext{
BottomUpMutatorContext: mctx,
requiredTraits: traits,
}
}
type dependencyContext struct {
android.BottomUpMutatorContext
// Map from member name to the set of traits that the sdk requires the member provides.
requiredTraits map[string]android.SdkMemberTraitSet
}
func (d *dependencyContext) RequiredTraits(name string) android.SdkMemberTraitSet {
if s, ok := d.requiredTraits[name]; ok {
return s
} else {
return android.EmptySdkMemberTraitSet()
}
}
func (d *dependencyContext) RequiresTrait(name string, trait android.SdkMemberTrait) bool {
return d.RequiredTraits(name).Contains(trait)
}
var _ android.SdkDependencyContext = (*dependencyContext)(nil)
@@ -287,8 +350,21 @@ func memberMutator(mctx android.BottomUpMutatorContext) {
}
names := memberListProperty.getter(s.dynamicMemberTypeListProperties)
if len(names) > 0 {
memberType := memberListProperty.memberType
// Verify that the member type supports the specified traits.
supportedTraits := memberType.SupportedTraits()
for _, name := range names {
requiredTraits := ctx.RequiredTraits(name)
unsupportedTraits := requiredTraits.Subtract(supportedTraits)
if !unsupportedTraits.Empty() {
ctx.ModuleErrorf("sdk member %q has traits %s that are unsupported by its member type %q", name, unsupportedTraits, memberType.SdkPropertyName())
}
}
// Add dependencies using the appropriate tag.
tag := memberListProperty.dependencyTag
memberListProperty.memberType.AddDependencies(ctx, tag, names)
memberType.AddDependencies(ctx, tag, names)
}
}
}

View File

@@ -393,10 +393,18 @@ be unnecessary as every module in the sdk already has its own licenses property.
members := s.groupMemberVariantsByMemberThenType(ctx, memberVariantDeps)
// Create the prebuilt modules for each of the member modules.
traits := s.gatherTraits()
for _, member := range members {
memberType := member.memberType
memberCtx := &memberContext{ctx, builder, memberType, member.name}
name := member.name
requiredTraits := traits[name]
if requiredTraits == nil {
requiredTraits = android.EmptySdkMemberTraitSet()
}
// Create the snapshot for the member.
memberCtx := &memberContext{ctx, builder, memberType, name, requiredTraits}
prebuiltModule := memberType.AddPrebuiltModule(memberCtx, member)
s.createMemberSnapshot(memberCtx, member, prebuiltModule.(*bpModule))
@@ -1651,6 +1659,9 @@ type memberContext struct {
builder *snapshotBuilder
memberType android.SdkMemberType
name string
// The set of traits required of this member.
requiredTraits android.SdkMemberTraitSet
}
func (m *memberContext) SdkModuleContext() android.ModuleContext {
@@ -1669,6 +1680,10 @@ func (m *memberContext) Name() string {
return m.name
}
func (m *memberContext) RequiresTrait(trait android.SdkMemberTrait) bool {
return m.requiredTraits.Contains(trait)
}
func (s *sdk) createMemberSnapshot(ctx *memberContext, member *sdkMember, bpModule *bpModule) {
memberType := member.memberType