22152f6cda
VndkUseCoreVariant feature is deprecated along with VNDK. This change removes related code from Soong. Bug: 330100430 Test: AOSP CF build succeeded Change-Id: Ie182c9e4dc9cf0a4fe9d5fddf5b36754ddb53d18
526 lines
17 KiB
Go
526 lines
17 KiB
Go
// 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/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 })
|
|
)
|
|
|
|
// 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)
|
|
}
|
|
}
|
|
|
|
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 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
|
|
}
|
|
return lib.shared() && m.InVendor() && m.IsVndk() && !m.IsVndkExt()
|
|
}
|
|
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)
|
|
}
|
|
|
|
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")
|
|
}
|
|
|
|
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)
|
|
}
|