blob: f733515d5e4752fd3ca574a8c30612b0c63d00ff [file] [log] [blame]
Dan Albert914449f2016-06-17 16:45:24 -07001// Copyright 2016 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package cc
16
17import (
18 "fmt"
19 "strconv"
20 "strings"
Colin Crosse8a67a72016-08-07 21:17:54 -070021 "sync"
Dan Albert914449f2016-06-17 16:45:24 -070022
23 "github.com/google/blueprint"
24
25 "android/soong/android"
26)
27
28var (
29 toolPath = pctx.SourcePathVariable("toolPath", "build/soong/cc/gen_stub_libs.py")
30
Colin Cross9d45bb72016-08-29 16:14:13 -070031 genStubSrc = pctx.AndroidStaticRule("genStubSrc",
Dan Albert914449f2016-06-17 16:45:24 -070032 blueprint.RuleParams{
33 Command: "$toolPath --arch $arch --api $apiLevel $in $out",
34 Description: "genStubSrc $out",
35 CommandDeps: []string{"$toolPath"},
36 }, "arch", "apiLevel")
37
38 ndkLibrarySuffix = ".ndk"
Colin Cross4d9c2d12016-07-29 12:48:20 -070039
40 ndkPrebuiltSharedLibs = []string{
41 "android",
42 "c",
43 "dl",
44 "EGL",
45 "GLESv1_CM",
46 "GLESv2",
47 "GLESv3",
48 "jnigraphics",
49 "log",
50 "mediandk",
51 "m",
52 "OpenMAXAL",
53 "OpenSLES",
54 "stdc++",
55 "vulkan",
56 "z",
57 }
58 ndkPrebuiltSharedLibraries = addPrefix(append([]string(nil), ndkPrebuiltSharedLibs...), "lib")
59
60 // These libraries have migrated over to the new ndk_library, which is added
61 // as a variation dependency via depsMutator.
Colin Crosse8a67a72016-08-07 21:17:54 -070062 ndkMigratedLibs = []string{}
63 ndkMigratedLibsLock sync.Mutex // protects ndkMigratedLibs writes during parallel beginMutator
Dan Albert914449f2016-06-17 16:45:24 -070064)
65
66// Creates a stub shared library based on the provided version file.
67//
68// The name of the generated file will be based on the module name by stripping
69// the ".ndk" suffix from the module name. Module names must end with ".ndk"
70// (as a convention to allow soong to guess the NDK name of a dependency when
71// needed). "libfoo.ndk" will generate "libfoo.so.
72//
73// Example:
74//
75// ndk_library {
76// name: "libfoo.ndk",
77// symbol_file: "libfoo.map.txt",
78// first_version: "9",
79// }
80//
81type libraryProperties struct {
82 // Relative path to the symbol map.
83 // An example file can be seen here: TODO(danalbert): Make an example.
84 Symbol_file string
85
86 // The first API level a library was available. A library will be generated
87 // for every API level beginning with this one.
88 First_version string
89
90 // Private property for use by the mutator that splits per-API level.
Dan Albertfd86e9e2016-11-08 13:35:12 -080091 ApiLevel string `blueprint:"mutated"`
Dan Albert914449f2016-06-17 16:45:24 -070092}
93
Colin Crossb916a382016-07-29 17:28:03 -070094type stubDecorator struct {
95 *libraryDecorator
Dan Albert914449f2016-06-17 16:45:24 -070096
97 properties libraryProperties
Dan Albert2bc91ba2016-07-28 17:40:28 -070098
Colin Crossb916a382016-07-29 17:28:03 -070099 versionScriptPath android.ModuleGenPath
100 installPath string
Dan Albert914449f2016-06-17 16:45:24 -0700101}
102
103// OMG GO
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700104func intMax(a int, b int) int {
105 if a > b {
Dan Albert914449f2016-06-17 16:45:24 -0700106 return a
107 } else {
108 return b
109 }
110}
111
Dan Albert90f7a4d2016-11-08 14:34:24 -0800112func normalizeNdkApiLevel(apiLevel string, arch android.Arch) (string, error) {
113 if apiLevel == "current" {
114 return apiLevel, nil
115 }
116
Dan Albert914449f2016-06-17 16:45:24 -0700117 minVersion := 9 // Minimum version supported by the NDK.
Dan Albert914449f2016-06-17 16:45:24 -0700118 firstArchVersions := map[string]int{
119 "arm": 9,
120 "arm64": 21,
121 "mips": 9,
122 "mips64": 21,
123 "x86": 9,
124 "x86_64": 21,
125 }
126
127 // If the NDK drops support for a platform version, we don't want to have to
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700128 // fix up every module that was using it as its SDK version. Clip to the
Dan Albert914449f2016-06-17 16:45:24 -0700129 // supported version here instead.
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700130 version, err := strconv.Atoi(apiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700131 if err != nil {
Dan Albert90f7a4d2016-11-08 14:34:24 -0800132 return "", fmt.Errorf("API level must be an integer (is %q)", apiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700133 }
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700134 version = intMax(version, minVersion)
135
136 archStr := arch.ArchType.String()
137 firstArchVersion, ok := firstArchVersions[archStr]
138 if !ok {
139 panic(fmt.Errorf("Arch %q not found in firstArchVersions", archStr))
Dan Albert914449f2016-06-17 16:45:24 -0700140 }
141
Dan Albert90f7a4d2016-11-08 14:34:24 -0800142 return strconv.Itoa(intMax(version, firstArchVersion)), nil
143}
144
145func getFirstGeneratedVersion(firstSupportedVersion string, platformVersion int) (int, error) {
146 if firstSupportedVersion == "current" {
147 return platformVersion + 1, nil
148 }
149
150 return strconv.Atoi(firstSupportedVersion)
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700151}
152
153func generateStubApiVariants(mctx android.BottomUpMutatorContext, c *stubDecorator) {
Dan Albert90f7a4d2016-11-08 14:34:24 -0800154 platformVersion := mctx.AConfig().PlatformSdkVersionInt()
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700155
Dan Albert90f7a4d2016-11-08 14:34:24 -0800156 firstSupportedVersion, err := normalizeNdkApiLevel(c.properties.First_version,
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700157 mctx.Arch())
158 if err != nil {
159 mctx.PropertyErrorf("first_version", err.Error())
Dan Albert914449f2016-06-17 16:45:24 -0700160 }
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700161
Dan Albert90f7a4d2016-11-08 14:34:24 -0800162 firstGenVersion, err := getFirstGeneratedVersion(firstSupportedVersion, platformVersion)
163 if err != nil {
164 // In theory this is impossible because we've already run this through
165 // normalizeNdkApiLevel above.
166 mctx.PropertyErrorf("first_version", err.Error())
167 }
168
Dan Albertfd86e9e2016-11-08 13:35:12 -0800169 var versionStrs []string
Dan Albert90f7a4d2016-11-08 14:34:24 -0800170 for version := firstGenVersion; version <= platformVersion; version++ {
Dan Albertfd86e9e2016-11-08 13:35:12 -0800171 versionStrs = append(versionStrs, strconv.Itoa(version))
Dan Albert914449f2016-06-17 16:45:24 -0700172 }
Dan Albertfd86e9e2016-11-08 13:35:12 -0800173 versionStrs = append(versionStrs, "current")
Dan Albert914449f2016-06-17 16:45:24 -0700174
175 modules := mctx.CreateVariations(versionStrs...)
176 for i, module := range modules {
Dan Albertfd86e9e2016-11-08 13:35:12 -0800177 module.(*Module).compiler.(*stubDecorator).properties.ApiLevel = versionStrs[i]
Dan Albert914449f2016-06-17 16:45:24 -0700178 }
179}
180
181func ndkApiMutator(mctx android.BottomUpMutatorContext) {
182 if m, ok := mctx.Module().(*Module); ok {
Colin Crossb916a382016-07-29 17:28:03 -0700183 if compiler, ok := m.compiler.(*stubDecorator); ok {
Dan Albert914449f2016-06-17 16:45:24 -0700184 generateStubApiVariants(mctx, compiler)
185 }
186 }
187}
188
Colin Crossb916a382016-07-29 17:28:03 -0700189func (c *stubDecorator) compilerInit(ctx BaseModuleContext) {
Dan Albert7e9d2952016-08-04 13:02:36 -0700190 c.baseCompiler.compilerInit(ctx)
191
192 name := strings.TrimSuffix(ctx.ModuleName(), ".ndk")
Colin Crosse8a67a72016-08-07 21:17:54 -0700193 ndkMigratedLibsLock.Lock()
194 defer ndkMigratedLibsLock.Unlock()
Dan Albert7e9d2952016-08-04 13:02:36 -0700195 for _, lib := range ndkMigratedLibs {
196 if lib == name {
197 return
198 }
199 }
200 ndkMigratedLibs = append(ndkMigratedLibs, name)
201}
202
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700203func (c *stubDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Dan Albert914449f2016-06-17 16:45:24 -0700204 arch := ctx.Arch().ArchType.String()
205
206 if !strings.HasSuffix(ctx.ModuleName(), ndkLibrarySuffix) {
207 ctx.ModuleErrorf("ndk_library modules names must be suffixed with %q\n",
208 ndkLibrarySuffix)
209 }
210 libName := strings.TrimSuffix(ctx.ModuleName(), ndkLibrarySuffix)
Dan Albertfd86e9e2016-11-08 13:35:12 -0800211 fileBase := fmt.Sprintf("%s.%s.%s", libName, arch, c.properties.ApiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700212 stubSrcName := fileBase + ".c"
213 stubSrcPath := android.PathForModuleGen(ctx, stubSrcName)
214 versionScriptName := fileBase + ".map"
215 versionScriptPath := android.PathForModuleGen(ctx, versionScriptName)
Colin Crossb916a382016-07-29 17:28:03 -0700216 c.versionScriptPath = versionScriptPath
Dan Albert914449f2016-06-17 16:45:24 -0700217 symbolFilePath := android.PathForModuleSrc(ctx, c.properties.Symbol_file)
218 ctx.ModuleBuild(pctx, android.ModuleBuildParams{
219 Rule: genStubSrc,
220 Outputs: []android.WritablePath{stubSrcPath, versionScriptPath},
221 Input: symbolFilePath,
222 Args: map[string]string{
223 "arch": arch,
Dan Albertfd86e9e2016-11-08 13:35:12 -0800224 "apiLevel": c.properties.ApiLevel,
Dan Albert914449f2016-06-17 16:45:24 -0700225 },
226 })
227
228 flags.CFlags = append(flags.CFlags,
229 // We're knowingly doing some otherwise unsightly things with builtin
230 // functions here. We're just generating stub libraries, so ignore it.
231 "-Wno-incompatible-library-redeclaration",
232 "-Wno-builtin-requires-header",
233 "-Wno-invalid-noreturn",
234
235 // These libraries aren't actually used. Don't worry about unwinding
236 // (avoids the need to link an unwinder into a fake library).
237 "-fno-unwind-tables",
238 )
239
240 subdir := ""
Colin Cross2f336352016-10-26 10:03:47 -0700241 srcs := []android.Path{stubSrcPath}
242 return compileObjs(ctx, flagsToBuilderFlags(flags), subdir, srcs, nil)
Dan Albert914449f2016-06-17 16:45:24 -0700243}
244
Colin Crossb916a382016-07-29 17:28:03 -0700245func (linker *stubDecorator) linkerDeps(ctx BaseModuleContext, deps Deps) Deps {
Dan Albert914449f2016-06-17 16:45:24 -0700246 return Deps{}
247}
248
Colin Crossb916a382016-07-29 17:28:03 -0700249func (stub *stubDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
250 stub.libraryDecorator.libName = strings.TrimSuffix(ctx.ModuleName(),
Dan Albert914449f2016-06-17 16:45:24 -0700251 ndkLibrarySuffix)
Colin Crossb916a382016-07-29 17:28:03 -0700252 return stub.libraryDecorator.linkerFlags(ctx, flags)
Dan Albert914449f2016-06-17 16:45:24 -0700253}
254
Colin Crossb916a382016-07-29 17:28:03 -0700255func (stub *stubDecorator) link(ctx ModuleContext, flags Flags, deps PathDeps,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700256 objs Objects) android.Path {
Dan Albert2bc91ba2016-07-28 17:40:28 -0700257
Colin Crossb916a382016-07-29 17:28:03 -0700258 linkerScriptFlag := "-Wl,--version-script," + stub.versionScriptPath.String()
Dan Albert2bc91ba2016-07-28 17:40:28 -0700259 flags.LdFlags = append(flags.LdFlags, linkerScriptFlag)
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700260 return stub.libraryDecorator.link(ctx, flags, deps, objs)
Dan Albert2bc91ba2016-07-28 17:40:28 -0700261}
262
Colin Crossb916a382016-07-29 17:28:03 -0700263func (stub *stubDecorator) install(ctx ModuleContext, path android.Path) {
Dan Albert914449f2016-06-17 16:45:24 -0700264 arch := ctx.Target().Arch.ArchType.Name
Colin Crossb916a382016-07-29 17:28:03 -0700265 apiLevel := stub.properties.ApiLevel
Dan Albert914449f2016-06-17 16:45:24 -0700266
267 // arm64 isn't actually a multilib toolchain, so unlike the other LP64
268 // architectures it's just installed to lib.
269 libDir := "lib"
270 if ctx.toolchain().Is64Bit() && arch != "arm64" {
271 libDir = "lib64"
272 }
273
274 installDir := getNdkInstallBase(ctx).Join(ctx, fmt.Sprintf(
Dan Albertfd86e9e2016-11-08 13:35:12 -0800275 "platforms/android-%s/arch-%s/usr/%s", apiLevel, arch, libDir))
Colin Crossb916a382016-07-29 17:28:03 -0700276 stub.installPath = ctx.InstallFile(installDir, path).String()
Dan Albert914449f2016-06-17 16:45:24 -0700277}
278
Dan Albert705c84b2016-08-08 10:45:03 -0700279func newStubLibrary() (*Module, []interface{}) {
Colin Crossb916a382016-07-29 17:28:03 -0700280 module, library := NewLibrary(android.DeviceSupported, true, false)
Dan Albert914449f2016-06-17 16:45:24 -0700281 module.stl = nil
Colin Crossb916a382016-07-29 17:28:03 -0700282 module.sanitize = nil
283 library.StripProperties.Strip.None = true
Dan Albert914449f2016-06-17 16:45:24 -0700284
Colin Crossb916a382016-07-29 17:28:03 -0700285 stub := &stubDecorator{
286 libraryDecorator: library,
287 }
288 module.compiler = stub
289 module.linker = stub
290 module.installer = stub
Dan Albert914449f2016-06-17 16:45:24 -0700291
Dan Albert705c84b2016-08-08 10:45:03 -0700292 return module, []interface{}{&stub.properties}
Dan Albert914449f2016-06-17 16:45:24 -0700293}
294
295func ndkLibraryFactory() (blueprint.Module, []interface{}) {
Dan Albert705c84b2016-08-08 10:45:03 -0700296 module, properties := newStubLibrary()
297 return android.InitAndroidArchModule(module, android.DeviceSupported,
298 android.MultilibBoth, properties...)
Dan Albert914449f2016-06-17 16:45:24 -0700299}