| // Copyright 2017 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 cc |
| |
| import ( |
| "errors" |
| "fmt" |
| "strings" |
| |
| "android/soong/android" |
| "android/soong/cc/config" |
| "android/soong/etc" |
| |
| "github.com/google/blueprint" |
| "github.com/google/blueprint/proptools" |
| ) |
| |
| const ( |
| llndkLibrariesTxt = "llndk.libraries.txt" |
| llndkLibrariesTxtForApex = "llndk.libraries.txt.apex" |
| vndkCoreLibrariesTxt = "vndkcore.libraries.txt" |
| vndkSpLibrariesTxt = "vndksp.libraries.txt" |
| vndkPrivateLibrariesTxt = "vndkprivate.libraries.txt" |
| vndkProductLibrariesTxt = "vndkproduct.libraries.txt" |
| vndkUsingCoreVariantLibrariesTxt = "vndkcorevariant.libraries.txt" |
| ) |
| |
| func VndkLibrariesTxtModules(vndkVersion string, ctx android.BaseModuleContext) []string { |
| // Snapshot vndks have their own *.libraries.VER.txt files. |
| // Note that snapshots don't have "vndkcorevariant.libraries.VER.txt" |
| result := []string{ |
| insertVndkVersion(vndkCoreLibrariesTxt, vndkVersion), |
| insertVndkVersion(vndkSpLibrariesTxt, vndkVersion), |
| insertVndkVersion(vndkPrivateLibrariesTxt, vndkVersion), |
| insertVndkVersion(vndkProductLibrariesTxt, vndkVersion), |
| insertVndkVersion(llndkLibrariesTxt, vndkVersion), |
| } |
| |
| return result |
| } |
| |
| type VndkProperties struct { |
| Vndk struct { |
| // declared as a VNDK or VNDK-SP module. The vendor variant |
| // will be installed in /system instead of /vendor partition. |
| // |
| // `vendor_available` and `product_available` must be explicitly |
| // set to either true or false together with `vndk: {enabled: true}`. |
| Enabled *bool |
| |
| // declared as a VNDK-SP module, which is a subset of VNDK. |
| // |
| // `vndk: { enabled: true }` must set together. |
| // |
| // All these modules are allowed to link to VNDK-SP or LL-NDK |
| // modules only. Other dependency will cause link-type errors. |
| // |
| // If `support_system_process` is not set or set to false, |
| // the module is VNDK-core and can link to other VNDK-core, |
| // VNDK-SP or LL-NDK modules only. |
| Support_system_process *bool |
| |
| // declared as a VNDK-private module. |
| // This module still creates the vendor and product variants refering |
| // to the `vendor_available: true` and `product_available: true` |
| // properties. However, it is only available to the other VNDK modules |
| // but not to the non-VNDK vendor or product modules. |
| Private *bool |
| |
| // Extending another module |
| Extends *string |
| } |
| } |
| |
| type vndkdep struct { |
| Properties VndkProperties |
| } |
| |
| func (vndk *vndkdep) props() []interface{} { |
| return []interface{}{&vndk.Properties} |
| } |
| |
| func (vndk *vndkdep) isVndk() bool { |
| return Bool(vndk.Properties.Vndk.Enabled) |
| } |
| |
| func (vndk *vndkdep) isVndkSp() bool { |
| return Bool(vndk.Properties.Vndk.Support_system_process) |
| } |
| |
| func (vndk *vndkdep) isVndkExt() bool { |
| return vndk.Properties.Vndk.Extends != nil |
| } |
| |
| func (vndk *vndkdep) getVndkExtendsModuleName() string { |
| return String(vndk.Properties.Vndk.Extends) |
| } |
| |
| func (vndk *vndkdep) typeName() string { |
| if !vndk.isVndk() { |
| return "native:vendor" |
| } |
| if !vndk.isVndkExt() { |
| if !vndk.isVndkSp() { |
| return "native:vendor:vndk" |
| } |
| return "native:vendor:vndksp" |
| } |
| if !vndk.isVndkSp() { |
| return "native:vendor:vndkext" |
| } |
| return "native:vendor:vndkspext" |
| } |
| |
| // VNDK link type check from a module with UseVndk() == true. |
| func (vndk *vndkdep) vndkCheckLinkType(ctx android.BaseModuleContext, to *Module, tag blueprint.DependencyTag) { |
| if to.linker == nil { |
| return |
| } |
| if !vndk.isVndk() { |
| // Non-VNDK modules those installed to /vendor, /system/vendor, |
| // /product or /system/product cannot depend on VNDK-private modules |
| // that include VNDK-core-private, VNDK-SP-private and LLNDK-private. |
| if to.IsVndkPrivate() { |
| ctx.ModuleErrorf("non-VNDK module should not link to %q which has `private: true`", to.Name()) |
| } |
| } |
| if lib, ok := to.linker.(*libraryDecorator); !ok || !lib.shared() { |
| // Check only shared libraries. |
| // Other (static) libraries are allowed to link. |
| return |
| } |
| |
| if to.IsLlndk() { |
| // LL-NDK libraries are allowed to link |
| return |
| } |
| |
| if !to.UseVndk() { |
| ctx.ModuleErrorf("(%s) should not link to %q which is not a vendor-available library", |
| vndk.typeName(), to.Name()) |
| return |
| } |
| if tag == vndkExtDepTag { |
| // Ensure `extends: "name"` property refers a vndk module that has vendor_available |
| // and has identical vndk properties. |
| if to.vndkdep == nil || !to.vndkdep.isVndk() { |
| ctx.ModuleErrorf("`extends` refers a non-vndk module %q", to.Name()) |
| return |
| } |
| if vndk.isVndkSp() != to.vndkdep.isVndkSp() { |
| ctx.ModuleErrorf( |
| "`extends` refers a module %q with mismatched support_system_process", |
| to.Name()) |
| return |
| } |
| if to.IsVndkPrivate() { |
| ctx.ModuleErrorf( |
| "`extends` refers module %q which has `private: true`", |
| to.Name()) |
| return |
| } |
| } |
| if to.vndkdep == nil { |
| return |
| } |
| |
| // Check the dependencies of VNDK shared libraries. |
| if err := vndkIsVndkDepAllowed(vndk, to.vndkdep); err != nil { |
| ctx.ModuleErrorf("(%s) should not link to %q (%s): %v", |
| vndk.typeName(), to.Name(), to.vndkdep.typeName(), err) |
| return |
| } |
| } |
| |
| func vndkIsVndkDepAllowed(from *vndkdep, to *vndkdep) error { |
| // Check the dependencies of VNDK, VNDK-Ext, VNDK-SP, VNDK-SP-Ext and vendor modules. |
| if from.isVndkExt() { |
| if from.isVndkSp() { |
| if to.isVndk() && !to.isVndkSp() { |
| return errors.New("VNDK-SP extensions must not depend on VNDK or VNDK extensions") |
| } |
| return nil |
| } |
| // VNDK-Ext may depend on VNDK, VNDK-Ext, VNDK-SP, VNDK-SP-Ext, or vendor libs. |
| return nil |
| } |
| if from.isVndk() { |
| if to.isVndkExt() { |
| return errors.New("VNDK-core and VNDK-SP must not depend on VNDK extensions") |
| } |
| if from.isVndkSp() { |
| if !to.isVndkSp() { |
| return errors.New("VNDK-SP must only depend on VNDK-SP") |
| } |
| return nil |
| } |
| if !to.isVndk() { |
| return errors.New("VNDK-core must only depend on VNDK-core or VNDK-SP") |
| } |
| return nil |
| } |
| // Vendor modules may depend on VNDK, VNDK-Ext, VNDK-SP, VNDK-SP-Ext, or vendor libs. |
| return nil |
| } |
| |
| type moduleListerFunc func(ctx android.SingletonContext) (moduleNames, fileNames []string) |
| |
| var ( |
| vndkSPLibraries = vndkModuleLister(func(m *Module) bool { return m.VendorProperties.IsVNDKSP }) |
| vndkCoreLibraries = vndkModuleLister(func(m *Module) bool { return m.VendorProperties.IsVNDKCore }) |
| vndkPrivateLibraries = vndkModuleLister(func(m *Module) bool { return m.VendorProperties.IsVNDKPrivate }) |
| vndkProductLibraries = vndkModuleLister(func(m *Module) bool { return m.VendorProperties.IsVNDKProduct }) |
| vndkUsingCoreVariantLibraries = vndkModuleLister(func(m *Module) bool { return m.VendorProperties.IsVNDKUsingCoreVariant }) |
| ) |
| |
| // vndkModuleLister takes a predicate that operates on a Module and returns a moduleListerFunc |
| // that produces a list of module names and output file names for which the predicate returns true. |
| func vndkModuleLister(predicate func(*Module) bool) moduleListerFunc { |
| return func(ctx android.SingletonContext) (moduleNames, fileNames []string) { |
| ctx.VisitAllModules(func(m android.Module) { |
| if c, ok := m.(*Module); ok && predicate(c) && !c.IsVndkPrebuiltLibrary() { |
| filename, err := getVndkFileName(c) |
| if err != nil { |
| ctx.ModuleErrorf(m, "%s", err) |
| } |
| moduleNames = append(moduleNames, ctx.ModuleName(m)) |
| fileNames = append(fileNames, filename) |
| } |
| }) |
| moduleNames = android.SortedUniqueStrings(moduleNames) |
| fileNames = android.SortedUniqueStrings(fileNames) |
| return |
| } |
| } |
| |
| // vndkModuleListRemover takes a moduleListerFunc and a prefix and returns a moduleListerFunc |
| // that returns the same lists as the input moduleListerFunc, but with modules with the |
| // given prefix removed. |
| func vndkModuleListRemover(lister moduleListerFunc, prefix string) moduleListerFunc { |
| return func(ctx android.SingletonContext) (moduleNames, fileNames []string) { |
| moduleNames, fileNames = lister(ctx) |
| filter := func(in []string) []string { |
| out := make([]string, 0, len(in)) |
| for _, lib := range in { |
| if strings.HasPrefix(lib, prefix) { |
| continue |
| } |
| out = append(out, lib) |
| } |
| return out |
| } |
| return filter(moduleNames), filter(fileNames) |
| } |
| } |
| |
| var vndkMustUseVendorVariantListKey = android.NewOnceKey("vndkMustUseVendorVariantListKey") |
| |
| func vndkMustUseVendorVariantList(cfg android.Config) []string { |
| return cfg.Once(vndkMustUseVendorVariantListKey, func() interface{} { |
| return config.VndkMustUseVendorVariantList |
| }).([]string) |
| } |
| |
| // test may call this to override global configuration(config.VndkMustUseVendorVariantList) |
| // when it is called, it must be before the first call to vndkMustUseVendorVariantList() |
| func setVndkMustUseVendorVariantListForTest(config android.Config, mustUseVendorVariantList []string) { |
| config.Once(vndkMustUseVendorVariantListKey, func() interface{} { |
| return mustUseVendorVariantList |
| }) |
| } |
| |
| func processVndkLibrary(mctx android.BottomUpMutatorContext, m *Module) { |
| if m.InProduct() { |
| // We may skip the steps for the product variants because they |
| // are already covered by the vendor variants. |
| return |
| } |
| |
| name := m.BaseModuleName() |
| |
| if lib := m.library; lib != nil && lib.hasStubsVariants() && name != "libz" { |
| // b/155456180 libz is the ONLY exception here. We don't want to make |
| // libz an LLNDK library because we in general can't guarantee that |
| // libz will behave consistently especially about the compression. |
| // i.e. the compressed output might be different across releases. |
| // As the library is an external one, it's risky to keep the compatibility |
| // promise if it becomes an LLNDK. |
| mctx.PropertyErrorf("vndk.enabled", "This library provides stubs. Shouldn't be VNDK. Consider making it as LLNDK") |
| } |
| |
| if inList(name, vndkMustUseVendorVariantList(mctx.Config())) { |
| m.Properties.MustUseVendorVariant = true |
| } |
| if mctx.DeviceConfig().VndkUseCoreVariant() && !m.Properties.MustUseVendorVariant { |
| m.VendorProperties.IsVNDKUsingCoreVariant = true |
| } |
| |
| if m.vndkdep.isVndkSp() { |
| m.VendorProperties.IsVNDKSP = true |
| } else { |
| m.VendorProperties.IsVNDKCore = true |
| } |
| if m.IsVndkPrivate() { |
| m.VendorProperties.IsVNDKPrivate = true |
| } |
| if Bool(m.VendorProperties.Product_available) { |
| m.VendorProperties.IsVNDKProduct = true |
| } |
| } |
| |
| // Check for modules that mustn't be VNDK |
| func shouldSkipVndkMutator(ctx android.ConfigAndErrorContext, m *Module) bool { |
| if !m.Enabled(ctx) { |
| return true |
| } |
| if !m.Device() { |
| // Skip non-device modules |
| return true |
| } |
| if m.Target().NativeBridge == android.NativeBridgeEnabled { |
| // Skip native_bridge modules |
| return true |
| } |
| return false |
| } |
| |
| func IsForVndkApex(mctx android.BottomUpMutatorContext, m *Module) bool { |
| if shouldSkipVndkMutator(mctx, m) { |
| return false |
| } |
| |
| // TODO(b/142675459): Use enabled: to select target device in vndk_prebuilt_shared |
| // When b/142675459 is landed, remove following check |
| if p, ok := m.linker.(*vndkPrebuiltLibraryDecorator); ok { |
| // prebuilt vndk modules should match with device |
| if !p.MatchesWithDevice(mctx.DeviceConfig()) { |
| return false |
| } |
| } |
| |
| if lib, ok := m.linker.(libraryInterface); ok { |
| // VNDK APEX doesn't need stub variants |
| if lib.buildStubs() { |
| return false |
| } |
| useCoreVariant := mctx.DeviceConfig().VndkUseCoreVariant() && !m.MustUseVendorVariant() |
| return lib.shared() && m.InVendor() && m.IsVndk() && !m.IsVndkExt() && !useCoreVariant |
| } |
| return false |
| } |
| |
| // gather list of vndk-core, vndk-sp, and ll-ndk libs |
| func VndkMutator(mctx android.BottomUpMutatorContext) { |
| m, ok := mctx.Module().(*Module) |
| if !ok { |
| return |
| } |
| |
| if shouldSkipVndkMutator(mctx, m) { |
| return |
| } |
| |
| lib, isLib := m.linker.(*libraryDecorator) |
| prebuiltLib, isPrebuiltLib := m.linker.(*prebuiltLibraryLinker) |
| |
| if m.InVendorOrProduct() && isLib && lib.hasLLNDKStubs() { |
| m.VendorProperties.IsVNDKPrivate = Bool(lib.Properties.Llndk.Private) |
| } |
| if m.InVendorOrProduct() && isPrebuiltLib && prebuiltLib.hasLLNDKStubs() { |
| m.VendorProperties.IsVNDKPrivate = Bool(prebuiltLib.Properties.Llndk.Private) |
| } |
| |
| if (isLib && lib.buildShared()) || (isPrebuiltLib && prebuiltLib.buildShared()) { |
| if m.vndkdep != nil && m.vndkdep.isVndk() && !m.vndkdep.isVndkExt() { |
| processVndkLibrary(mctx, m) |
| return |
| } |
| } |
| } |
| |
| func init() { |
| RegisterVndkLibraryTxtTypes(android.InitRegistrationContext) |
| } |
| |
| func RegisterVndkLibraryTxtTypes(ctx android.RegistrationContext) { |
| ctx.RegisterParallelSingletonModuleType("vndksp_libraries_txt", vndkSPLibrariesTxtFactory) |
| ctx.RegisterParallelSingletonModuleType("vndkcore_libraries_txt", vndkCoreLibrariesTxtFactory) |
| ctx.RegisterParallelSingletonModuleType("vndkprivate_libraries_txt", vndkPrivateLibrariesTxtFactory) |
| ctx.RegisterParallelSingletonModuleType("vndkproduct_libraries_txt", vndkProductLibrariesTxtFactory) |
| ctx.RegisterParallelSingletonModuleType("vndkcorevariant_libraries_txt", vndkUsingCoreVariantLibrariesTxtFactory) |
| } |
| |
| type vndkLibrariesTxt struct { |
| android.SingletonModuleBase |
| |
| lister moduleListerFunc |
| makeVarName string |
| filterOutFromMakeVar string |
| |
| properties VndkLibrariesTxtProperties |
| |
| outputFile android.OutputPath |
| moduleNames []string |
| fileNames []string |
| } |
| |
| type VndkLibrariesTxtProperties struct { |
| Insert_vndk_version *bool |
| Stem *string |
| } |
| |
| var _ etc.PrebuiltEtcModule = &vndkLibrariesTxt{} |
| var _ android.OutputFileProducer = &vndkLibrariesTxt{} |
| |
| // vndksp_libraries_txt is a singleton module whose content is a list of VNDKSP libraries |
| // generated by Soong but can be referenced by other modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| func vndkSPLibrariesTxtFactory() android.SingletonModule { |
| return newVndkLibrariesTxt(vndkSPLibraries, "VNDK_SAMEPROCESS_LIBRARIES") |
| } |
| |
| // vndkcore_libraries_txt is a singleton module whose content is a list of VNDK core libraries |
| // generated by Soong but can be referenced by other modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| func vndkCoreLibrariesTxtFactory() android.SingletonModule { |
| return newVndkLibrariesTxt(vndkCoreLibraries, "VNDK_CORE_LIBRARIES") |
| } |
| |
| // vndkprivate_libraries_txt is a singleton module whose content is a list of VNDK private libraries |
| // generated by Soong but can be referenced by other modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| func vndkPrivateLibrariesTxtFactory() android.SingletonModule { |
| return newVndkLibrariesTxt(vndkPrivateLibraries, "VNDK_PRIVATE_LIBRARIES") |
| } |
| |
| // vndkproduct_libraries_txt is a singleton module whose content is a list of VNDK product libraries |
| // generated by Soong but can be referenced by other modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| func vndkProductLibrariesTxtFactory() android.SingletonModule { |
| return newVndkLibrariesTxt(vndkProductLibraries, "VNDK_PRODUCT_LIBRARIES") |
| } |
| |
| // vndkcorevariant_libraries_txt is a singleton module whose content is a list of VNDK libraries |
| // that are using the core variant, generated by Soong but can be referenced by other modules. |
| // For example, apex_vndk can depend on these files as prebuilt. |
| func vndkUsingCoreVariantLibrariesTxtFactory() android.SingletonModule { |
| return newVndkLibrariesTxt(vndkUsingCoreVariantLibraries, "VNDK_USING_CORE_VARIANT_LIBRARIES") |
| } |
| |
| func newVndkLibrariesWithMakeVarFilter(lister moduleListerFunc, makeVarName string, filter string) android.SingletonModule { |
| m := &vndkLibrariesTxt{ |
| lister: lister, |
| makeVarName: makeVarName, |
| filterOutFromMakeVar: filter, |
| } |
| m.AddProperties(&m.properties) |
| android.InitAndroidArchModule(m, android.DeviceSupported, android.MultilibCommon) |
| return m |
| } |
| |
| func newVndkLibrariesTxt(lister moduleListerFunc, makeVarName string) android.SingletonModule { |
| return newVndkLibrariesWithMakeVarFilter(lister, makeVarName, "") |
| } |
| |
| func insertVndkVersion(filename string, vndkVersion string) string { |
| if index := strings.LastIndex(filename, "."); index != -1 { |
| return filename[:index] + "." + vndkVersion + filename[index:] |
| } |
| return filename |
| } |
| |
| func (txt *vndkLibrariesTxt) GenerateAndroidBuildActions(ctx android.ModuleContext) { |
| filename := proptools.StringDefault(txt.properties.Stem, txt.Name()) |
| |
| txt.outputFile = android.PathForModuleOut(ctx, filename).OutputPath |
| |
| installPath := android.PathForModuleInstall(ctx, "etc") |
| ctx.InstallFile(installPath, filename, txt.outputFile) |
| } |
| |
| func (txt *vndkLibrariesTxt) GenerateSingletonBuildActions(ctx android.SingletonContext) { |
| txt.moduleNames, txt.fileNames = txt.lister(ctx) |
| android.WriteFileRule(ctx, txt.outputFile, strings.Join(txt.fileNames, "\n")) |
| } |
| |
| func (txt *vndkLibrariesTxt) AndroidMkEntries() []android.AndroidMkEntries { |
| return []android.AndroidMkEntries{android.AndroidMkEntries{ |
| Class: "ETC", |
| OutputFile: android.OptionalPathForPath(txt.outputFile), |
| ExtraEntries: []android.AndroidMkExtraEntriesFunc{ |
| func(ctx android.AndroidMkExtraEntriesContext, entries *android.AndroidMkEntries) { |
| entries.SetString("LOCAL_MODULE_STEM", txt.outputFile.Base()) |
| }, |
| }, |
| }} |
| } |
| |
| func (txt *vndkLibrariesTxt) MakeVars(ctx android.MakeVarsContext) { |
| if txt.makeVarName == "" { |
| return |
| } |
| |
| filter := func(modules []string, prefix string) []string { |
| if prefix == "" { |
| return modules |
| } |
| var result []string |
| for _, module := range modules { |
| if strings.HasPrefix(module, prefix) { |
| continue |
| } else { |
| result = append(result, module) |
| } |
| } |
| return result |
| } |
| ctx.Strict(txt.makeVarName, strings.Join(filter(txt.moduleNames, txt.filterOutFromMakeVar), " ")) |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *vndkLibrariesTxt) OutputFile() android.OutputPath { |
| return txt.outputFile |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *vndkLibrariesTxt) BaseDir() string { |
| return "etc" |
| } |
| |
| // PrebuiltEtcModule interface |
| func (txt *vndkLibrariesTxt) SubDir() string { |
| return "" |
| } |
| |
| func (txt *vndkLibrariesTxt) OutputFiles(tag string) (android.Paths, error) { |
| return android.Paths{txt.outputFile}, nil |
| } |
| |
| func getVndkFileName(m *Module) (string, error) { |
| if library, ok := m.linker.(*libraryDecorator); ok { |
| return library.getLibNameHelper(m.BaseModuleName(), true, false) + ".so", nil |
| } |
| if prebuilt, ok := m.linker.(*prebuiltLibraryLinker); ok { |
| return prebuilt.libraryDecorator.getLibNameHelper(m.BaseModuleName(), true, false) + ".so", nil |
| } |
| return "", fmt.Errorf("VNDK library should have libraryDecorator or prebuiltLibraryLinker as linker: %T", m.linker) |
| } |