90f7a4dcab
Test: Added a test module that uses this find $SOONG_OUT -name libfoo.so # only shows up in current Bug: None Change-Id: Ic88d7c598cd5b6ded8fb851a7eaf859806336af7
299 lines
8.7 KiB
Go
299 lines
8.7 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"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/google/blueprint"
|
|
|
|
"android/soong/android"
|
|
)
|
|
|
|
var (
|
|
toolPath = pctx.SourcePathVariable("toolPath", "build/soong/cc/gen_stub_libs.py")
|
|
|
|
genStubSrc = pctx.AndroidStaticRule("genStubSrc",
|
|
blueprint.RuleParams{
|
|
Command: "$toolPath --arch $arch --api $apiLevel $in $out",
|
|
Description: "genStubSrc $out",
|
|
CommandDeps: []string{"$toolPath"},
|
|
}, "arch", "apiLevel")
|
|
|
|
ndkLibrarySuffix = ".ndk"
|
|
|
|
ndkPrebuiltSharedLibs = []string{
|
|
"android",
|
|
"c",
|
|
"dl",
|
|
"EGL",
|
|
"GLESv1_CM",
|
|
"GLESv2",
|
|
"GLESv3",
|
|
"jnigraphics",
|
|
"log",
|
|
"mediandk",
|
|
"m",
|
|
"OpenMAXAL",
|
|
"OpenSLES",
|
|
"stdc++",
|
|
"vulkan",
|
|
"z",
|
|
}
|
|
ndkPrebuiltSharedLibraries = addPrefix(append([]string(nil), ndkPrebuiltSharedLibs...), "lib")
|
|
|
|
// These libraries have migrated over to the new ndk_library, which is added
|
|
// as a variation dependency via depsMutator.
|
|
ndkMigratedLibs = []string{}
|
|
ndkMigratedLibsLock sync.Mutex // protects ndkMigratedLibs writes during parallel beginMutator
|
|
)
|
|
|
|
// Creates a stub shared library based on the provided version file.
|
|
//
|
|
// The name of the generated file will be based on the module name by stripping
|
|
// the ".ndk" suffix from the module name. Module names must end with ".ndk"
|
|
// (as a convention to allow soong to guess the NDK name of a dependency when
|
|
// needed). "libfoo.ndk" will generate "libfoo.so.
|
|
//
|
|
// Example:
|
|
//
|
|
// ndk_library {
|
|
// name: "libfoo.ndk",
|
|
// symbol_file: "libfoo.map.txt",
|
|
// first_version: "9",
|
|
// }
|
|
//
|
|
type libraryProperties struct {
|
|
// Relative path to the symbol map.
|
|
// An example file can be seen here: TODO(danalbert): Make an example.
|
|
Symbol_file string
|
|
|
|
// The first API level a library was available. A library will be generated
|
|
// for every API level beginning with this one.
|
|
First_version string
|
|
|
|
// Private property for use by the mutator that splits per-API level.
|
|
ApiLevel string `blueprint:"mutated"`
|
|
}
|
|
|
|
type stubDecorator struct {
|
|
*libraryDecorator
|
|
|
|
properties libraryProperties
|
|
|
|
versionScriptPath android.ModuleGenPath
|
|
installPath string
|
|
}
|
|
|
|
// OMG GO
|
|
func intMax(a int, b int) int {
|
|
if a > b {
|
|
return a
|
|
} else {
|
|
return b
|
|
}
|
|
}
|
|
|
|
func normalizeNdkApiLevel(apiLevel string, arch android.Arch) (string, error) {
|
|
if apiLevel == "current" {
|
|
return apiLevel, nil
|
|
}
|
|
|
|
minVersion := 9 // Minimum version supported by the NDK.
|
|
firstArchVersions := map[string]int{
|
|
"arm": 9,
|
|
"arm64": 21,
|
|
"mips": 9,
|
|
"mips64": 21,
|
|
"x86": 9,
|
|
"x86_64": 21,
|
|
}
|
|
|
|
// If the NDK drops support for a platform version, we don't want to have to
|
|
// fix up every module that was using it as its SDK version. Clip to the
|
|
// supported version here instead.
|
|
version, err := strconv.Atoi(apiLevel)
|
|
if err != nil {
|
|
return "", fmt.Errorf("API level must be an integer (is %q)", apiLevel)
|
|
}
|
|
version = intMax(version, minVersion)
|
|
|
|
archStr := arch.ArchType.String()
|
|
firstArchVersion, ok := firstArchVersions[archStr]
|
|
if !ok {
|
|
panic(fmt.Errorf("Arch %q not found in firstArchVersions", archStr))
|
|
}
|
|
|
|
return strconv.Itoa(intMax(version, firstArchVersion)), nil
|
|
}
|
|
|
|
func getFirstGeneratedVersion(firstSupportedVersion string, platformVersion int) (int, error) {
|
|
if firstSupportedVersion == "current" {
|
|
return platformVersion + 1, nil
|
|
}
|
|
|
|
return strconv.Atoi(firstSupportedVersion)
|
|
}
|
|
|
|
func generateStubApiVariants(mctx android.BottomUpMutatorContext, c *stubDecorator) {
|
|
platformVersion := mctx.AConfig().PlatformSdkVersionInt()
|
|
|
|
firstSupportedVersion, err := normalizeNdkApiLevel(c.properties.First_version,
|
|
mctx.Arch())
|
|
if err != nil {
|
|
mctx.PropertyErrorf("first_version", err.Error())
|
|
}
|
|
|
|
firstGenVersion, err := getFirstGeneratedVersion(firstSupportedVersion, platformVersion)
|
|
if err != nil {
|
|
// In theory this is impossible because we've already run this through
|
|
// normalizeNdkApiLevel above.
|
|
mctx.PropertyErrorf("first_version", err.Error())
|
|
}
|
|
|
|
var versionStrs []string
|
|
for version := firstGenVersion; version <= platformVersion; version++ {
|
|
versionStrs = append(versionStrs, strconv.Itoa(version))
|
|
}
|
|
versionStrs = append(versionStrs, "current")
|
|
|
|
modules := mctx.CreateVariations(versionStrs...)
|
|
for i, module := range modules {
|
|
module.(*Module).compiler.(*stubDecorator).properties.ApiLevel = versionStrs[i]
|
|
}
|
|
}
|
|
|
|
func ndkApiMutator(mctx android.BottomUpMutatorContext) {
|
|
if m, ok := mctx.Module().(*Module); ok {
|
|
if compiler, ok := m.compiler.(*stubDecorator); ok {
|
|
generateStubApiVariants(mctx, compiler)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *stubDecorator) compilerInit(ctx BaseModuleContext) {
|
|
c.baseCompiler.compilerInit(ctx)
|
|
|
|
name := strings.TrimSuffix(ctx.ModuleName(), ".ndk")
|
|
ndkMigratedLibsLock.Lock()
|
|
defer ndkMigratedLibsLock.Unlock()
|
|
for _, lib := range ndkMigratedLibs {
|
|
if lib == name {
|
|
return
|
|
}
|
|
}
|
|
ndkMigratedLibs = append(ndkMigratedLibs, name)
|
|
}
|
|
|
|
func (c *stubDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
|
|
arch := ctx.Arch().ArchType.String()
|
|
|
|
if !strings.HasSuffix(ctx.ModuleName(), ndkLibrarySuffix) {
|
|
ctx.ModuleErrorf("ndk_library modules names must be suffixed with %q\n",
|
|
ndkLibrarySuffix)
|
|
}
|
|
libName := strings.TrimSuffix(ctx.ModuleName(), ndkLibrarySuffix)
|
|
fileBase := fmt.Sprintf("%s.%s.%s", libName, arch, c.properties.ApiLevel)
|
|
stubSrcName := fileBase + ".c"
|
|
stubSrcPath := android.PathForModuleGen(ctx, stubSrcName)
|
|
versionScriptName := fileBase + ".map"
|
|
versionScriptPath := android.PathForModuleGen(ctx, versionScriptName)
|
|
c.versionScriptPath = versionScriptPath
|
|
symbolFilePath := android.PathForModuleSrc(ctx, c.properties.Symbol_file)
|
|
ctx.ModuleBuild(pctx, android.ModuleBuildParams{
|
|
Rule: genStubSrc,
|
|
Outputs: []android.WritablePath{stubSrcPath, versionScriptPath},
|
|
Input: symbolFilePath,
|
|
Args: map[string]string{
|
|
"arch": arch,
|
|
"apiLevel": c.properties.ApiLevel,
|
|
},
|
|
})
|
|
|
|
flags.CFlags = append(flags.CFlags,
|
|
// We're knowingly doing some otherwise unsightly things with builtin
|
|
// functions here. We're just generating stub libraries, so ignore it.
|
|
"-Wno-incompatible-library-redeclaration",
|
|
"-Wno-builtin-requires-header",
|
|
"-Wno-invalid-noreturn",
|
|
|
|
// These libraries aren't actually used. Don't worry about unwinding
|
|
// (avoids the need to link an unwinder into a fake library).
|
|
"-fno-unwind-tables",
|
|
)
|
|
|
|
subdir := ""
|
|
srcs := []android.Path{stubSrcPath}
|
|
return compileObjs(ctx, flagsToBuilderFlags(flags), subdir, srcs, nil)
|
|
}
|
|
|
|
func (linker *stubDecorator) linkerDeps(ctx BaseModuleContext, deps Deps) Deps {
|
|
return Deps{}
|
|
}
|
|
|
|
func (stub *stubDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
|
|
stub.libraryDecorator.libName = strings.TrimSuffix(ctx.ModuleName(),
|
|
ndkLibrarySuffix)
|
|
return stub.libraryDecorator.linkerFlags(ctx, flags)
|
|
}
|
|
|
|
func (stub *stubDecorator) link(ctx ModuleContext, flags Flags, deps PathDeps,
|
|
objs Objects) android.Path {
|
|
|
|
linkerScriptFlag := "-Wl,--version-script," + stub.versionScriptPath.String()
|
|
flags.LdFlags = append(flags.LdFlags, linkerScriptFlag)
|
|
return stub.libraryDecorator.link(ctx, flags, deps, objs)
|
|
}
|
|
|
|
func (stub *stubDecorator) install(ctx ModuleContext, path android.Path) {
|
|
arch := ctx.Target().Arch.ArchType.Name
|
|
apiLevel := stub.properties.ApiLevel
|
|
|
|
// arm64 isn't actually a multilib toolchain, so unlike the other LP64
|
|
// architectures it's just installed to lib.
|
|
libDir := "lib"
|
|
if ctx.toolchain().Is64Bit() && arch != "arm64" {
|
|
libDir = "lib64"
|
|
}
|
|
|
|
installDir := getNdkInstallBase(ctx).Join(ctx, fmt.Sprintf(
|
|
"platforms/android-%s/arch-%s/usr/%s", apiLevel, arch, libDir))
|
|
stub.installPath = ctx.InstallFile(installDir, path).String()
|
|
}
|
|
|
|
func newStubLibrary() (*Module, []interface{}) {
|
|
module, library := NewLibrary(android.DeviceSupported, true, false)
|
|
module.stl = nil
|
|
module.sanitize = nil
|
|
library.StripProperties.Strip.None = true
|
|
|
|
stub := &stubDecorator{
|
|
libraryDecorator: library,
|
|
}
|
|
module.compiler = stub
|
|
module.linker = stub
|
|
module.installer = stub
|
|
|
|
return module, []interface{}{&stub.properties}
|
|
}
|
|
|
|
func ndkLibraryFactory() (blueprint.Module, []interface{}) {
|
|
module, properties := newStubLibrary()
|
|
return android.InitAndroidArchModule(module, android.DeviceSupported,
|
|
android.MultilibBoth, properties...)
|
|
}
|