2016-06-18 01:45:24 +02:00
|
|
|
// 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"
|
|
|
|
|
|
|
|
"github.com/google/blueprint"
|
|
|
|
|
|
|
|
"android/soong/android"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
toolPath = pctx.SourcePathVariable("toolPath", "build/soong/cc/gen_stub_libs.py")
|
|
|
|
|
|
|
|
genStubSrc = pctx.StaticRule("genStubSrc",
|
|
|
|
blueprint.RuleParams{
|
|
|
|
Command: "$toolPath --arch $arch --api $apiLevel $in $out",
|
|
|
|
Description: "genStubSrc $out",
|
|
|
|
CommandDeps: []string{"$toolPath"},
|
|
|
|
}, "arch", "apiLevel")
|
|
|
|
|
|
|
|
ndkLibrarySuffix = ".ndk"
|
2016-07-29 21:48:20 +02:00
|
|
|
|
|
|
|
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{}
|
2016-06-18 01:45:24 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// 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 int `blueprint:"mutated"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type stubCompiler struct {
|
|
|
|
baseCompiler
|
|
|
|
|
|
|
|
properties libraryProperties
|
2016-07-29 02:40:28 +02:00
|
|
|
|
|
|
|
linker *stubLinker
|
2016-06-18 01:45:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// OMG GO
|
|
|
|
func intMin(a int, b int) int {
|
|
|
|
if a < b {
|
|
|
|
return a
|
|
|
|
} else {
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func generateStubApiVariants(mctx android.BottomUpMutatorContext, c *stubCompiler) {
|
|
|
|
minVersion := 9 // Minimum version supported by the NDK.
|
|
|
|
// TODO(danalbert): Use PlatformSdkVersion when possible.
|
|
|
|
// This is an interesting case because for the moment we actually need 24
|
|
|
|
// even though the latest released version in aosp is 23. prebuilts/ndk/r11
|
|
|
|
// has android-24 versions of libraries, and as platform libraries get
|
|
|
|
// migrated the libraries in prebuilts will need to depend on them.
|
|
|
|
//
|
|
|
|
// Once everything is all moved over to the new stuff (when there isn't a
|
|
|
|
// prebuilts/ndk any more) then this should be fixable, but for now I think
|
|
|
|
// it needs to remain as-is.
|
|
|
|
maxVersion := 24
|
|
|
|
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 minimum version. Clip to the
|
|
|
|
// supported version here instead.
|
|
|
|
firstVersion, err := strconv.Atoi(c.properties.First_version)
|
|
|
|
if err != nil {
|
|
|
|
mctx.ModuleErrorf("Invalid first_version value (must be int): %q",
|
|
|
|
c.properties.First_version)
|
|
|
|
}
|
|
|
|
if firstVersion < minVersion {
|
|
|
|
firstVersion = minVersion
|
|
|
|
}
|
|
|
|
|
|
|
|
arch := mctx.Arch().ArchType.String()
|
|
|
|
firstArchVersion, ok := firstArchVersions[arch]
|
|
|
|
if !ok {
|
|
|
|
panic(fmt.Errorf("Arch %q not found in firstArchVersions", arch))
|
|
|
|
}
|
|
|
|
firstGenVersion := intMin(firstVersion, firstArchVersion)
|
|
|
|
versionStrs := make([]string, maxVersion-firstGenVersion+1)
|
|
|
|
for version := firstGenVersion; version <= maxVersion; version++ {
|
|
|
|
versionStrs[version-firstGenVersion] = strconv.Itoa(version)
|
|
|
|
}
|
|
|
|
|
|
|
|
modules := mctx.CreateVariations(versionStrs...)
|
|
|
|
for i, module := range modules {
|
|
|
|
module.(*Module).compiler.(*stubCompiler).properties.ApiLevel = firstGenVersion + i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func ndkApiMutator(mctx android.BottomUpMutatorContext) {
|
|
|
|
if m, ok := mctx.Module().(*Module); ok {
|
|
|
|
if compiler, ok := m.compiler.(*stubCompiler); ok {
|
|
|
|
generateStubApiVariants(mctx, compiler)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-04 22:02:36 +02:00
|
|
|
func (c *stubCompiler) compilerInit(ctx BaseModuleContext) {
|
|
|
|
c.baseCompiler.compilerInit(ctx)
|
|
|
|
|
|
|
|
name := strings.TrimSuffix(ctx.ModuleName(), ".ndk")
|
|
|
|
for _, lib := range ndkMigratedLibs {
|
|
|
|
if lib == name {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ndkMigratedLibs = append(ndkMigratedLibs, name)
|
|
|
|
}
|
|
|
|
|
2016-06-18 01:45:24 +02:00
|
|
|
func (c *stubCompiler) compile(ctx ModuleContext, flags Flags, deps PathDeps) android.Paths {
|
|
|
|
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.%d", libName, arch, c.properties.ApiLevel)
|
|
|
|
stubSrcName := fileBase + ".c"
|
|
|
|
stubSrcPath := android.PathForModuleGen(ctx, stubSrcName)
|
|
|
|
versionScriptName := fileBase + ".map"
|
|
|
|
versionScriptPath := android.PathForModuleGen(ctx, versionScriptName)
|
2016-07-29 02:40:28 +02:00
|
|
|
c.linker.versionScriptPath = versionScriptPath
|
2016-06-18 01:45:24 +02:00
|
|
|
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": strconv.Itoa(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 := []string{}
|
|
|
|
excludeSrcs := []string{}
|
|
|
|
extraSrcs := []android.Path{stubSrcPath}
|
|
|
|
extraDeps := []android.Path{}
|
|
|
|
return c.baseCompiler.compileObjs(ctx, flags, subdir, srcs, excludeSrcs,
|
|
|
|
extraSrcs, extraDeps)
|
|
|
|
}
|
|
|
|
|
|
|
|
type stubLinker struct {
|
|
|
|
libraryLinker
|
2016-07-29 02:40:28 +02:00
|
|
|
|
|
|
|
versionScriptPath android.ModuleGenPath
|
2016-06-18 01:45:24 +02:00
|
|
|
}
|
|
|
|
|
2016-08-01 22:20:05 +02:00
|
|
|
func (linker *stubLinker) linkerDeps(ctx BaseModuleContext, deps Deps) Deps {
|
2016-06-18 01:45:24 +02:00
|
|
|
return Deps{}
|
|
|
|
}
|
|
|
|
|
2016-08-01 22:20:05 +02:00
|
|
|
func (linker *stubLinker) linkerFlags(ctx ModuleContext, flags Flags) Flags {
|
2016-06-18 01:45:24 +02:00
|
|
|
linker.libraryLinker.libName = strings.TrimSuffix(ctx.ModuleName(),
|
|
|
|
ndkLibrarySuffix)
|
2016-08-01 22:20:05 +02:00
|
|
|
return linker.libraryLinker.linkerFlags(ctx, flags)
|
2016-06-18 01:45:24 +02:00
|
|
|
}
|
|
|
|
|
2016-07-29 02:40:28 +02:00
|
|
|
func (linker *stubLinker) link(ctx ModuleContext, flags Flags, deps PathDeps,
|
|
|
|
objFiles android.Paths) android.Path {
|
|
|
|
|
|
|
|
linkerScriptFlag := "-Wl,--version-script," + linker.versionScriptPath.String()
|
|
|
|
flags.LdFlags = append(flags.LdFlags, linkerScriptFlag)
|
|
|
|
return linker.libraryLinker.link(ctx, flags, deps, objFiles)
|
|
|
|
}
|
|
|
|
|
2016-06-18 01:45:24 +02:00
|
|
|
type stubInstaller struct {
|
|
|
|
baseInstaller
|
|
|
|
|
|
|
|
compiler *stubCompiler
|
|
|
|
|
|
|
|
installPath string
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ installer = (*stubInstaller)(nil)
|
|
|
|
|
|
|
|
func (installer *stubInstaller) install(ctx ModuleContext, path android.Path) {
|
|
|
|
arch := ctx.Target().Arch.ArchType.Name
|
|
|
|
apiLevel := installer.compiler.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-%d/arch-%s/usr/%s", apiLevel, arch, libDir))
|
|
|
|
installer.installPath = ctx.InstallFile(installDir, path).String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func newStubLibrary() *Module {
|
|
|
|
module := newModule(android.DeviceSupported, android.MultilibBoth)
|
|
|
|
module.stl = nil
|
|
|
|
|
|
|
|
linker := &stubLinker{}
|
|
|
|
linker.dynamicProperties.BuildShared = true
|
|
|
|
linker.dynamicProperties.BuildStatic = false
|
|
|
|
linker.stripper.StripProperties.Strip.None = true
|
|
|
|
module.linker = linker
|
|
|
|
|
|
|
|
compiler := &stubCompiler{}
|
2016-07-29 02:40:28 +02:00
|
|
|
compiler.linker = linker
|
2016-06-18 01:45:24 +02:00
|
|
|
module.compiler = compiler
|
|
|
|
module.installer = &stubInstaller{baseInstaller{
|
|
|
|
dir: "lib",
|
|
|
|
dir64: "lib64",
|
|
|
|
}, compiler, ""}
|
|
|
|
|
|
|
|
return module
|
|
|
|
}
|
|
|
|
|
|
|
|
func ndkLibraryFactory() (blueprint.Module, []interface{}) {
|
|
|
|
module := newStubLibrary()
|
|
|
|
return android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibBoth,
|
|
|
|
&module.compiler.(*stubCompiler).properties)
|
|
|
|
}
|