5f228b0c13
This commit fixes the incomplete ABI dumps for libc. The ABI checker counts export_system_include_dir in the exported headers. It writes the functions and types defined in the system headers to the dump files. Test: make Bug: 314010764 Change-Id: Ieae48d13b4fc4381c87a017213019e940d498967
2304 lines
88 KiB
Go
2304 lines
88 KiB
Go
// Copyright 2016 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 (
|
|
"fmt"
|
|
"io"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"android/soong/android"
|
|
|
|
"github.com/google/blueprint"
|
|
"github.com/google/blueprint/pathtools"
|
|
"github.com/google/blueprint/proptools"
|
|
)
|
|
|
|
// LibraryProperties is a collection of properties shared by cc library rules/cc.
|
|
type LibraryProperties struct {
|
|
// local file name to pass to the linker as -unexported_symbols_list
|
|
Unexported_symbols_list *string `android:"path,arch_variant"`
|
|
// local file name to pass to the linker as -force_symbols_not_weak_list
|
|
Force_symbols_not_weak_list *string `android:"path,arch_variant"`
|
|
// local file name to pass to the linker as -force_symbols_weak_list
|
|
Force_symbols_weak_list *string `android:"path,arch_variant"`
|
|
|
|
// rename host libraries to prevent overlap with system installed libraries
|
|
Unique_host_soname *bool
|
|
|
|
Aidl struct {
|
|
// export headers generated from .aidl sources
|
|
Export_aidl_headers *bool
|
|
}
|
|
|
|
Proto struct {
|
|
// export headers generated from .proto sources
|
|
Export_proto_headers *bool
|
|
}
|
|
|
|
Sysprop struct {
|
|
// Whether platform owns this sysprop library.
|
|
Platform *bool
|
|
} `blueprint:"mutated"`
|
|
|
|
Static_ndk_lib *bool
|
|
|
|
// Generate stubs to make this library accessible to APEXes.
|
|
Stubs struct {
|
|
// Relative path to the symbol map. The symbol map provides the list of
|
|
// symbols that are exported for stubs variant of this library.
|
|
Symbol_file *string `android:"path"`
|
|
|
|
// List versions to generate stubs libs for. The version name "current" is always
|
|
// implicitly added.
|
|
Versions []string
|
|
|
|
// Whether to not require the implementation of the library to be installed if a
|
|
// client of the stubs is installed. Defaults to true; set to false if the
|
|
// implementation is made available by some other means, e.g. in a Microdroid
|
|
// virtual machine.
|
|
Implementation_installable *bool
|
|
}
|
|
|
|
// set the name of the output
|
|
Stem *string `android:"arch_variant"`
|
|
|
|
// set suffix of the name of the output
|
|
Suffix *string `android:"arch_variant"`
|
|
|
|
// Properties for ABI compatibility checker.
|
|
Header_abi_checker headerAbiCheckerProperties
|
|
|
|
Target struct {
|
|
Vendor, Product struct {
|
|
// set suffix of the name of the output
|
|
Suffix *string `android:"arch_variant"`
|
|
|
|
Header_abi_checker headerAbiCheckerProperties
|
|
|
|
// Disable stubs for vendor/product variants
|
|
// This is a workaround to keep `stubs` only for "core" variant (not product/vendor).
|
|
// It would be nice if we could put `stubs` into a `target: { core: {} }`
|
|
// block but it's not supported in soong yet. This could be removed/simplified once we have
|
|
// a better syntax.
|
|
No_stubs bool
|
|
}
|
|
|
|
Platform struct {
|
|
Header_abi_checker headerAbiCheckerProperties
|
|
}
|
|
}
|
|
|
|
// Names of modules to be overridden. Listed modules can only be other shared libraries
|
|
// (in Make or Soong).
|
|
// This does not completely prevent installation of the overridden libraries, but if both
|
|
// binaries would be installed by default (in PRODUCT_PACKAGES) the other library will be removed
|
|
// from PRODUCT_PACKAGES.
|
|
Overrides []string
|
|
|
|
// Inject boringssl hash into the shared library. This is only intended for use by external/boringssl.
|
|
Inject_bssl_hash *bool `android:"arch_variant"`
|
|
|
|
// If this is an LLNDK library, properties to describe the LLNDK stubs. Will be copied from
|
|
// the module pointed to by llndk_stubs if it is set.
|
|
Llndk llndkLibraryProperties
|
|
|
|
// If this is a vendor public library, properties to describe the vendor public library stubs.
|
|
Vendor_public_library vendorPublicLibraryProperties
|
|
}
|
|
|
|
// StaticProperties is a properties stanza to affect only attributes of the "static" variants of a
|
|
// library module.
|
|
type StaticProperties struct {
|
|
Static StaticOrSharedProperties `android:"arch_variant"`
|
|
}
|
|
|
|
// SharedProperties is a properties stanza to affect only attributes of the "shared" variants of a
|
|
// library module.
|
|
type SharedProperties struct {
|
|
Shared StaticOrSharedProperties `android:"arch_variant"`
|
|
}
|
|
|
|
// StaticOrSharedProperties is an embedded struct representing properties to affect attributes of
|
|
// either only the "static" variants or only the "shared" variants of a library module. These override
|
|
// the base properties of the same name.
|
|
// Use `StaticProperties` or `SharedProperties`, depending on which variant is needed.
|
|
// `StaticOrSharedProperties` exists only to avoid duplication.
|
|
type StaticOrSharedProperties struct {
|
|
Srcs []string `android:"path,arch_variant"`
|
|
|
|
Tidy_disabled_srcs []string `android:"path,arch_variant"`
|
|
|
|
Tidy_timeout_srcs []string `android:"path,arch_variant"`
|
|
|
|
Sanitized Sanitized `android:"arch_variant"`
|
|
|
|
Cflags []string `android:"arch_variant"`
|
|
|
|
Enabled *bool `android:"arch_variant"`
|
|
Whole_static_libs []string `android:"arch_variant"`
|
|
Static_libs []string `android:"arch_variant"`
|
|
Shared_libs []string `android:"arch_variant"`
|
|
System_shared_libs []string `android:"arch_variant"`
|
|
|
|
Export_shared_lib_headers []string `android:"arch_variant"`
|
|
Export_static_lib_headers []string `android:"arch_variant"`
|
|
|
|
Apex_available []string `android:"arch_variant"`
|
|
|
|
Installable *bool `android:"arch_variant"`
|
|
}
|
|
|
|
type LibraryMutatedProperties struct {
|
|
// Build a static variant
|
|
BuildStatic bool `blueprint:"mutated"`
|
|
// Build a shared variant
|
|
BuildShared bool `blueprint:"mutated"`
|
|
// This variant is shared
|
|
VariantIsShared bool `blueprint:"mutated"`
|
|
// This variant is static
|
|
VariantIsStatic bool `blueprint:"mutated"`
|
|
|
|
// This variant is a stubs lib
|
|
BuildStubs bool `blueprint:"mutated"`
|
|
// This variant is the latest version
|
|
IsLatestVersion bool `blueprint:"mutated"`
|
|
// Version of the stubs lib
|
|
StubsVersion string `blueprint:"mutated"`
|
|
// List of all stubs versions associated with an implementation lib
|
|
AllStubsVersions []string `blueprint:"mutated"`
|
|
}
|
|
|
|
type FlagExporterProperties struct {
|
|
// list of directories relative to the Blueprints file that will
|
|
// be added to the include path (using -I) for this module and any module that links
|
|
// against this module. Directories listed in export_include_dirs do not need to be
|
|
// listed in local_include_dirs.
|
|
Export_include_dirs []string `android:"arch_variant,variant_prepend"`
|
|
|
|
// list of directories that will be added to the system include path
|
|
// using -isystem for this module and any module that links against this module.
|
|
Export_system_include_dirs []string `android:"arch_variant,variant_prepend"`
|
|
|
|
Target struct {
|
|
Vendor, Product struct {
|
|
// list of exported include directories, like
|
|
// export_include_dirs, that will be applied to
|
|
// vendor or product variant of this library.
|
|
// This will overwrite any other declarations.
|
|
Override_export_include_dirs []string
|
|
}
|
|
}
|
|
}
|
|
|
|
func init() {
|
|
RegisterLibraryBuildComponents(android.InitRegistrationContext)
|
|
}
|
|
|
|
func RegisterLibraryBuildComponents(ctx android.RegistrationContext) {
|
|
ctx.RegisterModuleType("cc_library_static", LibraryStaticFactory)
|
|
ctx.RegisterModuleType("cc_library_shared", LibrarySharedFactory)
|
|
ctx.RegisterModuleType("cc_library", LibraryFactory)
|
|
ctx.RegisterModuleType("cc_library_host_static", LibraryHostStaticFactory)
|
|
ctx.RegisterModuleType("cc_library_host_shared", LibraryHostSharedFactory)
|
|
}
|
|
|
|
// cc_library creates both static and/or shared libraries for a device and/or
|
|
// host. By default, a cc_library has a single variant that targets the device.
|
|
// Specifying `host_supported: true` also creates a library that targets the
|
|
// host.
|
|
func LibraryFactory() android.Module {
|
|
module, _ := NewLibrary(android.HostAndDeviceSupported)
|
|
// Can be used as both a static and a shared library.
|
|
module.sdkMemberTypes = []android.SdkMemberType{
|
|
sharedLibrarySdkMemberType,
|
|
staticLibrarySdkMemberType,
|
|
staticAndSharedLibrarySdkMemberType,
|
|
}
|
|
return module.Init()
|
|
}
|
|
|
|
// cc_library_static creates a static library for a device and/or host binary.
|
|
func LibraryStaticFactory() android.Module {
|
|
module, library := NewLibrary(android.HostAndDeviceSupported)
|
|
library.BuildOnlyStatic()
|
|
module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
|
|
return module.Init()
|
|
}
|
|
|
|
// cc_library_shared creates a shared library for a device and/or host.
|
|
func LibrarySharedFactory() android.Module {
|
|
module, library := NewLibrary(android.HostAndDeviceSupported)
|
|
library.BuildOnlyShared()
|
|
module.sdkMemberTypes = []android.SdkMemberType{sharedLibrarySdkMemberType}
|
|
return module.Init()
|
|
}
|
|
|
|
// cc_library_host_static creates a static library that is linkable to a host
|
|
// binary.
|
|
func LibraryHostStaticFactory() android.Module {
|
|
module, library := NewLibrary(android.HostSupported)
|
|
library.BuildOnlyStatic()
|
|
module.sdkMemberTypes = []android.SdkMemberType{staticLibrarySdkMemberType}
|
|
return module.Init()
|
|
}
|
|
|
|
// cc_library_host_shared creates a shared library that is usable on a host.
|
|
func LibraryHostSharedFactory() android.Module {
|
|
module, library := NewLibrary(android.HostSupported)
|
|
library.BuildOnlyShared()
|
|
module.sdkMemberTypes = []android.SdkMemberType{sharedLibrarySdkMemberType}
|
|
return module.Init()
|
|
}
|
|
|
|
// flagExporter is a separated portion of libraryDecorator pertaining to exported
|
|
// include paths and flags. Keeping this dependency-related information separate
|
|
// from the rest of library information is helpful in keeping data more structured
|
|
// and explicit.
|
|
type flagExporter struct {
|
|
Properties FlagExporterProperties
|
|
|
|
dirs android.Paths // Include directories to be included with -I
|
|
systemDirs android.Paths // System include directories to be included with -isystem
|
|
flags []string // Exported raw flags.
|
|
deps android.Paths
|
|
headers android.Paths
|
|
}
|
|
|
|
// exportedIncludes returns the effective include paths for this module and
|
|
// any module that links against this module. This is obtained from
|
|
// the export_include_dirs property in the appropriate target stanza.
|
|
func (f *flagExporter) exportedIncludes(ctx ModuleContext) android.Paths {
|
|
if ctx.inVendor() && f.Properties.Target.Vendor.Override_export_include_dirs != nil {
|
|
return android.PathsForModuleSrc(ctx, f.Properties.Target.Vendor.Override_export_include_dirs)
|
|
}
|
|
if ctx.inProduct() && f.Properties.Target.Product.Override_export_include_dirs != nil {
|
|
return android.PathsForModuleSrc(ctx, f.Properties.Target.Product.Override_export_include_dirs)
|
|
}
|
|
return android.PathsForModuleSrc(ctx, f.Properties.Export_include_dirs)
|
|
}
|
|
|
|
func (f *flagExporter) exportedSystemIncludes(ctx ModuleContext) android.Paths {
|
|
return android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)
|
|
}
|
|
|
|
// exportIncludes registers the include directories and system include directories to be exported
|
|
// transitively to modules depending on this module.
|
|
func (f *flagExporter) exportIncludes(ctx ModuleContext) {
|
|
f.dirs = append(f.dirs, f.exportedIncludes(ctx)...)
|
|
f.systemDirs = append(f.systemDirs, android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)...)
|
|
}
|
|
|
|
// exportIncludesAsSystem registers the include directories and system include directories to be
|
|
// exported transitively both as system include directories to modules depending on this module.
|
|
func (f *flagExporter) exportIncludesAsSystem(ctx ModuleContext) {
|
|
// all dirs are force exported as system
|
|
f.systemDirs = append(f.systemDirs, f.exportedIncludes(ctx)...)
|
|
f.systemDirs = append(f.systemDirs, android.PathsForModuleSrc(ctx, f.Properties.Export_system_include_dirs)...)
|
|
}
|
|
|
|
// reexportDirs registers the given directories as include directories to be exported transitively
|
|
// to modules depending on this module.
|
|
func (f *flagExporter) reexportDirs(dirs ...android.Path) {
|
|
f.dirs = append(f.dirs, dirs...)
|
|
}
|
|
|
|
// reexportSystemDirs registers the given directories as system include directories
|
|
// to be exported transitively to modules depending on this module.
|
|
func (f *flagExporter) reexportSystemDirs(dirs ...android.Path) {
|
|
f.systemDirs = append(f.systemDirs, dirs...)
|
|
}
|
|
|
|
// reexportFlags registers the flags to be exported transitively to modules depending on this
|
|
// module.
|
|
func (f *flagExporter) reexportFlags(flags ...string) {
|
|
if android.PrefixInList(flags, "-I") || android.PrefixInList(flags, "-isystem") {
|
|
panic(fmt.Errorf("Exporting invalid flag %q: "+
|
|
"use reexportDirs or reexportSystemDirs to export directories", flag))
|
|
}
|
|
f.flags = append(f.flags, flags...)
|
|
}
|
|
|
|
func (f *flagExporter) reexportDeps(deps ...android.Path) {
|
|
f.deps = append(f.deps, deps...)
|
|
}
|
|
|
|
// addExportedGeneratedHeaders does nothing but collects generated header files.
|
|
// This can be differ to exportedDeps which may contain phony files to minimize ninja.
|
|
func (f *flagExporter) addExportedGeneratedHeaders(headers ...android.Path) {
|
|
f.headers = append(f.headers, headers...)
|
|
}
|
|
|
|
func (f *flagExporter) setProvider(ctx android.ModuleContext) {
|
|
android.SetProvider(ctx, FlagExporterInfoProvider, FlagExporterInfo{
|
|
// Comes from Export_include_dirs property, and those of exported transitive deps
|
|
IncludeDirs: android.FirstUniquePaths(f.dirs),
|
|
// Comes from Export_system_include_dirs property, and those of exported transitive deps
|
|
SystemIncludeDirs: android.FirstUniquePaths(f.systemDirs),
|
|
// Used in very few places as a one-off way of adding extra defines.
|
|
Flags: f.flags,
|
|
// Used sparingly, for extra files that need to be explicitly exported to dependers,
|
|
// or for phony files to minimize ninja.
|
|
Deps: f.deps,
|
|
// For exported generated headers, such as exported aidl headers, proto headers, or
|
|
// sysprop headers.
|
|
GeneratedHeaders: f.headers,
|
|
})
|
|
}
|
|
|
|
// libraryDecorator wraps baseCompiler, baseLinker and baseInstaller to provide library-specific
|
|
// functionality: static vs. shared linkage, reusing object files for shared libraries
|
|
type libraryDecorator struct {
|
|
Properties LibraryProperties
|
|
StaticProperties StaticProperties
|
|
SharedProperties SharedProperties
|
|
MutatedProperties LibraryMutatedProperties
|
|
|
|
// For reusing static library objects for shared library
|
|
reuseObjects Objects
|
|
|
|
// table-of-contents file to optimize out relinking when possible
|
|
tocFile android.OptionalPath
|
|
|
|
flagExporter
|
|
flagExporterInfo *FlagExporterInfo
|
|
stripper Stripper
|
|
|
|
// For whole_static_libs
|
|
objects Objects
|
|
wholeStaticLibsFromPrebuilts android.Paths
|
|
|
|
// Uses the module's name if empty, but can be overridden. Does not include
|
|
// shlib suffix.
|
|
libName string
|
|
|
|
sabi *sabi
|
|
|
|
// Output archive of gcno coverage information files
|
|
coverageOutputFile android.OptionalPath
|
|
|
|
// Source Abi Diff
|
|
sAbiDiff android.Paths
|
|
|
|
// Location of the static library in the sysroot. Empty if the library is
|
|
// not included in the NDK.
|
|
ndkSysrootPath android.Path
|
|
|
|
// Location of the linked, unstripped library for shared libraries
|
|
unstrippedOutputFile android.Path
|
|
// Location of the linked, stripped library for shared libraries, strip: "all"
|
|
strippedAllOutputFile android.Path
|
|
|
|
// Location of the file that should be copied to dist dir when requested
|
|
distFile android.Path
|
|
|
|
versionScriptPath android.OptionalPath
|
|
|
|
postInstallCmds []string
|
|
|
|
// If useCoreVariant is true, the vendor variant of a VNDK library is
|
|
// not installed.
|
|
useCoreVariant bool
|
|
checkSameCoreVariant bool
|
|
|
|
skipAPIDefine bool
|
|
|
|
// Decorated interfaces
|
|
*baseCompiler
|
|
*baseLinker
|
|
*baseInstaller
|
|
|
|
apiListCoverageXmlPath android.ModuleOutPath
|
|
}
|
|
|
|
// linkerProps returns the list of properties structs relevant for this library. (For example, if
|
|
// the library is cc_shared_library, then static-library properties are omitted.)
|
|
func (library *libraryDecorator) linkerProps() []interface{} {
|
|
var props []interface{}
|
|
props = append(props, library.baseLinker.linkerProps()...)
|
|
props = append(props,
|
|
&library.Properties,
|
|
&library.MutatedProperties,
|
|
&library.flagExporter.Properties,
|
|
&library.stripper.StripProperties)
|
|
|
|
if library.MutatedProperties.BuildShared {
|
|
props = append(props, &library.SharedProperties)
|
|
}
|
|
if library.MutatedProperties.BuildStatic {
|
|
props = append(props, &library.StaticProperties)
|
|
}
|
|
|
|
return props
|
|
}
|
|
|
|
// linkerFlags takes a Flags struct and augments it to contain linker flags that are defined by this
|
|
// library, or that are implied by attributes of this library (such as whether this library is a
|
|
// shared library).
|
|
func (library *libraryDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
|
|
flags = library.baseLinker.linkerFlags(ctx, flags)
|
|
|
|
// MinGW spits out warnings about -fPIC even for -fpie?!) being ignored because
|
|
// all code is position independent, and then those warnings get promoted to
|
|
// errors.
|
|
if !ctx.Windows() {
|
|
flags.Global.CFlags = append(flags.Global.CFlags, "-fPIC")
|
|
}
|
|
|
|
if library.static() {
|
|
flags.Local.CFlags = append(flags.Local.CFlags, library.StaticProperties.Static.Cflags...)
|
|
} else if library.shared() {
|
|
flags.Local.CFlags = append(flags.Local.CFlags, library.SharedProperties.Shared.Cflags...)
|
|
}
|
|
|
|
if library.shared() {
|
|
libName := library.getLibName(ctx)
|
|
var f []string
|
|
if ctx.toolchain().Bionic() {
|
|
f = append(f,
|
|
"-nostdlib",
|
|
"-Wl,--gc-sections",
|
|
)
|
|
}
|
|
|
|
if ctx.Darwin() {
|
|
f = append(f,
|
|
"-dynamiclib",
|
|
"-install_name @rpath/"+libName+flags.Toolchain.ShlibSuffix(),
|
|
)
|
|
if ctx.Arch().ArchType == android.X86 {
|
|
f = append(f,
|
|
"-read_only_relocs suppress",
|
|
)
|
|
}
|
|
} else {
|
|
f = append(f, "-shared")
|
|
if !ctx.Windows() {
|
|
f = append(f, "-Wl,-soname,"+libName+flags.Toolchain.ShlibSuffix())
|
|
}
|
|
}
|
|
|
|
flags.Global.LdFlags = append(flags.Global.LdFlags, f...)
|
|
}
|
|
|
|
return flags
|
|
}
|
|
|
|
// compilerFlags takes a Flags and augments it to contain compile flags from global values,
|
|
// per-target values, module type values, per-module Blueprints properties, extra flags from
|
|
// `flags`, and generated sources from `deps`.
|
|
func (library *libraryDecorator) compilerFlags(ctx ModuleContext, flags Flags, deps PathDeps) Flags {
|
|
exportIncludeDirs := library.flagExporter.exportedIncludes(ctx)
|
|
if len(exportIncludeDirs) > 0 {
|
|
f := includeDirsToFlags(exportIncludeDirs)
|
|
flags.Local.CommonFlags = append(flags.Local.CommonFlags, f)
|
|
flags.Local.YasmFlags = append(flags.Local.YasmFlags, f)
|
|
}
|
|
|
|
flags = library.baseCompiler.compilerFlags(ctx, flags, deps)
|
|
if ctx.IsLlndk() {
|
|
// LLNDK libraries ignore most of the properties on the cc_library and use the
|
|
// LLNDK-specific properties instead.
|
|
// Wipe all the module-local properties, leaving only the global properties.
|
|
flags.Local = LocalOrGlobalFlags{}
|
|
}
|
|
if library.buildStubs() {
|
|
// Remove -include <file> when compiling stubs. Otherwise, the force included
|
|
// headers might cause conflicting types error with the symbols in the
|
|
// generated stubs source code. e.g.
|
|
// double acos(double); // in header
|
|
// void acos() {} // in the generated source code
|
|
removeInclude := func(flags []string) []string {
|
|
ret := flags[:0]
|
|
for _, f := range flags {
|
|
if strings.HasPrefix(f, "-include ") {
|
|
continue
|
|
}
|
|
ret = append(ret, f)
|
|
}
|
|
return ret
|
|
}
|
|
flags.Local.CommonFlags = removeInclude(flags.Local.CommonFlags)
|
|
flags.Local.CFlags = removeInclude(flags.Local.CFlags)
|
|
|
|
flags = addStubLibraryCompilerFlags(flags)
|
|
}
|
|
return flags
|
|
}
|
|
|
|
func (library *libraryDecorator) getHeaderAbiCheckerProperties(ctx android.BaseModuleContext) headerAbiCheckerProperties {
|
|
m := ctx.Module().(*Module)
|
|
variantProps := &library.Properties.Target.Platform.Header_abi_checker
|
|
if m.InVendor() {
|
|
variantProps = &library.Properties.Target.Vendor.Header_abi_checker
|
|
} else if m.InProduct() {
|
|
variantProps = &library.Properties.Target.Product.Header_abi_checker
|
|
}
|
|
props := library.Properties.Header_abi_checker
|
|
err := proptools.AppendProperties(&props, variantProps, nil)
|
|
if err != nil {
|
|
ctx.ModuleErrorf("Cannot merge headerAbiCheckerProperties: %s", err.Error())
|
|
}
|
|
return props
|
|
}
|
|
|
|
func (library *libraryDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
|
|
if ctx.IsLlndk() {
|
|
vendorApiLevel := ctx.Config().VendorApiLevel()
|
|
if vendorApiLevel == "" {
|
|
// TODO(b/321892570): Some tests relying on old fixtures which
|
|
// doesn't set vendorApiLevel. Needs to fix them.
|
|
vendorApiLevel = ctx.Config().PlatformSdkVersion().String()
|
|
}
|
|
// This is the vendor variant of an LLNDK library, build the LLNDK stubs.
|
|
nativeAbiResult := parseNativeAbiDefinition(ctx,
|
|
String(library.Properties.Llndk.Symbol_file),
|
|
android.ApiLevelOrPanic(ctx, vendorApiLevel), "--llndk")
|
|
objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
|
|
if !Bool(library.Properties.Llndk.Unversioned) {
|
|
library.versionScriptPath = android.OptionalPathForPath(
|
|
nativeAbiResult.versionScript)
|
|
}
|
|
return objs
|
|
}
|
|
if ctx.IsVendorPublicLibrary() {
|
|
nativeAbiResult := parseNativeAbiDefinition(ctx,
|
|
String(library.Properties.Vendor_public_library.Symbol_file),
|
|
android.FutureApiLevel, "")
|
|
objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
|
|
if !Bool(library.Properties.Vendor_public_library.Unversioned) {
|
|
library.versionScriptPath = android.OptionalPathForPath(nativeAbiResult.versionScript)
|
|
}
|
|
return objs
|
|
}
|
|
if library.buildStubs() {
|
|
symbolFile := String(library.Properties.Stubs.Symbol_file)
|
|
if symbolFile != "" && !strings.HasSuffix(symbolFile, ".map.txt") {
|
|
ctx.PropertyErrorf("symbol_file", "%q doesn't have .map.txt suffix", symbolFile)
|
|
return Objects{}
|
|
}
|
|
// b/239274367 --apex and --systemapi filters symbols tagged with # apex and #
|
|
// systemapi, respectively. The former is for symbols defined in platform libraries
|
|
// and the latter is for symbols defined in APEXes.
|
|
// A single library can contain either # apex or # systemapi, but not both.
|
|
// The stub generator (ndkstubgen) is additive, so passing _both_ of these to it should be a no-op.
|
|
// However, having this distinction helps guard accidental
|
|
// promotion or demotion of API and also helps the API review process b/191371676
|
|
var flag string
|
|
if ctx.Module().(android.ApexModule).NotInPlatform() {
|
|
flag = "--apex"
|
|
} else {
|
|
flag = "--systemapi"
|
|
}
|
|
// b/184712170, unless the lib is an NDK library, exclude all public symbols from
|
|
// the stub so that it is mandated that all symbols are explicitly marked with
|
|
// either apex or systemapi.
|
|
if !ctx.Module().(*Module).IsNdk(ctx.Config()) {
|
|
flag = flag + " --no-ndk"
|
|
}
|
|
nativeAbiResult := parseNativeAbiDefinition(ctx, symbolFile,
|
|
android.ApiLevelOrPanic(ctx, library.MutatedProperties.StubsVersion), flag)
|
|
objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
|
|
library.versionScriptPath = android.OptionalPathForPath(
|
|
nativeAbiResult.versionScript)
|
|
|
|
// Parse symbol file to get API list for coverage
|
|
if library.stubsVersion() == "current" && ctx.PrimaryArch() && !ctx.inRecovery() && !ctx.inProduct() && !ctx.inVendor() {
|
|
library.apiListCoverageXmlPath = parseSymbolFileForAPICoverage(ctx, symbolFile)
|
|
}
|
|
|
|
return objs
|
|
}
|
|
|
|
if !library.buildShared() && !library.buildStatic() {
|
|
if len(library.baseCompiler.Properties.Srcs) > 0 {
|
|
ctx.PropertyErrorf("srcs", "cc_library_headers must not have any srcs")
|
|
}
|
|
if len(library.StaticProperties.Static.Srcs) > 0 {
|
|
ctx.PropertyErrorf("static.srcs", "cc_library_headers must not have any srcs")
|
|
}
|
|
if len(library.SharedProperties.Shared.Srcs) > 0 {
|
|
ctx.PropertyErrorf("shared.srcs", "cc_library_headers must not have any srcs")
|
|
}
|
|
return Objects{}
|
|
}
|
|
if library.sabi.shouldCreateSourceAbiDump() {
|
|
dirs := library.exportedIncludeDirsForAbiCheck(ctx)
|
|
flags.SAbiFlags = make([]string, 0, len(dirs))
|
|
for _, dir := range dirs {
|
|
flags.SAbiFlags = append(flags.SAbiFlags, "-I"+dir)
|
|
}
|
|
totalLength := len(library.baseCompiler.Properties.Srcs) + len(deps.GeneratedSources) +
|
|
len(library.SharedProperties.Shared.Srcs) + len(library.StaticProperties.Static.Srcs)
|
|
if totalLength > 0 {
|
|
flags.SAbiDump = true
|
|
}
|
|
}
|
|
objs := library.baseCompiler.compile(ctx, flags, deps)
|
|
library.reuseObjects = objs
|
|
buildFlags := flagsToBuilderFlags(flags)
|
|
|
|
if library.static() {
|
|
srcs := android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Srcs)
|
|
objs = objs.Append(compileObjs(ctx, buildFlags, android.DeviceStaticLibrary, srcs,
|
|
android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Tidy_disabled_srcs),
|
|
android.PathsForModuleSrc(ctx, library.StaticProperties.Static.Tidy_timeout_srcs),
|
|
library.baseCompiler.pathDeps, library.baseCompiler.cFlagsDeps))
|
|
} else if library.shared() {
|
|
srcs := android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Srcs)
|
|
objs = objs.Append(compileObjs(ctx, buildFlags, android.DeviceSharedLibrary, srcs,
|
|
android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Tidy_disabled_srcs),
|
|
android.PathsForModuleSrc(ctx, library.SharedProperties.Shared.Tidy_timeout_srcs),
|
|
library.baseCompiler.pathDeps, library.baseCompiler.cFlagsDeps))
|
|
}
|
|
|
|
return objs
|
|
}
|
|
|
|
type libraryInterface interface {
|
|
versionedInterface
|
|
|
|
static() bool
|
|
shared() bool
|
|
objs() Objects
|
|
reuseObjs() Objects
|
|
toc() android.OptionalPath
|
|
|
|
// Returns true if the build options for the module have selected a static or shared build
|
|
buildStatic() bool
|
|
buildShared() bool
|
|
|
|
// Sets whether a specific variant is static or shared
|
|
setStatic()
|
|
setShared()
|
|
|
|
// Gets the ABI properties for vendor, product, or platform variant
|
|
getHeaderAbiCheckerProperties(ctx android.BaseModuleContext) headerAbiCheckerProperties
|
|
|
|
// Write LOCAL_ADDITIONAL_DEPENDENCIES for ABI diff
|
|
androidMkWriteAdditionalDependenciesForSourceAbiDiff(w io.Writer)
|
|
|
|
availableFor(string) bool
|
|
|
|
getAPIListCoverageXMLPath() android.ModuleOutPath
|
|
|
|
installable() *bool
|
|
}
|
|
|
|
type versionedInterface interface {
|
|
buildStubs() bool
|
|
setBuildStubs(isLatest bool)
|
|
hasStubsVariants() bool
|
|
isStubsImplementationRequired() bool
|
|
setStubsVersion(string)
|
|
stubsVersion() string
|
|
|
|
stubsVersions(ctx android.BaseMutatorContext) []string
|
|
setAllStubsVersions([]string)
|
|
allStubsVersions() []string
|
|
|
|
implementationModuleName(name string) string
|
|
hasLLNDKStubs() bool
|
|
hasLLNDKHeaders() bool
|
|
hasVendorPublicLibrary() bool
|
|
}
|
|
|
|
var _ libraryInterface = (*libraryDecorator)(nil)
|
|
var _ versionedInterface = (*libraryDecorator)(nil)
|
|
|
|
func (library *libraryDecorator) getLibNameHelper(baseModuleName string, inVendor bool, inProduct bool) string {
|
|
name := library.libName
|
|
if name == "" {
|
|
name = String(library.Properties.Stem)
|
|
if name == "" {
|
|
name = baseModuleName
|
|
}
|
|
}
|
|
|
|
suffix := ""
|
|
if inVendor {
|
|
suffix = String(library.Properties.Target.Vendor.Suffix)
|
|
} else if inProduct {
|
|
suffix = String(library.Properties.Target.Product.Suffix)
|
|
}
|
|
if suffix == "" {
|
|
suffix = String(library.Properties.Suffix)
|
|
}
|
|
|
|
return name + suffix
|
|
}
|
|
|
|
// getLibName returns the actual canonical name of the library (the name which
|
|
// should be passed to the linker via linker flags).
|
|
func (library *libraryDecorator) getLibName(ctx BaseModuleContext) string {
|
|
name := library.getLibNameHelper(ctx.baseModuleName(), ctx.inVendor(), ctx.inProduct())
|
|
|
|
if ctx.Host() && Bool(library.Properties.Unique_host_soname) {
|
|
if !strings.HasSuffix(name, "-host") {
|
|
name = name + "-host"
|
|
}
|
|
}
|
|
|
|
return name
|
|
}
|
|
|
|
var versioningMacroNamesListMutex sync.Mutex
|
|
|
|
func (library *libraryDecorator) linkerInit(ctx BaseModuleContext) {
|
|
location := InstallInSystem
|
|
if library.baseLinker.sanitize.inSanitizerDir() {
|
|
location = InstallInSanitizerDir
|
|
}
|
|
library.baseInstaller.location = location
|
|
library.baseLinker.linkerInit(ctx)
|
|
// Let baseLinker know whether this variant is for stubs or not, so that
|
|
// it can omit things that are not required for linking stubs.
|
|
library.baseLinker.dynamicProperties.BuildStubs = library.buildStubs()
|
|
|
|
if library.buildStubs() {
|
|
macroNames := versioningMacroNamesList(ctx.Config())
|
|
myName := versioningMacroName(ctx.ModuleName())
|
|
versioningMacroNamesListMutex.Lock()
|
|
defer versioningMacroNamesListMutex.Unlock()
|
|
if (*macroNames)[myName] == "" {
|
|
(*macroNames)[myName] = ctx.ModuleName()
|
|
} else if (*macroNames)[myName] != ctx.ModuleName() {
|
|
ctx.ModuleErrorf("Macro name %q for versioning conflicts with macro name from module %q ", myName, (*macroNames)[myName])
|
|
}
|
|
}
|
|
}
|
|
|
|
func (library *libraryDecorator) compilerDeps(ctx DepsContext, deps Deps) Deps {
|
|
if ctx.IsLlndk() {
|
|
// LLNDK libraries ignore most of the properties on the cc_library and use the
|
|
// LLNDK-specific properties instead.
|
|
return deps
|
|
}
|
|
|
|
deps = library.baseCompiler.compilerDeps(ctx, deps)
|
|
|
|
return deps
|
|
}
|
|
|
|
func (library *libraryDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
|
|
if ctx.IsLlndk() {
|
|
// LLNDK libraries ignore most of the properties on the cc_library and use the
|
|
// LLNDK-specific properties instead.
|
|
deps.HeaderLibs = append([]string(nil), library.Properties.Llndk.Export_llndk_headers...)
|
|
deps.ReexportHeaderLibHeaders = append([]string(nil), library.Properties.Llndk.Export_llndk_headers...)
|
|
return deps
|
|
}
|
|
if ctx.IsVendorPublicLibrary() {
|
|
headers := library.Properties.Vendor_public_library.Export_public_headers
|
|
deps.HeaderLibs = append([]string(nil), headers...)
|
|
deps.ReexportHeaderLibHeaders = append([]string(nil), headers...)
|
|
return deps
|
|
}
|
|
|
|
if library.static() {
|
|
// Compare with nil because an empty list needs to be propagated.
|
|
if library.StaticProperties.Static.System_shared_libs != nil {
|
|
library.baseLinker.Properties.System_shared_libs = library.StaticProperties.Static.System_shared_libs
|
|
}
|
|
} else if library.shared() {
|
|
// Compare with nil because an empty list needs to be propagated.
|
|
if library.SharedProperties.Shared.System_shared_libs != nil {
|
|
library.baseLinker.Properties.System_shared_libs = library.SharedProperties.Shared.System_shared_libs
|
|
}
|
|
}
|
|
|
|
deps = library.baseLinker.linkerDeps(ctx, deps)
|
|
|
|
if library.static() {
|
|
deps.WholeStaticLibs = append(deps.WholeStaticLibs,
|
|
library.StaticProperties.Static.Whole_static_libs...)
|
|
deps.StaticLibs = append(deps.StaticLibs, library.StaticProperties.Static.Static_libs...)
|
|
deps.SharedLibs = append(deps.SharedLibs, library.StaticProperties.Static.Shared_libs...)
|
|
|
|
deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, library.StaticProperties.Static.Export_shared_lib_headers...)
|
|
deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, library.StaticProperties.Static.Export_static_lib_headers...)
|
|
} else if library.shared() {
|
|
if library.baseLinker.Properties.crt() {
|
|
deps.CrtBegin = append(deps.CrtBegin, ctx.toolchain().CrtBeginSharedLibrary()...)
|
|
deps.CrtEnd = append(deps.CrtEnd, ctx.toolchain().CrtEndSharedLibrary()...)
|
|
|
|
}
|
|
if library.baseLinker.Properties.crtPadSegment() {
|
|
deps.CrtEnd = append(deps.CrtEnd, ctx.toolchain().CrtPadSegmentSharedLibrary()...)
|
|
}
|
|
deps.WholeStaticLibs = append(deps.WholeStaticLibs, library.SharedProperties.Shared.Whole_static_libs...)
|
|
deps.StaticLibs = append(deps.StaticLibs, library.SharedProperties.Shared.Static_libs...)
|
|
deps.SharedLibs = append(deps.SharedLibs, library.SharedProperties.Shared.Shared_libs...)
|
|
|
|
deps.ReexportSharedLibHeaders = append(deps.ReexportSharedLibHeaders, library.SharedProperties.Shared.Export_shared_lib_headers...)
|
|
deps.ReexportStaticLibHeaders = append(deps.ReexportStaticLibHeaders, library.SharedProperties.Shared.Export_static_lib_headers...)
|
|
|
|
deps.LlndkHeaderLibs = append(deps.LlndkHeaderLibs, library.Properties.Llndk.Export_llndk_headers...)
|
|
}
|
|
if ctx.inVendor() {
|
|
deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
|
|
deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Vendor.Exclude_shared_libs)
|
|
deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
|
|
deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Vendor.Exclude_shared_libs)
|
|
deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Vendor.Exclude_static_libs)
|
|
}
|
|
if ctx.inProduct() {
|
|
deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
|
|
deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Product.Exclude_shared_libs)
|
|
deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
|
|
deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Product.Exclude_shared_libs)
|
|
deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Product.Exclude_static_libs)
|
|
}
|
|
if ctx.inRecovery() {
|
|
deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
|
|
deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Recovery.Exclude_shared_libs)
|
|
deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
|
|
deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Recovery.Exclude_shared_libs)
|
|
deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Recovery.Exclude_static_libs)
|
|
}
|
|
if ctx.inRamdisk() {
|
|
deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
|
|
deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_shared_libs)
|
|
deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
|
|
deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Ramdisk.Exclude_shared_libs)
|
|
deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Ramdisk.Exclude_static_libs)
|
|
}
|
|
if ctx.inVendorRamdisk() {
|
|
deps.WholeStaticLibs = removeListFromList(deps.WholeStaticLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
|
|
deps.SharedLibs = removeListFromList(deps.SharedLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
|
|
deps.StaticLibs = removeListFromList(deps.StaticLibs, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
|
|
deps.ReexportSharedLibHeaders = removeListFromList(deps.ReexportSharedLibHeaders, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_shared_libs)
|
|
deps.ReexportStaticLibHeaders = removeListFromList(deps.ReexportStaticLibHeaders, library.baseLinker.Properties.Target.Vendor_ramdisk.Exclude_static_libs)
|
|
}
|
|
|
|
return deps
|
|
}
|
|
|
|
func (library *libraryDecorator) linkerSpecifiedDeps(specifiedDeps specifiedDeps) specifiedDeps {
|
|
specifiedDeps = library.baseLinker.linkerSpecifiedDeps(specifiedDeps)
|
|
var properties StaticOrSharedProperties
|
|
if library.static() {
|
|
properties = library.StaticProperties.Static
|
|
} else if library.shared() {
|
|
properties = library.SharedProperties.Shared
|
|
}
|
|
|
|
specifiedDeps.sharedLibs = append(specifiedDeps.sharedLibs, properties.Shared_libs...)
|
|
|
|
// Must distinguish nil and [] in system_shared_libs - ensure that [] in
|
|
// either input list doesn't come out as nil.
|
|
if specifiedDeps.systemSharedLibs == nil {
|
|
specifiedDeps.systemSharedLibs = properties.System_shared_libs
|
|
} else {
|
|
specifiedDeps.systemSharedLibs = append(specifiedDeps.systemSharedLibs, properties.System_shared_libs...)
|
|
}
|
|
|
|
specifiedDeps.sharedLibs = android.FirstUniqueStrings(specifiedDeps.sharedLibs)
|
|
if len(specifiedDeps.systemSharedLibs) > 0 {
|
|
// Skip this if systemSharedLibs is either nil or [], to ensure they are
|
|
// retained.
|
|
specifiedDeps.systemSharedLibs = android.FirstUniqueStrings(specifiedDeps.systemSharedLibs)
|
|
}
|
|
return specifiedDeps
|
|
}
|
|
|
|
func (library *libraryDecorator) moduleInfoJSON(ctx ModuleContext, moduleInfoJSON *android.ModuleInfoJSON) {
|
|
if library.static() {
|
|
moduleInfoJSON.Class = []string{"STATIC_LIBRARIES"}
|
|
moduleInfoJSON.Uninstallable = true
|
|
} else if library.shared() {
|
|
moduleInfoJSON.Class = []string{"SHARED_LIBRARIES"}
|
|
} else if library.header() {
|
|
moduleInfoJSON.Class = []string{"HEADER_LIBRARIES"}
|
|
moduleInfoJSON.Uninstallable = true
|
|
}
|
|
|
|
if library.buildStubs() && library.stubsVersion() != "" {
|
|
moduleInfoJSON.SubName += "." + library.stubsVersion()
|
|
}
|
|
|
|
// If a library providing a stub is included in an APEX, the private APIs of the library
|
|
// is accessible only inside the APEX. From outside of the APEX, clients can only use the
|
|
// public APIs via the stub. To enforce this, the (latest version of the) stub gets the
|
|
// name of the library. The impl library instead gets the `.bootstrap` suffix to so that
|
|
// they can be exceptionally used directly when APEXes are not available (e.g. during the
|
|
// very early stage in the boot process).
|
|
if len(library.Properties.Stubs.Versions) > 0 && !ctx.Host() && ctx.notInPlatform() &&
|
|
!ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() && !ctx.useVndk() && !ctx.static() {
|
|
if library.buildStubs() && library.isLatestStubVersion() {
|
|
moduleInfoJSON.SubName = ""
|
|
}
|
|
if !library.buildStubs() {
|
|
moduleInfoJSON.SubName = ".bootstrap"
|
|
}
|
|
}
|
|
|
|
library.baseLinker.moduleInfoJSON(ctx, moduleInfoJSON)
|
|
}
|
|
|
|
func (library *libraryDecorator) linkStatic(ctx ModuleContext,
|
|
flags Flags, deps PathDeps, objs Objects) android.Path {
|
|
|
|
library.objects = deps.WholeStaticLibObjs.Copy()
|
|
library.objects = library.objects.Append(objs)
|
|
library.wholeStaticLibsFromPrebuilts = android.CopyOfPaths(deps.WholeStaticLibsFromPrebuilts)
|
|
|
|
fileName := ctx.ModuleName() + staticLibraryExtension
|
|
outputFile := android.PathForModuleOut(ctx, fileName)
|
|
builderFlags := flagsToBuilderFlags(flags)
|
|
|
|
if Bool(library.baseLinker.Properties.Use_version_lib) {
|
|
if ctx.Host() {
|
|
versionedOutputFile := outputFile
|
|
outputFile = android.PathForModuleOut(ctx, "unversioned", fileName)
|
|
library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
|
|
} else {
|
|
versionedOutputFile := android.PathForModuleOut(ctx, "versioned", fileName)
|
|
library.distFile = versionedOutputFile
|
|
library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
|
|
}
|
|
}
|
|
|
|
transformObjToStaticLib(ctx, library.objects.objFiles, deps.WholeStaticLibsFromPrebuilts, builderFlags, outputFile, nil, objs.tidyDepFiles)
|
|
|
|
library.coverageOutputFile = transformCoverageFilesToZip(ctx, library.objects, ctx.ModuleName())
|
|
|
|
ctx.CheckbuildFile(outputFile)
|
|
|
|
if library.static() {
|
|
android.SetProvider(ctx, StaticLibraryInfoProvider, StaticLibraryInfo{
|
|
StaticLibrary: outputFile,
|
|
ReuseObjects: library.reuseObjects,
|
|
Objects: library.objects,
|
|
WholeStaticLibsFromPrebuilts: library.wholeStaticLibsFromPrebuilts,
|
|
|
|
TransitiveStaticLibrariesForOrdering: android.NewDepSetBuilder[android.Path](android.TOPOLOGICAL).
|
|
Direct(outputFile).
|
|
Transitive(deps.TranstiveStaticLibrariesForOrdering).
|
|
Build(),
|
|
})
|
|
}
|
|
|
|
if library.header() {
|
|
android.SetProvider(ctx, HeaderLibraryInfoProvider, HeaderLibraryInfo{})
|
|
}
|
|
|
|
return outputFile
|
|
}
|
|
|
|
func ndkSharedLibDeps(ctx ModuleContext) android.Paths {
|
|
if ctx.Module().(*Module).IsSdkVariant() {
|
|
// The NDK sysroot timestamp file depends on all the NDK
|
|
// sysroot header and shared library files.
|
|
return android.Paths{getNdkBaseTimestampFile(ctx)}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (library *libraryDecorator) linkShared(ctx ModuleContext,
|
|
flags Flags, deps PathDeps, objs Objects) android.Path {
|
|
|
|
var linkerDeps android.Paths
|
|
linkerDeps = append(linkerDeps, flags.LdFlagsDeps...)
|
|
linkerDeps = append(linkerDeps, ndkSharedLibDeps(ctx)...)
|
|
|
|
unexportedSymbols := ctx.ExpandOptionalSource(library.Properties.Unexported_symbols_list, "unexported_symbols_list")
|
|
forceNotWeakSymbols := ctx.ExpandOptionalSource(library.Properties.Force_symbols_not_weak_list, "force_symbols_not_weak_list")
|
|
forceWeakSymbols := ctx.ExpandOptionalSource(library.Properties.Force_symbols_weak_list, "force_symbols_weak_list")
|
|
if !ctx.Darwin() {
|
|
if unexportedSymbols.Valid() {
|
|
ctx.PropertyErrorf("unexported_symbols_list", "Only supported on Darwin")
|
|
}
|
|
if forceNotWeakSymbols.Valid() {
|
|
ctx.PropertyErrorf("force_symbols_not_weak_list", "Only supported on Darwin")
|
|
}
|
|
if forceWeakSymbols.Valid() {
|
|
ctx.PropertyErrorf("force_symbols_weak_list", "Only supported on Darwin")
|
|
}
|
|
} else {
|
|
if unexportedSymbols.Valid() {
|
|
flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-unexported_symbols_list,"+unexportedSymbols.String())
|
|
linkerDeps = append(linkerDeps, unexportedSymbols.Path())
|
|
}
|
|
if forceNotWeakSymbols.Valid() {
|
|
flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-force_symbols_not_weak_list,"+forceNotWeakSymbols.String())
|
|
linkerDeps = append(linkerDeps, forceNotWeakSymbols.Path())
|
|
}
|
|
if forceWeakSymbols.Valid() {
|
|
flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,-force_symbols_weak_list,"+forceWeakSymbols.String())
|
|
linkerDeps = append(linkerDeps, forceWeakSymbols.Path())
|
|
}
|
|
}
|
|
if library.versionScriptPath.Valid() {
|
|
linkerScriptFlags := "-Wl,--version-script," + library.versionScriptPath.String()
|
|
flags.Local.LdFlags = append(flags.Local.LdFlags, linkerScriptFlags)
|
|
linkerDeps = append(linkerDeps, library.versionScriptPath.Path())
|
|
}
|
|
|
|
fileName := library.getLibName(ctx) + flags.Toolchain.ShlibSuffix()
|
|
outputFile := android.PathForModuleOut(ctx, fileName)
|
|
unstrippedOutputFile := outputFile
|
|
|
|
var implicitOutputs android.WritablePaths
|
|
if ctx.Windows() {
|
|
importLibraryPath := android.PathForModuleOut(ctx, pathtools.ReplaceExtension(fileName, "lib"))
|
|
|
|
flags.Local.LdFlags = append(flags.Local.LdFlags, "-Wl,--out-implib="+importLibraryPath.String())
|
|
implicitOutputs = append(implicitOutputs, importLibraryPath)
|
|
}
|
|
|
|
builderFlags := flagsToBuilderFlags(flags)
|
|
|
|
if ctx.Darwin() && deps.DarwinSecondArchOutput.Valid() {
|
|
fatOutputFile := outputFile
|
|
outputFile = android.PathForModuleOut(ctx, "pre-fat", fileName)
|
|
transformDarwinUniversalBinary(ctx, fatOutputFile, outputFile, deps.DarwinSecondArchOutput.Path())
|
|
}
|
|
|
|
// Optimize out relinking against shared libraries whose interface hasn't changed by
|
|
// depending on a table of contents file instead of the library itself.
|
|
tocFile := outputFile.ReplaceExtension(ctx, flags.Toolchain.ShlibSuffix()[1:]+".toc")
|
|
library.tocFile = android.OptionalPathForPath(tocFile)
|
|
TransformSharedObjectToToc(ctx, outputFile, tocFile)
|
|
|
|
stripFlags := flagsToStripFlags(flags)
|
|
needsStrip := library.stripper.NeedsStrip(ctx)
|
|
if library.buildStubs() {
|
|
// No need to strip stubs libraries
|
|
needsStrip = false
|
|
}
|
|
if needsStrip {
|
|
if ctx.Darwin() {
|
|
stripFlags.StripUseGnuStrip = true
|
|
}
|
|
strippedOutputFile := outputFile
|
|
outputFile = android.PathForModuleOut(ctx, "unstripped", fileName)
|
|
library.stripper.StripExecutableOrSharedLib(ctx, outputFile, strippedOutputFile, stripFlags)
|
|
}
|
|
library.unstrippedOutputFile = outputFile
|
|
|
|
outputFile = maybeInjectBoringSSLHash(ctx, outputFile, library.Properties.Inject_bssl_hash, fileName)
|
|
|
|
if Bool(library.baseLinker.Properties.Use_version_lib) {
|
|
if ctx.Host() {
|
|
versionedOutputFile := outputFile
|
|
outputFile = android.PathForModuleOut(ctx, "unversioned", fileName)
|
|
library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
|
|
} else {
|
|
versionedOutputFile := android.PathForModuleOut(ctx, "versioned", fileName)
|
|
library.distFile = versionedOutputFile
|
|
|
|
if library.stripper.NeedsStrip(ctx) {
|
|
out := android.PathForModuleOut(ctx, "versioned-stripped", fileName)
|
|
library.distFile = out
|
|
library.stripper.StripExecutableOrSharedLib(ctx, versionedOutputFile, out, stripFlags)
|
|
}
|
|
|
|
library.injectVersionSymbol(ctx, outputFile, versionedOutputFile)
|
|
}
|
|
}
|
|
|
|
// Generate an output file for dist as if strip: "all" is set on the module.
|
|
// Currently this is for layoutlib release process only.
|
|
for _, dist := range ctx.Module().(*Module).Dists() {
|
|
if dist.Tag != nil && *dist.Tag == "stripped_all" {
|
|
strippedAllOutputFile := android.PathForModuleOut(ctx, "stripped_all", fileName)
|
|
transformStrip(ctx, outputFile, strippedAllOutputFile, StripFlags{Toolchain: flags.Toolchain})
|
|
library.strippedAllOutputFile = strippedAllOutputFile
|
|
break
|
|
}
|
|
}
|
|
|
|
sharedLibs := deps.EarlySharedLibs
|
|
sharedLibs = append(sharedLibs, deps.SharedLibs...)
|
|
sharedLibs = append(sharedLibs, deps.LateSharedLibs...)
|
|
|
|
linkerDeps = append(linkerDeps, deps.EarlySharedLibsDeps...)
|
|
linkerDeps = append(linkerDeps, deps.SharedLibsDeps...)
|
|
linkerDeps = append(linkerDeps, deps.LateSharedLibsDeps...)
|
|
transformObjToDynamicBinary(ctx, objs.objFiles, sharedLibs,
|
|
deps.StaticLibs, deps.LateStaticLibs, deps.WholeStaticLibs,
|
|
linkerDeps, deps.CrtBegin, deps.CrtEnd, false, builderFlags, outputFile, implicitOutputs, objs.tidyDepFiles)
|
|
|
|
objs.coverageFiles = append(objs.coverageFiles, deps.StaticLibObjs.coverageFiles...)
|
|
objs.coverageFiles = append(objs.coverageFiles, deps.WholeStaticLibObjs.coverageFiles...)
|
|
objs.sAbiDumpFiles = append(objs.sAbiDumpFiles, deps.StaticLibObjs.sAbiDumpFiles...)
|
|
objs.sAbiDumpFiles = append(objs.sAbiDumpFiles, deps.WholeStaticLibObjs.sAbiDumpFiles...)
|
|
|
|
library.coverageOutputFile = transformCoverageFilesToZip(ctx, objs, library.getLibName(ctx))
|
|
library.linkSAbiDumpFiles(ctx, deps, objs, fileName, unstrippedOutputFile)
|
|
|
|
var transitiveStaticLibrariesForOrdering *android.DepSet[android.Path]
|
|
if static := ctx.GetDirectDepsWithTag(staticVariantTag); len(static) > 0 {
|
|
s, _ := android.OtherModuleProvider(ctx, static[0], StaticLibraryInfoProvider)
|
|
transitiveStaticLibrariesForOrdering = s.TransitiveStaticLibrariesForOrdering
|
|
}
|
|
|
|
android.SetProvider(ctx, SharedLibraryInfoProvider, SharedLibraryInfo{
|
|
TableOfContents: android.OptionalPathForPath(tocFile),
|
|
SharedLibrary: unstrippedOutputFile,
|
|
TransitiveStaticLibrariesForOrdering: transitiveStaticLibrariesForOrdering,
|
|
Target: ctx.Target(),
|
|
})
|
|
|
|
addStubDependencyProviders(ctx)
|
|
|
|
return unstrippedOutputFile
|
|
}
|
|
|
|
func addStubDependencyProviders(ctx ModuleContext) {
|
|
stubs := ctx.GetDirectDepsWithTag(stubImplDepTag)
|
|
if len(stubs) > 0 {
|
|
var stubsInfo []SharedStubLibrary
|
|
for _, stub := range stubs {
|
|
stubInfo, _ := android.OtherModuleProvider(ctx, stub, SharedLibraryInfoProvider)
|
|
flagInfo, _ := android.OtherModuleProvider(ctx, stub, FlagExporterInfoProvider)
|
|
stubsInfo = append(stubsInfo, SharedStubLibrary{
|
|
Version: moduleLibraryInterface(stub).stubsVersion(),
|
|
SharedLibraryInfo: stubInfo,
|
|
FlagExporterInfo: flagInfo,
|
|
})
|
|
}
|
|
android.SetProvider(ctx, SharedLibraryStubsProvider, SharedLibraryStubsInfo{
|
|
SharedStubLibraries: stubsInfo,
|
|
IsLLNDK: ctx.IsLlndk(),
|
|
})
|
|
}
|
|
}
|
|
|
|
func (library *libraryDecorator) unstrippedOutputFilePath() android.Path {
|
|
return library.unstrippedOutputFile
|
|
}
|
|
|
|
func (library *libraryDecorator) strippedAllOutputFilePath() android.Path {
|
|
return library.strippedAllOutputFile
|
|
}
|
|
|
|
func (library *libraryDecorator) disableStripping() {
|
|
library.stripper.StripProperties.Strip.None = BoolPtr(true)
|
|
}
|
|
|
|
func (library *libraryDecorator) nativeCoverage() bool {
|
|
if library.header() || library.buildStubs() {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (library *libraryDecorator) coverageOutputFilePath() android.OptionalPath {
|
|
return library.coverageOutputFile
|
|
}
|
|
|
|
func (library *libraryDecorator) exportedIncludeDirsForAbiCheck(ctx ModuleContext) []string {
|
|
exportIncludeDirs := library.flagExporter.exportedIncludes(ctx).Strings()
|
|
exportIncludeDirs = append(exportIncludeDirs, library.sabi.Properties.ReexportedIncludes...)
|
|
exportSystemIncludeDirs := library.flagExporter.exportedSystemIncludes(ctx).Strings()
|
|
exportSystemIncludeDirs = append(exportSystemIncludeDirs, library.sabi.Properties.ReexportedSystemIncludes...)
|
|
// The ABI checker does not distinguish normal and system headers.
|
|
return append(exportIncludeDirs, exportSystemIncludeDirs...)
|
|
}
|
|
|
|
func (library *libraryDecorator) llndkIncludeDirsForAbiCheck(ctx ModuleContext, deps PathDeps) []string {
|
|
var includeDirs, systemIncludeDirs []string
|
|
|
|
// The ABI checker does not need the preprocess which adds macro guards to function declarations.
|
|
preprocessedDirs := android.PathsForModuleSrc(ctx, library.Properties.Llndk.Export_preprocessed_headers).Strings()
|
|
if Bool(library.Properties.Llndk.Export_headers_as_system) {
|
|
systemIncludeDirs = append(systemIncludeDirs, preprocessedDirs...)
|
|
} else {
|
|
includeDirs = append(includeDirs, preprocessedDirs...)
|
|
}
|
|
|
|
if library.Properties.Llndk.Override_export_include_dirs != nil {
|
|
includeDirs = append(includeDirs, android.PathsForModuleSrc(
|
|
ctx, library.Properties.Llndk.Override_export_include_dirs).Strings()...)
|
|
} else {
|
|
includeDirs = append(includeDirs, library.flagExporter.exportedIncludes(ctx).Strings()...)
|
|
// Ignore library.sabi.Properties.ReexportedIncludes because
|
|
// LLNDK does not reexport the implementation's dependencies, such as export_header_libs.
|
|
}
|
|
|
|
systemIncludeDirs = append(systemIncludeDirs,
|
|
library.flagExporter.exportedSystemIncludes(ctx).Strings()...)
|
|
if Bool(library.Properties.Llndk.Export_headers_as_system) {
|
|
systemIncludeDirs = append(systemIncludeDirs, includeDirs...)
|
|
includeDirs = nil
|
|
}
|
|
// Header libs.
|
|
includeDirs = append(includeDirs, deps.LlndkIncludeDirs.Strings()...)
|
|
systemIncludeDirs = append(systemIncludeDirs, deps.LlndkSystemIncludeDirs.Strings()...)
|
|
// The ABI checker does not distinguish normal and system headers.
|
|
return append(includeDirs, systemIncludeDirs...)
|
|
}
|
|
|
|
func (library *libraryDecorator) linkLlndkSAbiDumpFiles(ctx ModuleContext,
|
|
deps PathDeps, sAbiDumpFiles android.Paths, soFile android.Path, libFileName string,
|
|
excludeSymbolVersions, excludeSymbolTags []string) android.Path {
|
|
// NDK symbols in version 34 are LLNDK symbols. Those in version 35 are not.
|
|
// TODO(b/314010764): Add parameters to read LLNDK symbols from the symbol file.
|
|
return transformDumpToLinkedDump(ctx,
|
|
sAbiDumpFiles, soFile, libFileName+".llndk",
|
|
library.llndkIncludeDirsForAbiCheck(ctx, deps),
|
|
android.OptionalPathForModuleSrc(ctx, library.Properties.Llndk.Symbol_file),
|
|
append([]string{"*_PLATFORM", "*_PRIVATE"}, excludeSymbolVersions...),
|
|
append([]string{"platform-only"}, excludeSymbolTags...),
|
|
"34")
|
|
}
|
|
|
|
func getRefAbiDumpFile(ctx android.ModuleInstallPathContext,
|
|
versionedDumpDir, fileName string) android.OptionalPath {
|
|
|
|
currentArchType := ctx.Arch().ArchType
|
|
primaryArchType := ctx.Config().DevicePrimaryArchType()
|
|
archName := currentArchType.String()
|
|
if currentArchType != primaryArchType {
|
|
archName += "_" + primaryArchType.String()
|
|
}
|
|
|
|
return android.ExistentPathForSource(ctx, versionedDumpDir, archName, "source-based",
|
|
fileName+".lsdump")
|
|
}
|
|
|
|
// Return the previous and current SDK versions for cross-version ABI diff.
|
|
func crossVersionAbiDiffSdkVersions(ctx ModuleContext, dumpDir string) (string, string) {
|
|
sdkVersionInt := ctx.Config().PlatformSdkVersion().FinalInt()
|
|
sdkVersionStr := ctx.Config().PlatformSdkVersion().String()
|
|
|
|
if ctx.Config().PlatformSdkFinal() {
|
|
return strconv.Itoa(sdkVersionInt - 1), sdkVersionStr
|
|
} else {
|
|
// The platform SDK version can be upgraded before finalization while the corresponding abi dumps hasn't
|
|
// been generated. Thus the Cross-Version Check chooses PLATFORM_SDK_VERION - 1 as previous version.
|
|
// This situation could be identified by checking the existence of the PLATFORM_SDK_VERION dump directory.
|
|
versionedDumpDir := android.ExistentPathForSource(ctx, dumpDir, sdkVersionStr)
|
|
if versionedDumpDir.Valid() {
|
|
return sdkVersionStr, strconv.Itoa(sdkVersionInt + 1)
|
|
} else {
|
|
return strconv.Itoa(sdkVersionInt - 1), sdkVersionStr
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return the SDK version for same-version ABI diff.
|
|
func currRefAbiDumpSdkVersion(ctx ModuleContext) string {
|
|
if ctx.Config().PlatformSdkFinal() {
|
|
// After sdk finalization, the ABI of the latest API level must be consistent with the source code,
|
|
// so choose PLATFORM_SDK_VERSION as the current version.
|
|
return ctx.Config().PlatformSdkVersion().String()
|
|
} else {
|
|
return "current"
|
|
}
|
|
}
|
|
|
|
// sourceAbiDiff registers a build statement to compare linked sAbi dump files (.lsdump).
|
|
func (library *libraryDecorator) sourceAbiDiff(ctx android.ModuleContext,
|
|
sourceDump, referenceDump android.Path,
|
|
baseName, nameExt string, isLlndkOrNdk, allowExtensions bool,
|
|
sourceVersion, errorMessage string) {
|
|
|
|
extraFlags := []string{"-target-version", sourceVersion}
|
|
headerAbiChecker := library.getHeaderAbiCheckerProperties(ctx)
|
|
if Bool(headerAbiChecker.Check_all_apis) {
|
|
extraFlags = append(extraFlags, "-check-all-apis")
|
|
} else {
|
|
extraFlags = append(extraFlags,
|
|
"-allow-unreferenced-changes",
|
|
"-allow-unreferenced-elf-symbol-changes")
|
|
}
|
|
if isLlndkOrNdk {
|
|
extraFlags = append(extraFlags, "-consider-opaque-types-different")
|
|
}
|
|
if allowExtensions {
|
|
extraFlags = append(extraFlags, "-allow-extensions")
|
|
}
|
|
extraFlags = append(extraFlags, headerAbiChecker.Diff_flags...)
|
|
|
|
library.sAbiDiff = append(
|
|
library.sAbiDiff,
|
|
transformAbiDumpToAbiDiff(ctx, sourceDump, referenceDump,
|
|
baseName, nameExt, extraFlags, errorMessage))
|
|
}
|
|
|
|
func (library *libraryDecorator) crossVersionAbiDiff(ctx android.ModuleContext,
|
|
sourceDump, referenceDump android.Path,
|
|
baseName string, isLlndkOrNdk bool, sourceVersion, prevVersion string) {
|
|
|
|
errorMessage := "error: Please follow https://android.googlesource.com/platform/development/+/main/vndk/tools/header-checker/README.md#configure-cross_version-abi-check to resolve the ABI difference between your source code and version " + prevVersion + "."
|
|
|
|
library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, prevVersion,
|
|
isLlndkOrNdk, true /* allowExtensions */, sourceVersion, errorMessage)
|
|
}
|
|
|
|
func (library *libraryDecorator) sameVersionAbiDiff(ctx android.ModuleContext,
|
|
sourceDump, referenceDump android.Path,
|
|
baseName, nameExt string, isLlndkOrNdk bool) {
|
|
|
|
libName := strings.TrimSuffix(baseName, filepath.Ext(baseName))
|
|
errorMessage := "error: Please update ABI references with: $$ANDROID_BUILD_TOP/development/vndk/tools/header-checker/utils/create_reference_dumps.py -l " + libName
|
|
|
|
library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
|
|
isLlndkOrNdk, false /* allowExtensions */, "current", errorMessage)
|
|
}
|
|
|
|
func (library *libraryDecorator) optInAbiDiff(ctx android.ModuleContext,
|
|
sourceDump, referenceDump android.Path,
|
|
baseName, nameExt string, refDumpDir string) {
|
|
|
|
libName := strings.TrimSuffix(baseName, filepath.Ext(baseName))
|
|
errorMessage := "error: Please update ABI references with: $$ANDROID_BUILD_TOP/development/vndk/tools/header-checker/utils/create_reference_dumps.py -l " + libName + " -ref-dump-dir $$ANDROID_BUILD_TOP/" + refDumpDir
|
|
// Most opt-in libraries do not have dumps for all default architectures.
|
|
if ctx.Config().HasDeviceProduct() {
|
|
errorMessage += " -products " + ctx.Config().DeviceProduct()
|
|
}
|
|
|
|
library.sourceAbiDiff(ctx, sourceDump, referenceDump, baseName, nameExt,
|
|
false /* isLlndkOrNdk */, false /* allowExtensions */, "current", errorMessage)
|
|
}
|
|
|
|
func (library *libraryDecorator) linkSAbiDumpFiles(ctx ModuleContext, deps PathDeps, objs Objects, fileName string, soFile android.Path) {
|
|
if library.sabi.shouldCreateSourceAbiDump() {
|
|
exportedIncludeDirs := library.exportedIncludeDirsForAbiCheck(ctx)
|
|
headerAbiChecker := library.getHeaderAbiCheckerProperties(ctx)
|
|
currSdkVersion := currRefAbiDumpSdkVersion(ctx)
|
|
currVendorVersion := ctx.Config().VendorApiLevel()
|
|
|
|
// Generate source dumps.
|
|
implDump := transformDumpToLinkedDump(ctx,
|
|
objs.sAbiDumpFiles, soFile, fileName,
|
|
exportedIncludeDirs,
|
|
android.OptionalPathForModuleSrc(ctx, library.symbolFileForAbiCheck(ctx)),
|
|
headerAbiChecker.Exclude_symbol_versions,
|
|
headerAbiChecker.Exclude_symbol_tags,
|
|
currSdkVersion)
|
|
|
|
var llndkDump android.Path
|
|
tags := classifySourceAbiDump(ctx)
|
|
for _, tag := range tags {
|
|
if tag == llndkLsdumpTag {
|
|
if llndkDump == nil {
|
|
// TODO(b/323447559): Evaluate if replacing sAbiDumpFiles with implDump is faster
|
|
llndkDump = library.linkLlndkSAbiDumpFiles(ctx,
|
|
deps, objs.sAbiDumpFiles, soFile, fileName,
|
|
headerAbiChecker.Exclude_symbol_versions,
|
|
headerAbiChecker.Exclude_symbol_tags)
|
|
}
|
|
addLsdumpPath(string(tag) + ":" + llndkDump.String())
|
|
} else {
|
|
addLsdumpPath(string(tag) + ":" + implDump.String())
|
|
}
|
|
}
|
|
|
|
// Diff source dumps and reference dumps.
|
|
for _, tag := range tags {
|
|
dumpDirName := tag.dirName()
|
|
if dumpDirName == "" {
|
|
continue
|
|
}
|
|
dumpDir := filepath.Join("prebuilts", "abi-dumps", dumpDirName)
|
|
isLlndk := (tag == llndkLsdumpTag)
|
|
isNdk := (tag == ndkLsdumpTag)
|
|
binderBitness := ctx.DeviceConfig().BinderBitness()
|
|
nameExt := ""
|
|
if isLlndk {
|
|
nameExt = "llndk"
|
|
}
|
|
// Check against the previous version.
|
|
var prevVersion, currVersion string
|
|
sourceDump := implDump
|
|
// If this release config does not define VendorApiLevel, fall back to the old policy.
|
|
if isLlndk && currVendorVersion != "" {
|
|
prevVersion = ctx.Config().PrevVendorApiLevel()
|
|
currVersion = currVendorVersion
|
|
// LLNDK dumps are generated by different rules after trunk stable.
|
|
if android.IsTrunkStableVendorApiLevel(prevVersion) {
|
|
sourceDump = llndkDump
|
|
}
|
|
} else {
|
|
prevVersion, currVersion = crossVersionAbiDiffSdkVersions(ctx, dumpDir)
|
|
}
|
|
prevDumpDir := filepath.Join(dumpDir, prevVersion, binderBitness)
|
|
prevDumpFile := getRefAbiDumpFile(ctx, prevDumpDir, fileName)
|
|
if prevDumpFile.Valid() {
|
|
library.crossVersionAbiDiff(ctx, sourceDump, prevDumpFile.Path(),
|
|
fileName, isLlndk || isNdk, currVersion, nameExt+prevVersion)
|
|
}
|
|
// Check against the current version.
|
|
sourceDump = implDump
|
|
if isLlndk && currVendorVersion != "" {
|
|
currVersion = currVendorVersion
|
|
if android.IsTrunkStableVendorApiLevel(currVersion) {
|
|
sourceDump = llndkDump
|
|
}
|
|
} else {
|
|
currVersion = currSdkVersion
|
|
}
|
|
currDumpDir := filepath.Join(dumpDir, currVersion, binderBitness)
|
|
currDumpFile := getRefAbiDumpFile(ctx, currDumpDir, fileName)
|
|
if currDumpFile.Valid() {
|
|
library.sameVersionAbiDiff(ctx, sourceDump, currDumpFile.Path(),
|
|
fileName, nameExt, isLlndk || isNdk)
|
|
}
|
|
}
|
|
// Check against the opt-in reference dumps.
|
|
for i, optInDumpDir := range headerAbiChecker.Ref_dump_dirs {
|
|
optInDumpDirPath := android.PathForModuleSrc(ctx, optInDumpDir)
|
|
// Ref_dump_dirs are not versioned.
|
|
// They do not contain subdir for binder bitness because 64-bit binder has been mandatory.
|
|
optInDumpFile := getRefAbiDumpFile(ctx, optInDumpDirPath.String(), fileName)
|
|
if !optInDumpFile.Valid() {
|
|
continue
|
|
}
|
|
library.optInAbiDiff(ctx,
|
|
implDump, optInDumpFile.Path(),
|
|
fileName, "opt"+strconv.Itoa(i), optInDumpDirPath.String())
|
|
}
|
|
}
|
|
}
|
|
|
|
func processLLNDKHeaders(ctx ModuleContext, srcHeaderDir string, outDir android.ModuleGenPath) (timestamp android.Path, installPaths android.WritablePaths) {
|
|
srcDir := android.PathForModuleSrc(ctx, srcHeaderDir)
|
|
srcFiles := ctx.GlobFiles(filepath.Join(srcDir.String(), "**/*.h"), nil)
|
|
|
|
for _, header := range srcFiles {
|
|
headerDir := filepath.Dir(header.String())
|
|
relHeaderDir, err := filepath.Rel(srcDir.String(), headerDir)
|
|
if err != nil {
|
|
ctx.ModuleErrorf("filepath.Rel(%q, %q) failed: %s",
|
|
srcDir.String(), headerDir, err)
|
|
continue
|
|
}
|
|
|
|
installPaths = append(installPaths, outDir.Join(ctx, relHeaderDir, header.Base()))
|
|
}
|
|
|
|
return processHeadersWithVersioner(ctx, srcDir, outDir, srcFiles, installPaths), installPaths
|
|
}
|
|
|
|
// link registers actions to link this library, and sets various fields
|
|
// on this library to reflect information that should be exported up the build
|
|
// tree (for example, exported flags and include paths).
|
|
func (library *libraryDecorator) link(ctx ModuleContext,
|
|
flags Flags, deps PathDeps, objs Objects) android.Path {
|
|
|
|
if ctx.IsLlndk() {
|
|
if len(library.Properties.Llndk.Export_preprocessed_headers) > 0 {
|
|
// This is the vendor variant of an LLNDK library with preprocessed headers.
|
|
genHeaderOutDir := android.PathForModuleGen(ctx, "include")
|
|
|
|
var timestampFiles android.Paths
|
|
for _, dir := range library.Properties.Llndk.Export_preprocessed_headers {
|
|
timestampFile, installPaths := processLLNDKHeaders(ctx, dir, genHeaderOutDir)
|
|
timestampFiles = append(timestampFiles, timestampFile)
|
|
library.addExportedGeneratedHeaders(installPaths.Paths()...)
|
|
}
|
|
|
|
if Bool(library.Properties.Llndk.Export_headers_as_system) {
|
|
library.reexportSystemDirs(genHeaderOutDir)
|
|
} else {
|
|
library.reexportDirs(genHeaderOutDir)
|
|
}
|
|
|
|
library.reexportDeps(timestampFiles...)
|
|
}
|
|
|
|
// override the module's export_include_dirs with llndk.override_export_include_dirs
|
|
// if it is set.
|
|
if override := library.Properties.Llndk.Override_export_include_dirs; override != nil {
|
|
library.flagExporter.Properties.Export_include_dirs = override
|
|
}
|
|
|
|
if Bool(library.Properties.Llndk.Export_headers_as_system) {
|
|
library.flagExporter.Properties.Export_system_include_dirs = append(
|
|
library.flagExporter.Properties.Export_system_include_dirs,
|
|
library.flagExporter.Properties.Export_include_dirs...)
|
|
library.flagExporter.Properties.Export_include_dirs = nil
|
|
}
|
|
}
|
|
|
|
if ctx.IsVendorPublicLibrary() {
|
|
// override the module's export_include_dirs with vendor_public_library.override_export_include_dirs
|
|
// if it is set.
|
|
if override := library.Properties.Vendor_public_library.Override_export_include_dirs; override != nil {
|
|
library.flagExporter.Properties.Export_include_dirs = override
|
|
}
|
|
}
|
|
|
|
// Linking this library consists of linking `deps.Objs` (.o files in dependencies
|
|
// of this library), together with `objs` (.o files created by compiling this
|
|
// library).
|
|
objs = deps.Objs.Copy().Append(objs)
|
|
var out android.Path
|
|
if library.static() || library.header() {
|
|
out = library.linkStatic(ctx, flags, deps, objs)
|
|
} else {
|
|
out = library.linkShared(ctx, flags, deps, objs)
|
|
}
|
|
|
|
// Export include paths and flags to be propagated up the tree.
|
|
library.exportIncludes(ctx)
|
|
library.reexportDirs(deps.ReexportedDirs...)
|
|
library.reexportSystemDirs(deps.ReexportedSystemDirs...)
|
|
library.reexportFlags(deps.ReexportedFlags...)
|
|
library.reexportDeps(deps.ReexportedDeps...)
|
|
library.addExportedGeneratedHeaders(deps.ReexportedGeneratedHeaders...)
|
|
|
|
// Optionally export aidl headers.
|
|
if Bool(library.Properties.Aidl.Export_aidl_headers) {
|
|
if library.baseCompiler.hasAidl(deps) {
|
|
if library.baseCompiler.hasSrcExt(".aidl") {
|
|
dir := android.PathForModuleGen(ctx, "aidl")
|
|
library.reexportDirs(dir)
|
|
}
|
|
if len(deps.AidlLibraryInfos) > 0 {
|
|
dir := android.PathForModuleGen(ctx, "aidl_library")
|
|
library.reexportDirs(dir)
|
|
}
|
|
|
|
library.reexportDeps(library.baseCompiler.aidlOrderOnlyDeps...)
|
|
library.addExportedGeneratedHeaders(library.baseCompiler.aidlHeaders...)
|
|
}
|
|
}
|
|
|
|
// Optionally export proto headers.
|
|
if Bool(library.Properties.Proto.Export_proto_headers) {
|
|
if library.baseCompiler.hasSrcExt(".proto") {
|
|
var includes android.Paths
|
|
if flags.proto.CanonicalPathFromRoot {
|
|
includes = append(includes, flags.proto.SubDir)
|
|
}
|
|
includes = append(includes, flags.proto.Dir)
|
|
library.reexportDirs(includes...)
|
|
|
|
library.reexportDeps(library.baseCompiler.protoOrderOnlyDeps...)
|
|
library.addExportedGeneratedHeaders(library.baseCompiler.protoHeaders...)
|
|
}
|
|
}
|
|
|
|
// If the library is sysprop_library, expose either public or internal header selectively.
|
|
if library.baseCompiler.hasSrcExt(".sysprop") {
|
|
dir := android.PathForModuleGen(ctx, "sysprop", "include")
|
|
if library.Properties.Sysprop.Platform != nil {
|
|
isOwnerPlatform := Bool(library.Properties.Sysprop.Platform)
|
|
|
|
// If the owner is different from the user, expose public header. That is,
|
|
// 1) if the user is product (as owner can only be platform / vendor)
|
|
// 2) if the owner is platform and the client is vendor
|
|
// We don't care Platform -> Vendor dependency as it's already forbidden.
|
|
if ctx.Device() && (ctx.ProductSpecific() || (isOwnerPlatform && ctx.inVendor())) {
|
|
dir = android.PathForModuleGen(ctx, "sysprop/public", "include")
|
|
}
|
|
}
|
|
|
|
// Make sure to only export headers which are within the include directory.
|
|
_, headers := android.FilterPathListPredicate(library.baseCompiler.syspropHeaders, func(path android.Path) bool {
|
|
_, isRel := android.MaybeRel(ctx, dir.String(), path.String())
|
|
return isRel
|
|
})
|
|
|
|
// Add sysprop-related directories to the exported directories of this library.
|
|
library.reexportDirs(dir)
|
|
library.reexportDeps(library.baseCompiler.syspropOrderOnlyDeps...)
|
|
library.addExportedGeneratedHeaders(headers...)
|
|
}
|
|
|
|
// Add stub-related flags if this library is a stub library.
|
|
library.exportVersioningMacroIfNeeded(ctx)
|
|
|
|
// Propagate a Provider containing information about exported flags, deps, and include paths.
|
|
library.flagExporter.setProvider(ctx)
|
|
|
|
return out
|
|
}
|
|
|
|
func (library *libraryDecorator) exportVersioningMacroIfNeeded(ctx android.BaseModuleContext) {
|
|
if library.buildStubs() && library.stubsVersion() != "" && !library.skipAPIDefine {
|
|
name := versioningMacroName(ctx.Module().(*Module).ImplementationModuleName(ctx))
|
|
apiLevel, err := android.ApiLevelFromUser(ctx, library.stubsVersion())
|
|
if err != nil {
|
|
ctx.ModuleErrorf("Can't export version macro: %s", err.Error())
|
|
}
|
|
library.reexportFlags("-D" + name + "=" + strconv.Itoa(apiLevel.FinalOrPreviewInt()))
|
|
}
|
|
}
|
|
|
|
// buildStatic returns true if this library should be built as a static library.
|
|
func (library *libraryDecorator) buildStatic() bool {
|
|
return library.MutatedProperties.BuildStatic &&
|
|
BoolDefault(library.StaticProperties.Static.Enabled, true)
|
|
}
|
|
|
|
// buildShared returns true if this library should be built as a shared library.
|
|
func (library *libraryDecorator) buildShared() bool {
|
|
return library.MutatedProperties.BuildShared &&
|
|
BoolDefault(library.SharedProperties.Shared.Enabled, true)
|
|
}
|
|
|
|
func (library *libraryDecorator) objs() Objects {
|
|
return library.objects
|
|
}
|
|
|
|
func (library *libraryDecorator) reuseObjs() Objects {
|
|
return library.reuseObjects
|
|
}
|
|
|
|
func (library *libraryDecorator) toc() android.OptionalPath {
|
|
return library.tocFile
|
|
}
|
|
|
|
func (library *libraryDecorator) installSymlinkToRuntimeApex(ctx ModuleContext, file android.Path) {
|
|
dir := library.baseInstaller.installDir(ctx)
|
|
dirOnDevice := android.InstallPathToOnDevicePath(ctx, dir)
|
|
// libc_hwasan has relative_install_dir set, which would mess up the dir.Base() logic.
|
|
// hardcode here because it's the only target, if we have other targets that use this
|
|
// we can generalise this.
|
|
var target string
|
|
if ctx.baseModuleName() == "libc_hwasan" {
|
|
target = "/" + filepath.Join("apex", "com.android.runtime", "lib64", "bionic", "hwasan", file.Base())
|
|
} else {
|
|
base := dir.Base()
|
|
target = "/" + filepath.Join("apex", "com.android.runtime", base, "bionic", file.Base())
|
|
}
|
|
ctx.InstallAbsoluteSymlink(dir, file.Base(), target)
|
|
library.postInstallCmds = append(library.postInstallCmds, makeSymlinkCmd(dirOnDevice, file.Base(), target))
|
|
}
|
|
|
|
func (library *libraryDecorator) install(ctx ModuleContext, file android.Path) {
|
|
if library.shared() {
|
|
if ctx.Device() && ctx.useVndk() {
|
|
// set subDir for VNDK extensions
|
|
if ctx.IsVndkExt() {
|
|
if ctx.isVndkSp() {
|
|
library.baseInstaller.subDir = "vndk-sp"
|
|
} else {
|
|
library.baseInstaller.subDir = "vndk"
|
|
}
|
|
}
|
|
|
|
// In some cases we want to use core variant for VNDK-Core libs.
|
|
// Skip product variant since VNDKs use only the vendor variant.
|
|
if ctx.isVndk() && !ctx.isVndkSp() && !ctx.IsVndkExt() && !ctx.inProduct() {
|
|
mayUseCoreVariant := true
|
|
|
|
if ctx.mustUseVendorVariant() {
|
|
mayUseCoreVariant = false
|
|
}
|
|
|
|
if ctx.Config().CFIEnabledForPath(ctx.ModuleDir()) {
|
|
mayUseCoreVariant = false
|
|
}
|
|
|
|
if mayUseCoreVariant {
|
|
library.checkSameCoreVariant = true
|
|
if ctx.DeviceConfig().VndkUseCoreVariant() {
|
|
library.useCoreVariant = true
|
|
}
|
|
}
|
|
}
|
|
|
|
// do not install vndk libs
|
|
// vndk libs are packaged into VNDK APEX
|
|
if ctx.isVndk() && !ctx.IsVndkExt() && !ctx.Config().IsVndkDeprecated() && !ctx.inProduct() {
|
|
return
|
|
}
|
|
} else if library.hasStubsVariants() && !ctx.Host() && ctx.directlyInAnyApex() {
|
|
// Bionic libraries (e.g. libc.so) is installed to the bootstrap subdirectory.
|
|
// The original path becomes a symlink to the corresponding file in the
|
|
// runtime APEX.
|
|
translatedArch := ctx.Target().NativeBridge == android.NativeBridgeEnabled
|
|
if InstallToBootstrap(ctx.baseModuleName(), ctx.Config()) && !library.buildStubs() &&
|
|
!translatedArch && !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() {
|
|
if ctx.Device() {
|
|
library.installSymlinkToRuntimeApex(ctx, file)
|
|
}
|
|
library.baseInstaller.subDir = "bootstrap"
|
|
}
|
|
} else if ctx.directlyInAnyApex() && ctx.IsLlndk() && !isBionic(ctx.baseModuleName()) {
|
|
// Skip installing LLNDK (non-bionic) libraries moved to APEX.
|
|
ctx.Module().HideFromMake()
|
|
}
|
|
|
|
library.baseInstaller.install(ctx, file)
|
|
}
|
|
|
|
if Bool(library.Properties.Static_ndk_lib) && library.static() &&
|
|
!ctx.InVendorOrProduct() && !ctx.inRamdisk() && !ctx.inVendorRamdisk() && !ctx.inRecovery() && ctx.Device() &&
|
|
library.baseLinker.sanitize.isUnsanitizedVariant() &&
|
|
ctx.isForPlatform() && !ctx.isPreventInstall() {
|
|
installPath := getUnversionedLibraryInstallPath(ctx).Join(ctx, file.Base())
|
|
|
|
ctx.ModuleBuild(pctx, android.ModuleBuildParams{
|
|
Rule: android.Cp,
|
|
Description: "install " + installPath.Base(),
|
|
Output: installPath,
|
|
Input: file,
|
|
})
|
|
|
|
library.ndkSysrootPath = installPath
|
|
}
|
|
}
|
|
|
|
func (library *libraryDecorator) everInstallable() bool {
|
|
// Only shared and static libraries are installed. Header libraries (which are
|
|
// neither static or shared) are not installed.
|
|
return library.shared() || library.static()
|
|
}
|
|
|
|
// static returns true if this library is for a "static' variant.
|
|
func (library *libraryDecorator) static() bool {
|
|
return library.MutatedProperties.VariantIsStatic
|
|
}
|
|
|
|
// shared returns true if this library is for a "shared' variant.
|
|
func (library *libraryDecorator) shared() bool {
|
|
return library.MutatedProperties.VariantIsShared
|
|
}
|
|
|
|
// header returns true if this library is for a header-only variant.
|
|
func (library *libraryDecorator) header() bool {
|
|
// Neither "static" nor "shared" implies this library is header-only.
|
|
return !library.static() && !library.shared()
|
|
}
|
|
|
|
// setStatic marks the library variant as "static".
|
|
func (library *libraryDecorator) setStatic() {
|
|
library.MutatedProperties.VariantIsStatic = true
|
|
library.MutatedProperties.VariantIsShared = false
|
|
}
|
|
|
|
// setShared marks the library variant as "shared".
|
|
func (library *libraryDecorator) setShared() {
|
|
library.MutatedProperties.VariantIsStatic = false
|
|
library.MutatedProperties.VariantIsShared = true
|
|
}
|
|
|
|
// BuildOnlyStatic disables building this library as a shared library.
|
|
func (library *libraryDecorator) BuildOnlyStatic() {
|
|
library.MutatedProperties.BuildShared = false
|
|
}
|
|
|
|
// BuildOnlyShared disables building this library as a static library.
|
|
func (library *libraryDecorator) BuildOnlyShared() {
|
|
library.MutatedProperties.BuildStatic = false
|
|
}
|
|
|
|
// HeaderOnly disables building this library as a shared or static library;
|
|
// the library only exists to propagate header file dependencies up the build graph.
|
|
func (library *libraryDecorator) HeaderOnly() {
|
|
library.MutatedProperties.BuildShared = false
|
|
library.MutatedProperties.BuildStatic = false
|
|
}
|
|
|
|
// hasLLNDKStubs returns true if this cc_library module has a variant that will build LLNDK stubs.
|
|
func (library *libraryDecorator) hasLLNDKStubs() bool {
|
|
return String(library.Properties.Llndk.Symbol_file) != ""
|
|
}
|
|
|
|
// hasLLNDKStubs returns true if this cc_library module has a variant that will build LLNDK stubs.
|
|
func (library *libraryDecorator) hasLLNDKHeaders() bool {
|
|
return Bool(library.Properties.Llndk.Llndk_headers)
|
|
}
|
|
|
|
// hasVendorPublicLibrary returns true if this cc_library module has a variant that will build
|
|
// vendor public library stubs.
|
|
func (library *libraryDecorator) hasVendorPublicLibrary() bool {
|
|
return String(library.Properties.Vendor_public_library.Symbol_file) != ""
|
|
}
|
|
|
|
func (library *libraryDecorator) implementationModuleName(name string) string {
|
|
return name
|
|
}
|
|
|
|
func (library *libraryDecorator) buildStubs() bool {
|
|
return library.MutatedProperties.BuildStubs
|
|
}
|
|
|
|
func (library *libraryDecorator) symbolFileForAbiCheck(ctx ModuleContext) *string {
|
|
if props := library.getHeaderAbiCheckerProperties(ctx); props.Symbol_file != nil {
|
|
return props.Symbol_file
|
|
}
|
|
if library.hasStubsVariants() && library.Properties.Stubs.Symbol_file != nil {
|
|
return library.Properties.Stubs.Symbol_file
|
|
}
|
|
// TODO(b/309880485): Distinguish platform, NDK, LLNDK, and APEX version scripts.
|
|
if library.baseLinker.Properties.Version_script != nil {
|
|
return library.baseLinker.Properties.Version_script
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (library *libraryDecorator) hasStubsVariants() bool {
|
|
// Just having stubs.symbol_file is enough to create a stub variant. In that case
|
|
// the stub for the future API level is created.
|
|
return library.Properties.Stubs.Symbol_file != nil ||
|
|
len(library.Properties.Stubs.Versions) > 0
|
|
}
|
|
|
|
func (library *libraryDecorator) isStubsImplementationRequired() bool {
|
|
return BoolDefault(library.Properties.Stubs.Implementation_installable, true)
|
|
}
|
|
|
|
func (library *libraryDecorator) stubsVersions(ctx android.BaseMutatorContext) []string {
|
|
if !library.hasStubsVariants() {
|
|
return nil
|
|
}
|
|
|
|
if library.hasLLNDKStubs() && ctx.Module().(*Module).InVendorOrProduct() {
|
|
// LLNDK libraries only need a single stubs variant (""), which is
|
|
// added automatically in createVersionVariations().
|
|
return nil
|
|
}
|
|
|
|
// Future API level is implicitly added if there isn't
|
|
versions := addCurrentVersionIfNotPresent(library.Properties.Stubs.Versions)
|
|
normalizeVersions(ctx, versions)
|
|
return versions
|
|
}
|
|
|
|
func addCurrentVersionIfNotPresent(vers []string) []string {
|
|
if inList(android.FutureApiLevel.String(), vers) {
|
|
return vers
|
|
}
|
|
// In some cases, people use the raw value "10000" in the versions property.
|
|
// We shouldn't add the future API level in that case, otherwise there will
|
|
// be two identical versions.
|
|
if inList(strconv.Itoa(android.FutureApiLevel.FinalOrFutureInt()), vers) {
|
|
return vers
|
|
}
|
|
return append(vers, android.FutureApiLevel.String())
|
|
}
|
|
|
|
func (library *libraryDecorator) setStubsVersion(version string) {
|
|
library.MutatedProperties.StubsVersion = version
|
|
}
|
|
|
|
func (library *libraryDecorator) stubsVersion() string {
|
|
return library.MutatedProperties.StubsVersion
|
|
}
|
|
|
|
func (library *libraryDecorator) setBuildStubs(isLatest bool) {
|
|
library.MutatedProperties.BuildStubs = true
|
|
library.MutatedProperties.IsLatestVersion = isLatest
|
|
}
|
|
|
|
func (library *libraryDecorator) setAllStubsVersions(versions []string) {
|
|
library.MutatedProperties.AllStubsVersions = versions
|
|
}
|
|
|
|
func (library *libraryDecorator) allStubsVersions() []string {
|
|
return library.MutatedProperties.AllStubsVersions
|
|
}
|
|
|
|
func (library *libraryDecorator) isLatestStubVersion() bool {
|
|
return library.MutatedProperties.IsLatestVersion
|
|
}
|
|
|
|
func (library *libraryDecorator) availableFor(what string) bool {
|
|
var list []string
|
|
if library.static() {
|
|
list = library.StaticProperties.Static.Apex_available
|
|
} else if library.shared() {
|
|
list = library.SharedProperties.Shared.Apex_available
|
|
}
|
|
if len(list) == 0 {
|
|
return false
|
|
}
|
|
return android.CheckAvailableForApex(what, list)
|
|
}
|
|
|
|
func (library *libraryDecorator) installable() *bool {
|
|
if library.static() {
|
|
return library.StaticProperties.Static.Installable
|
|
} else if library.shared() {
|
|
return library.SharedProperties.Shared.Installable
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (library *libraryDecorator) makeUninstallable(mod *Module) {
|
|
if library.static() && library.buildStatic() && !library.buildStubs() {
|
|
// If we're asked to make a static library uninstallable we don't do
|
|
// anything since AndroidMkEntries always sets LOCAL_UNINSTALLABLE_MODULE
|
|
// for these entries. This is done to still get the make targets for NOTICE
|
|
// files from notice_files.mk, which other libraries might depend on.
|
|
return
|
|
}
|
|
mod.ModuleBase.MakeUninstallable()
|
|
}
|
|
|
|
func (library *libraryDecorator) getPartition() string {
|
|
return library.path.Partition()
|
|
}
|
|
|
|
func (library *libraryDecorator) getAPIListCoverageXMLPath() android.ModuleOutPath {
|
|
return library.apiListCoverageXmlPath
|
|
}
|
|
|
|
func (library *libraryDecorator) overriddenModules() []string {
|
|
return library.Properties.Overrides
|
|
}
|
|
|
|
var _ overridable = (*libraryDecorator)(nil)
|
|
|
|
var versioningMacroNamesListKey = android.NewOnceKey("versioningMacroNamesList")
|
|
|
|
// versioningMacroNamesList returns a singleton map, where keys are "version macro names",
|
|
// and values are the module name responsible for registering the version macro name.
|
|
//
|
|
// Version macros are used when building against stubs, to provide version information about
|
|
// the stub. Only stub libraries should have an entry in this list.
|
|
//
|
|
// For example, when building against libFoo#ver, __LIBFOO_API__ macro is set to ver so
|
|
// that headers from libFoo can be conditionally compiled (this may hide APIs
|
|
// that are not available for the version).
|
|
//
|
|
// This map is used to ensure that there aren't conflicts between these version macro names.
|
|
func versioningMacroNamesList(config android.Config) *map[string]string {
|
|
return config.Once(versioningMacroNamesListKey, func() interface{} {
|
|
m := make(map[string]string)
|
|
return &m
|
|
}).(*map[string]string)
|
|
}
|
|
|
|
// alphanumeric and _ characters are preserved.
|
|
// other characters are all converted to _
|
|
var charsNotForMacro = regexp.MustCompile("[^a-zA-Z0-9_]+")
|
|
|
|
// versioningMacroName returns the canonical version macro name for the given module.
|
|
func versioningMacroName(moduleName string) string {
|
|
macroName := charsNotForMacro.ReplaceAllString(moduleName, "_")
|
|
macroName = strings.ToUpper(macroName)
|
|
return "__" + macroName + "_API__"
|
|
}
|
|
|
|
// NewLibrary builds and returns a new Module corresponding to a C++ library.
|
|
// Individual module implementations which comprise a C++ library (or something like
|
|
// a C++ library) should call this function, set some fields on the result, and
|
|
// then call the Init function.
|
|
func NewLibrary(hod android.HostOrDeviceSupported) (*Module, *libraryDecorator) {
|
|
module := newModule(hod, android.MultilibBoth)
|
|
|
|
library := &libraryDecorator{
|
|
MutatedProperties: LibraryMutatedProperties{
|
|
BuildShared: true,
|
|
BuildStatic: true,
|
|
},
|
|
baseCompiler: NewBaseCompiler(),
|
|
baseLinker: NewBaseLinker(module.sanitize),
|
|
baseInstaller: NewBaseInstaller("lib", "lib64", InstallInSystem),
|
|
sabi: module.sabi,
|
|
}
|
|
|
|
module.compiler = library
|
|
module.linker = library
|
|
module.installer = library
|
|
module.library = library
|
|
|
|
return module, library
|
|
}
|
|
|
|
// connects a shared library to a static library in order to reuse its .o files to avoid
|
|
// compiling source files twice.
|
|
func reuseStaticLibrary(mctx android.BottomUpMutatorContext, static, shared *Module) {
|
|
if staticCompiler, ok := static.compiler.(*libraryDecorator); ok {
|
|
sharedCompiler := shared.compiler.(*libraryDecorator)
|
|
|
|
// Check libraries in addition to cflags, since libraries may be exporting different
|
|
// include directories.
|
|
if len(staticCompiler.StaticProperties.Static.Cflags) == 0 &&
|
|
len(sharedCompiler.SharedProperties.Shared.Cflags) == 0 &&
|
|
len(staticCompiler.StaticProperties.Static.Whole_static_libs) == 0 &&
|
|
len(sharedCompiler.SharedProperties.Shared.Whole_static_libs) == 0 &&
|
|
len(staticCompiler.StaticProperties.Static.Static_libs) == 0 &&
|
|
len(sharedCompiler.SharedProperties.Shared.Static_libs) == 0 &&
|
|
len(staticCompiler.StaticProperties.Static.Shared_libs) == 0 &&
|
|
len(sharedCompiler.SharedProperties.Shared.Shared_libs) == 0 &&
|
|
// Compare System_shared_libs properties with nil because empty lists are
|
|
// semantically significant for them.
|
|
staticCompiler.StaticProperties.Static.System_shared_libs == nil &&
|
|
sharedCompiler.SharedProperties.Shared.System_shared_libs == nil {
|
|
|
|
mctx.AddInterVariantDependency(reuseObjTag, shared, static)
|
|
sharedCompiler.baseCompiler.Properties.OriginalSrcs =
|
|
sharedCompiler.baseCompiler.Properties.Srcs
|
|
sharedCompiler.baseCompiler.Properties.Srcs = nil
|
|
sharedCompiler.baseCompiler.Properties.Generated_sources = nil
|
|
}
|
|
|
|
// This dep is just to reference static variant from shared variant
|
|
mctx.AddInterVariantDependency(staticVariantTag, shared, static)
|
|
}
|
|
}
|
|
|
|
// LinkageMutator adds "static" or "shared" variants for modules depending
|
|
// on whether the module can be built as a static library or a shared library.
|
|
func LinkageMutator(mctx android.BottomUpMutatorContext) {
|
|
ccPrebuilt := false
|
|
if m, ok := mctx.Module().(*Module); ok && m.linker != nil {
|
|
_, ccPrebuilt = m.linker.(prebuiltLibraryInterface)
|
|
}
|
|
if ccPrebuilt {
|
|
library := mctx.Module().(*Module).linker.(prebuiltLibraryInterface)
|
|
|
|
// Differentiate between header only and building an actual static/shared library
|
|
buildStatic := library.buildStatic()
|
|
buildShared := library.buildShared()
|
|
if buildStatic || buildShared {
|
|
// Always create both the static and shared variants for prebuilt libraries, and then disable the one
|
|
// that is not being used. This allows them to share the name of a cc_library module, which requires that
|
|
// all the variants of the cc_library also exist on the prebuilt.
|
|
modules := mctx.CreateLocalVariations("static", "shared")
|
|
static := modules[0].(*Module)
|
|
shared := modules[1].(*Module)
|
|
|
|
static.linker.(prebuiltLibraryInterface).setStatic()
|
|
shared.linker.(prebuiltLibraryInterface).setShared()
|
|
|
|
if buildShared {
|
|
mctx.AliasVariation("shared")
|
|
} else if buildStatic {
|
|
mctx.AliasVariation("static")
|
|
}
|
|
|
|
if !buildStatic {
|
|
static.linker.(prebuiltLibraryInterface).disablePrebuilt()
|
|
}
|
|
if !buildShared {
|
|
shared.linker.(prebuiltLibraryInterface).disablePrebuilt()
|
|
}
|
|
} else {
|
|
// Header only
|
|
}
|
|
|
|
} else if library, ok := mctx.Module().(LinkableInterface); ok && library.CcLibraryInterface() {
|
|
|
|
// Non-cc.Modules may need an empty variant for their mutators.
|
|
variations := []string{}
|
|
if library.NonCcVariants() {
|
|
variations = append(variations, "")
|
|
}
|
|
|
|
isLLNDK := false
|
|
if m, ok := mctx.Module().(*Module); ok {
|
|
isLLNDK = m.IsLlndk()
|
|
}
|
|
buildStatic := library.BuildStaticVariant() && !isLLNDK
|
|
buildShared := library.BuildSharedVariant()
|
|
if buildStatic && buildShared {
|
|
variations := append([]string{"static", "shared"}, variations...)
|
|
|
|
modules := mctx.CreateLocalVariations(variations...)
|
|
static := modules[0].(LinkableInterface)
|
|
shared := modules[1].(LinkableInterface)
|
|
|
|
static.SetStatic()
|
|
shared.SetShared()
|
|
|
|
if _, ok := library.(*Module); ok {
|
|
reuseStaticLibrary(mctx, static.(*Module), shared.(*Module))
|
|
}
|
|
mctx.AliasVariation("shared")
|
|
} else if buildStatic {
|
|
variations := append([]string{"static"}, variations...)
|
|
|
|
modules := mctx.CreateLocalVariations(variations...)
|
|
modules[0].(LinkableInterface).SetStatic()
|
|
mctx.AliasVariation("static")
|
|
} else if buildShared {
|
|
variations := append([]string{"shared"}, variations...)
|
|
|
|
modules := mctx.CreateLocalVariations(variations...)
|
|
modules[0].(LinkableInterface).SetShared()
|
|
mctx.AliasVariation("shared")
|
|
} else if len(variations) > 0 {
|
|
mctx.CreateLocalVariations(variations...)
|
|
mctx.AliasVariation(variations[0])
|
|
}
|
|
}
|
|
}
|
|
|
|
// normalizeVersions modifies `versions` in place, so that each raw version
|
|
// string becomes its normalized canonical form.
|
|
// Validates that the versions in `versions` are specified in least to greatest order.
|
|
func normalizeVersions(ctx android.BaseModuleContext, versions []string) {
|
|
var previous android.ApiLevel
|
|
for i, v := range versions {
|
|
ver, err := android.ApiLevelFromUser(ctx, v)
|
|
if err != nil {
|
|
ctx.PropertyErrorf("versions", "%s", err.Error())
|
|
return
|
|
}
|
|
if i > 0 && ver.LessThanOrEqualTo(previous) {
|
|
ctx.PropertyErrorf("versions", "not sorted: %v", versions)
|
|
}
|
|
versions[i] = ver.String()
|
|
previous = ver
|
|
}
|
|
}
|
|
|
|
func createVersionVariations(mctx android.BottomUpMutatorContext, versions []string) {
|
|
// "" is for the non-stubs (implementation) variant for system modules, or the LLNDK variant
|
|
// for LLNDK modules.
|
|
variants := append(android.CopyOf(versions), "")
|
|
|
|
m := mctx.Module().(*Module)
|
|
isLLNDK := m.IsLlndk()
|
|
isVendorPublicLibrary := m.IsVendorPublicLibrary()
|
|
isImportedApiLibrary := m.isImportedApiLibrary()
|
|
|
|
modules := mctx.CreateLocalVariations(variants...)
|
|
for i, m := range modules {
|
|
|
|
if variants[i] != "" || isLLNDK || isVendorPublicLibrary || isImportedApiLibrary {
|
|
// A stubs or LLNDK stubs variant.
|
|
c := m.(*Module)
|
|
c.sanitize = nil
|
|
c.stl = nil
|
|
c.Properties.PreventInstall = true
|
|
lib := moduleLibraryInterface(m)
|
|
isLatest := i == (len(versions) - 1)
|
|
lib.setBuildStubs(isLatest)
|
|
|
|
if variants[i] != "" {
|
|
// A non-LLNDK stubs module is hidden from make and has a dependency from the
|
|
// implementation module to the stubs module.
|
|
c.Properties.HideFromMake = true
|
|
lib.setStubsVersion(variants[i])
|
|
mctx.AddInterVariantDependency(stubImplDepTag, modules[len(modules)-1], modules[i])
|
|
}
|
|
}
|
|
}
|
|
mctx.AliasVariation("")
|
|
latestVersion := ""
|
|
if len(versions) > 0 {
|
|
latestVersion = versions[len(versions)-1]
|
|
}
|
|
mctx.CreateAliasVariation("latest", latestVersion)
|
|
}
|
|
|
|
func createPerApiVersionVariations(mctx android.BottomUpMutatorContext, minSdkVersion string) {
|
|
from, err := nativeApiLevelFromUser(mctx, minSdkVersion)
|
|
if err != nil {
|
|
mctx.PropertyErrorf("min_sdk_version", err.Error())
|
|
return
|
|
}
|
|
|
|
versionStrs := ndkLibraryVersions(mctx, from)
|
|
modules := mctx.CreateLocalVariations(versionStrs...)
|
|
|
|
for i, module := range modules {
|
|
module.(*Module).Properties.Sdk_version = StringPtr(versionStrs[i])
|
|
module.(*Module).Properties.Min_sdk_version = StringPtr(versionStrs[i])
|
|
}
|
|
}
|
|
|
|
func canBeOrLinkAgainstVersionVariants(module interface {
|
|
Host() bool
|
|
InRamdisk() bool
|
|
InVendorRamdisk() bool
|
|
}) bool {
|
|
return !module.Host() && !module.InRamdisk() && !module.InVendorRamdisk()
|
|
}
|
|
|
|
func canBeVersionVariant(module interface {
|
|
Host() bool
|
|
InRamdisk() bool
|
|
InVendorRamdisk() bool
|
|
CcLibraryInterface() bool
|
|
Shared() bool
|
|
}) bool {
|
|
return canBeOrLinkAgainstVersionVariants(module) &&
|
|
module.CcLibraryInterface() && module.Shared()
|
|
}
|
|
|
|
func moduleLibraryInterface(module blueprint.Module) libraryInterface {
|
|
if m, ok := module.(*Module); ok {
|
|
return m.library
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// setStubsVersions normalizes the versions in the Stubs.Versions property into MutatedProperties.AllStubsVersions.
|
|
func setStubsVersions(mctx android.BottomUpMutatorContext, library libraryInterface, module *Module) {
|
|
if !library.buildShared() || !canBeVersionVariant(module) {
|
|
return
|
|
}
|
|
versions := library.stubsVersions(mctx)
|
|
if mctx.Failed() {
|
|
return
|
|
}
|
|
// Set the versions on the pre-mutated module so they can be read by any llndk modules that
|
|
// depend on the implementation library and haven't been mutated yet.
|
|
library.setAllStubsVersions(versions)
|
|
}
|
|
|
|
// versionMutator splits a module into the mandatory non-stubs variant
|
|
// (which is unnamed) and zero or more stubs variants.
|
|
func versionMutator(mctx android.BottomUpMutatorContext) {
|
|
if mctx.Os() != android.Android {
|
|
return
|
|
}
|
|
|
|
m, ok := mctx.Module().(*Module)
|
|
if library := moduleLibraryInterface(mctx.Module()); library != nil && canBeVersionVariant(m) {
|
|
setStubsVersions(mctx, library, m)
|
|
|
|
createVersionVariations(mctx, library.allStubsVersions())
|
|
return
|
|
}
|
|
|
|
if ok {
|
|
if m.SplitPerApiLevel() && m.IsSdkVariant() {
|
|
createPerApiVersionVariations(mctx, m.MinSdkVersion())
|
|
}
|
|
}
|
|
}
|
|
|
|
// maybeInjectBoringSSLHash adds a rule to run bssl_inject_hash on the output file if the module has the
|
|
// inject_bssl_hash or if any static library dependencies have inject_bssl_hash set. It returns the output path
|
|
// that the linked output file should be written to.
|
|
// TODO(b/137267623): Remove this in favor of a cc_genrule when they support operating on shared libraries.
|
|
func maybeInjectBoringSSLHash(ctx android.ModuleContext, outputFile android.ModuleOutPath,
|
|
inject *bool, fileName string) android.ModuleOutPath {
|
|
// TODO(b/137267623): Remove this in favor of a cc_genrule when they support operating on shared libraries.
|
|
injectBoringSSLHash := Bool(inject)
|
|
ctx.VisitDirectDeps(func(dep android.Module) {
|
|
if tag, ok := ctx.OtherModuleDependencyTag(dep).(libraryDependencyTag); ok && tag.static() {
|
|
if cc, ok := dep.(*Module); ok {
|
|
if library, ok := cc.linker.(*libraryDecorator); ok {
|
|
if Bool(library.Properties.Inject_bssl_hash) {
|
|
injectBoringSSLHash = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|
|
if injectBoringSSLHash {
|
|
hashedOutputfile := outputFile
|
|
outputFile = android.PathForModuleOut(ctx, "unhashed", fileName)
|
|
|
|
rule := android.NewRuleBuilder(pctx, ctx)
|
|
rule.Command().
|
|
BuiltTool("bssl_inject_hash").
|
|
FlagWithInput("-in-object ", outputFile).
|
|
FlagWithOutput("-o ", hashedOutputfile)
|
|
rule.Build("injectCryptoHash", "inject crypto hash")
|
|
}
|
|
|
|
return outputFile
|
|
}
|