blob: 969cb3fa7e876c10faff6d6b6df50fe92abcf51a [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{
Dan Albert49927d22017-03-28 15:00:46 -070033 Command: "$toolPath --arch $arch --api $apiLevel --api-map " +
Jiyong Park3fd0baf2018-12-07 16:25:39 +090034 "$apiMap $flags $in $out",
Dan Albert914449f2016-06-17 16:45:24 -070035 CommandDeps: []string{"$toolPath"},
Jiyong Park3fd0baf2018-12-07 16:25:39 +090036 }, "arch", "apiLevel", "apiMap", "flags")
Dan Albert914449f2016-06-17 16:45:24 -070037
38 ndkLibrarySuffix = ".ndk"
Colin Cross4d9c2d12016-07-29 12:48:20 -070039
40 ndkPrebuiltSharedLibs = []string{
Sasha Smundakb500ce52019-03-19 17:13:17 -070041 "aaudio",
42 "amidi",
Colin Cross4d9c2d12016-07-29 12:48:20 -070043 "android",
Steven Morelandfa287842018-08-29 20:14:18 -070044 "binder_ndk",
Colin Cross4d9c2d12016-07-29 12:48:20 -070045 "c",
Sasha Smundakb500ce52019-03-19 17:13:17 -070046 "camera2ndk",
Colin Cross4d9c2d12016-07-29 12:48:20 -070047 "dl",
48 "EGL",
49 "GLESv1_CM",
50 "GLESv2",
51 "GLESv3",
52 "jnigraphics",
53 "log",
54 "mediandk",
Sasha Smundakb500ce52019-03-19 17:13:17 -070055 "nativewindow",
Colin Cross4d9c2d12016-07-29 12:48:20 -070056 "m",
57 "OpenMAXAL",
58 "OpenSLES",
59 "stdc++",
Dan Willemsen62b9cf92019-02-06 18:40:16 -080060 "sync",
Colin Cross4d9c2d12016-07-29 12:48:20 -070061 "vulkan",
62 "z",
63 }
64 ndkPrebuiltSharedLibraries = addPrefix(append([]string(nil), ndkPrebuiltSharedLibs...), "lib")
65
66 // These libraries have migrated over to the new ndk_library, which is added
67 // as a variation dependency via depsMutator.
Colin Crosse8a67a72016-08-07 21:17:54 -070068 ndkMigratedLibs = []string{}
Colin Crosse40b4ea2018-10-02 22:25:58 -070069 ndkMigratedLibsLock sync.Mutex // protects ndkMigratedLibs writes during parallel BeginMutator
Dan Albert914449f2016-06-17 16:45:24 -070070)
71
72// Creates a stub shared library based on the provided version file.
73//
Dan Albert914449f2016-06-17 16:45:24 -070074// Example:
75//
76// ndk_library {
Dan Willemsen01a90592017-04-07 15:21:13 -070077// name: "libfoo",
Dan Albert914449f2016-06-17 16:45:24 -070078// symbol_file: "libfoo.map.txt",
79// first_version: "9",
80// }
81//
82type libraryProperties struct {
83 // Relative path to the symbol map.
84 // An example file can be seen here: TODO(danalbert): Make an example.
Nan Zhang0007d812017-11-07 10:57:05 -080085 Symbol_file *string
Dan Albert914449f2016-06-17 16:45:24 -070086
87 // The first API level a library was available. A library will be generated
88 // for every API level beginning with this one.
Nan Zhang0007d812017-11-07 10:57:05 -080089 First_version *string
Dan Albert914449f2016-06-17 16:45:24 -070090
Dan Albert98dbb3b2017-01-03 15:16:29 -080091 // The first API level that library should have the version script applied.
92 // This defaults to the value of first_version, and should almost never be
93 // used. This is only needed to work around platform bugs like
94 // https://github.com/android-ndk/ndk/issues/265.
Nan Zhang0007d812017-11-07 10:57:05 -080095 Unversioned_until *string
Dan Albert98dbb3b2017-01-03 15:16:29 -080096
Dan Albert914449f2016-06-17 16:45:24 -070097 // Private property for use by the mutator that splits per-API level.
Dan Albertfd86e9e2016-11-08 13:35:12 -080098 ApiLevel string `blueprint:"mutated"`
Dan Albert23d37e02018-11-28 08:30:10 -080099
100 // True if this API is not yet ready to be shipped in the NDK. It will be
101 // available in the platform for testing, but will be excluded from the
102 // sysroot provided to the NDK proper.
103 Draft bool
Dan Albert914449f2016-06-17 16:45:24 -0700104}
105
Colin Crossb916a382016-07-29 17:28:03 -0700106type stubDecorator struct {
107 *libraryDecorator
Dan Albert914449f2016-06-17 16:45:24 -0700108
109 properties libraryProperties
Dan Albert2bc91ba2016-07-28 17:40:28 -0700110
Colin Crossb916a382016-07-29 17:28:03 -0700111 versionScriptPath android.ModuleGenPath
Colin Cross0875c522017-11-28 17:34:01 -0800112 installPath android.Path
Dan Albert914449f2016-06-17 16:45:24 -0700113}
114
115// OMG GO
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700116func intMax(a int, b int) int {
117 if a > b {
Dan Albert914449f2016-06-17 16:45:24 -0700118 return a
119 } else {
120 return b
121 }
122}
123
Colin Cross0ea8ba82019-06-06 14:33:29 -0700124func normalizeNdkApiLevel(ctx android.BaseModuleContext, apiLevel string,
Dan Albertf5415d72017-08-17 16:19:59 -0700125 arch android.Arch) (string, error) {
126
Dan Albert90f7a4d2016-11-08 14:34:24 -0800127 if apiLevel == "current" {
128 return apiLevel, nil
129 }
130
Colin Cross6510f912017-11-29 00:27:14 -0800131 minVersion := ctx.Config().MinSupportedSdkVersion()
Colin Crossce87b802017-04-13 13:00:26 -0700132 firstArchVersions := map[android.ArchType]int{
Dan Albertf5415d72017-08-17 16:19:59 -0700133 android.Arm: minVersion,
Colin Crossce87b802017-04-13 13:00:26 -0700134 android.Arm64: 21,
Dan Albertf5415d72017-08-17 16:19:59 -0700135 android.Mips: minVersion,
Colin Crossce87b802017-04-13 13:00:26 -0700136 android.Mips64: 21,
Dan Albertf5415d72017-08-17 16:19:59 -0700137 android.X86: minVersion,
Colin Crossce87b802017-04-13 13:00:26 -0700138 android.X86_64: 21,
Dan Albert914449f2016-06-17 16:45:24 -0700139 }
140
Colin Crossce87b802017-04-13 13:00:26 -0700141 firstArchVersion, ok := firstArchVersions[arch.ArchType]
Dan Albert2e5d7d42017-03-29 18:22:39 -0700142 if !ok {
Colin Crossce87b802017-04-13 13:00:26 -0700143 panic(fmt.Errorf("Arch %q not found in firstArchVersions", arch.ArchType))
Dan Albert2e5d7d42017-03-29 18:22:39 -0700144 }
145
146 if apiLevel == "minimum" {
147 return strconv.Itoa(firstArchVersion), nil
148 }
149
Dan Albert914449f2016-06-17 16:45:24 -0700150 // If the NDK drops support for a platform version, we don't want to have to
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700151 // fix up every module that was using it as its SDK version. Clip to the
Dan Albert914449f2016-06-17 16:45:24 -0700152 // supported version here instead.
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700153 version, err := strconv.Atoi(apiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700154 if err != nil {
Dan Albert90f7a4d2016-11-08 14:34:24 -0800155 return "", fmt.Errorf("API level must be an integer (is %q)", apiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700156 }
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700157 version = intMax(version, minVersion)
158
Dan Albert90f7a4d2016-11-08 14:34:24 -0800159 return strconv.Itoa(intMax(version, firstArchVersion)), nil
160}
161
162func getFirstGeneratedVersion(firstSupportedVersion string, platformVersion int) (int, error) {
163 if firstSupportedVersion == "current" {
164 return platformVersion + 1, nil
165 }
166
167 return strconv.Atoi(firstSupportedVersion)
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700168}
169
Colin Cross0ea8ba82019-06-06 14:33:29 -0700170func shouldUseVersionScript(ctx android.BaseModuleContext, stub *stubDecorator) (bool, error) {
Ryan Prichard37ebbde2018-07-24 12:37:24 -0700171 // unversioned_until is normally empty, in which case we should use the version script.
172 if String(stub.properties.Unversioned_until) == "" {
173 return true, nil
174 }
Dan Albert98dbb3b2017-01-03 15:16:29 -0800175
Nan Zhang0007d812017-11-07 10:57:05 -0800176 if String(stub.properties.Unversioned_until) == "current" {
Dan Albert022e7a32017-01-05 15:49:09 -0800177 if stub.properties.ApiLevel == "current" {
178 return true, nil
179 } else {
180 return false, nil
181 }
182 }
183
Dan Albert98dbb3b2017-01-03 15:16:29 -0800184 if stub.properties.ApiLevel == "current" {
185 return true, nil
186 }
187
Dan Alberte67144e2018-05-03 15:42:34 -0700188 unversionedUntil, err := android.ApiStrToNum(ctx, String(stub.properties.Unversioned_until))
Dan Albert98dbb3b2017-01-03 15:16:29 -0800189 if err != nil {
190 return true, err
191 }
192
Ryan Prichard37ebbde2018-07-24 12:37:24 -0700193 version, err := android.ApiStrToNum(ctx, stub.properties.ApiLevel)
194 if err != nil {
195 return true, err
Dan Alberte67144e2018-05-03 15:42:34 -0700196 }
197
Dan Albert98dbb3b2017-01-03 15:16:29 -0800198 return version >= unversionedUntil, nil
199}
200
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700201func generateStubApiVariants(mctx android.BottomUpMutatorContext, c *stubDecorator) {
Colin Cross6510f912017-11-29 00:27:14 -0800202 platformVersion := mctx.Config().PlatformSdkVersionInt()
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700203
Nan Zhang0007d812017-11-07 10:57:05 -0800204 firstSupportedVersion, err := normalizeNdkApiLevel(mctx, String(c.properties.First_version),
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700205 mctx.Arch())
206 if err != nil {
207 mctx.PropertyErrorf("first_version", err.Error())
Dan Albert914449f2016-06-17 16:45:24 -0700208 }
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700209
Dan Albert90f7a4d2016-11-08 14:34:24 -0800210 firstGenVersion, err := getFirstGeneratedVersion(firstSupportedVersion, platformVersion)
211 if err != nil {
212 // In theory this is impossible because we've already run this through
213 // normalizeNdkApiLevel above.
214 mctx.PropertyErrorf("first_version", err.Error())
215 }
216
Dan Albertfd86e9e2016-11-08 13:35:12 -0800217 var versionStrs []string
Dan Albert90f7a4d2016-11-08 14:34:24 -0800218 for version := firstGenVersion; version <= platformVersion; version++ {
Dan Albertfd86e9e2016-11-08 13:35:12 -0800219 versionStrs = append(versionStrs, strconv.Itoa(version))
Dan Albert914449f2016-06-17 16:45:24 -0700220 }
Colin Cross6510f912017-11-29 00:27:14 -0800221 versionStrs = append(versionStrs, mctx.Config().PlatformVersionActiveCodenames()...)
Dan Albertfd86e9e2016-11-08 13:35:12 -0800222 versionStrs = append(versionStrs, "current")
Dan Albert914449f2016-06-17 16:45:24 -0700223
224 modules := mctx.CreateVariations(versionStrs...)
225 for i, module := range modules {
Dan Albertfd86e9e2016-11-08 13:35:12 -0800226 module.(*Module).compiler.(*stubDecorator).properties.ApiLevel = versionStrs[i]
Dan Albert914449f2016-06-17 16:45:24 -0700227 }
228}
229
230func ndkApiMutator(mctx android.BottomUpMutatorContext) {
231 if m, ok := mctx.Module().(*Module); ok {
Colin Crossd4025822017-04-13 12:53:07 -0700232 if m.Enabled() {
233 if compiler, ok := m.compiler.(*stubDecorator); ok {
234 generateStubApiVariants(mctx, compiler)
235 }
Dan Albert914449f2016-06-17 16:45:24 -0700236 }
237 }
238}
239
Colin Crossb916a382016-07-29 17:28:03 -0700240func (c *stubDecorator) compilerInit(ctx BaseModuleContext) {
Dan Albert7e9d2952016-08-04 13:02:36 -0700241 c.baseCompiler.compilerInit(ctx)
242
Dan Willemsen01a90592017-04-07 15:21:13 -0700243 name := ctx.baseModuleName()
244 if strings.HasSuffix(name, ndkLibrarySuffix) {
245 ctx.PropertyErrorf("name", "Do not append %q manually, just use the base name", ndkLibrarySuffix)
246 }
247
Colin Crosse8a67a72016-08-07 21:17:54 -0700248 ndkMigratedLibsLock.Lock()
249 defer ndkMigratedLibsLock.Unlock()
Dan Albert7e9d2952016-08-04 13:02:36 -0700250 for _, lib := range ndkMigratedLibs {
251 if lib == name {
252 return
253 }
254 }
255 ndkMigratedLibs = append(ndkMigratedLibs, name)
256}
257
George Burgess IVf5310e32017-07-19 11:39:53 -0700258func addStubLibraryCompilerFlags(flags Flags) Flags {
259 flags.CFlags = append(flags.CFlags,
260 // We're knowingly doing some otherwise unsightly things with builtin
261 // functions here. We're just generating stub libraries, so ignore it.
262 "-Wno-incompatible-library-redeclaration",
263 "-Wno-builtin-requires-header",
264 "-Wno-invalid-noreturn",
Chih-Hung Hsieh64a38dc2017-11-14 14:09:14 -0800265 "-Wall",
266 "-Werror",
George Burgess IVf5310e32017-07-19 11:39:53 -0700267 // These libraries aren't actually used. Don't worry about unwinding
268 // (avoids the need to link an unwinder into a fake library).
269 "-fno-unwind-tables",
270 )
271 return flags
272}
273
Colin Crossf18e1102017-11-16 14:33:08 -0800274func (stub *stubDecorator) compilerFlags(ctx ModuleContext, flags Flags, deps PathDeps) Flags {
275 flags = stub.baseCompiler.compilerFlags(ctx, flags, deps)
George Burgess IVf5310e32017-07-19 11:39:53 -0700276 return addStubLibraryCompilerFlags(flags)
277}
278
Jiyong Park3fd0baf2018-12-07 16:25:39 +0900279func compileStubLibrary(ctx ModuleContext, flags Flags, symbolFile, apiLevel, genstubFlags string) (Objects, android.ModuleGenPath) {
Dan Albert914449f2016-06-17 16:45:24 -0700280 arch := ctx.Arch().ArchType.String()
281
Dan Willemsenb916b802017-03-19 13:44:32 -0700282 stubSrcPath := android.PathForModuleGen(ctx, "stub.c")
283 versionScriptPath := android.PathForModuleGen(ctx, "stub.map")
284 symbolFilePath := android.PathForModuleSrc(ctx, symbolFile)
Dan Albert49927d22017-03-28 15:00:46 -0700285 apiLevelsJson := android.GetApiLevelsJson(ctx)
Colin Crossae887032017-10-23 17:16:14 -0700286 ctx.Build(pctx, android.BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -0700287 Rule: genStubSrc,
288 Description: "generate stubs " + symbolFilePath.Rel(),
289 Outputs: []android.WritablePath{stubSrcPath, versionScriptPath},
290 Input: symbolFilePath,
291 Implicits: []android.Path{apiLevelsJson},
Dan Albert914449f2016-06-17 16:45:24 -0700292 Args: map[string]string{
293 "arch": arch,
Dan Willemsenb916b802017-03-19 13:44:32 -0700294 "apiLevel": apiLevel,
Dan Albert49927d22017-03-28 15:00:46 -0700295 "apiMap": apiLevelsJson.String(),
Jiyong Park3fd0baf2018-12-07 16:25:39 +0900296 "flags": genstubFlags,
Dan Albert914449f2016-06-17 16:45:24 -0700297 },
298 })
299
Dan Albert914449f2016-06-17 16:45:24 -0700300 subdir := ""
Colin Cross2f336352016-10-26 10:03:47 -0700301 srcs := []android.Path{stubSrcPath}
Pirama Arumuga Nainar70ba5a32017-12-19 15:11:01 -0800302 return compileObjs(ctx, flagsToBuilderFlags(flags), subdir, srcs, nil, nil), versionScriptPath
Dan Willemsenb916b802017-03-19 13:44:32 -0700303}
304
305func (c *stubDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Nan Zhang0007d812017-11-07 10:57:05 -0800306 if !strings.HasSuffix(String(c.properties.Symbol_file), ".map.txt") {
Dan Albert15be0c62017-06-13 15:14:56 -0700307 ctx.PropertyErrorf("symbol_file", "must end with .map.txt")
308 }
309
Nan Zhang0007d812017-11-07 10:57:05 -0800310 objs, versionScript := compileStubLibrary(ctx, flags, String(c.properties.Symbol_file),
311 c.properties.ApiLevel, "")
Dan Willemsenb916b802017-03-19 13:44:32 -0700312 c.versionScriptPath = versionScript
313 return objs
Dan Albert914449f2016-06-17 16:45:24 -0700314}
315
Colin Cross37047f12016-12-13 17:06:13 -0800316func (linker *stubDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Dan Albert914449f2016-06-17 16:45:24 -0700317 return Deps{}
318}
319
Dan Willemsen01a90592017-04-07 15:21:13 -0700320func (linker *stubDecorator) Name(name string) string {
321 return name + ndkLibrarySuffix
322}
323
Colin Crossb916a382016-07-29 17:28:03 -0700324func (stub *stubDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Dan Willemsen01a90592017-04-07 15:21:13 -0700325 stub.libraryDecorator.libName = ctx.baseModuleName()
Colin Crossb916a382016-07-29 17:28:03 -0700326 return stub.libraryDecorator.linkerFlags(ctx, flags)
Dan Albert914449f2016-06-17 16:45:24 -0700327}
328
Colin Crossb916a382016-07-29 17:28:03 -0700329func (stub *stubDecorator) link(ctx ModuleContext, flags Flags, deps PathDeps,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700330 objs Objects) android.Path {
Dan Albert2bc91ba2016-07-28 17:40:28 -0700331
Dan Alberte67144e2018-05-03 15:42:34 -0700332 useVersionScript, err := shouldUseVersionScript(ctx, stub)
Dan Albert98dbb3b2017-01-03 15:16:29 -0800333 if err != nil {
334 ctx.ModuleErrorf(err.Error())
335 }
336
337 if useVersionScript {
338 linkerScriptFlag := "-Wl,--version-script," + stub.versionScriptPath.String()
339 flags.LdFlags = append(flags.LdFlags, linkerScriptFlag)
Dan Willemsen939408a2019-06-10 18:02:25 -0700340 flags.LdFlagsDeps = append(flags.LdFlagsDeps, stub.versionScriptPath)
Dan Albert98dbb3b2017-01-03 15:16:29 -0800341 }
342
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700343 return stub.libraryDecorator.link(ctx, flags, deps, objs)
Dan Albert2bc91ba2016-07-28 17:40:28 -0700344}
345
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -0700346func (stub *stubDecorator) nativeCoverage() bool {
347 return false
348}
349
Colin Crossb916a382016-07-29 17:28:03 -0700350func (stub *stubDecorator) install(ctx ModuleContext, path android.Path) {
Dan Albert914449f2016-06-17 16:45:24 -0700351 arch := ctx.Target().Arch.ArchType.Name
Colin Crossb916a382016-07-29 17:28:03 -0700352 apiLevel := stub.properties.ApiLevel
Dan Albert914449f2016-06-17 16:45:24 -0700353
354 // arm64 isn't actually a multilib toolchain, so unlike the other LP64
355 // architectures it's just installed to lib.
356 libDir := "lib"
357 if ctx.toolchain().Is64Bit() && arch != "arm64" {
358 libDir = "lib64"
359 }
360
361 installDir := getNdkInstallBase(ctx).Join(ctx, fmt.Sprintf(
Dan Albertfd86e9e2016-11-08 13:35:12 -0800362 "platforms/android-%s/arch-%s/usr/%s", apiLevel, arch, libDir))
Colin Cross0875c522017-11-28 17:34:01 -0800363 stub.installPath = ctx.InstallFile(installDir, path.Base(), path)
Dan Albert914449f2016-06-17 16:45:24 -0700364}
365
Colin Cross36242852017-06-23 15:06:31 -0700366func newStubLibrary() *Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800367 module, library := NewLibrary(android.DeviceSupported)
368 library.BuildOnlyShared()
Dan Albert914449f2016-06-17 16:45:24 -0700369 module.stl = nil
Colin Crossb916a382016-07-29 17:28:03 -0700370 module.sanitize = nil
Nan Zhang0007d812017-11-07 10:57:05 -0800371 library.StripProperties.Strip.None = BoolPtr(true)
Dan Albert914449f2016-06-17 16:45:24 -0700372
Colin Crossb916a382016-07-29 17:28:03 -0700373 stub := &stubDecorator{
374 libraryDecorator: library,
375 }
376 module.compiler = stub
377 module.linker = stub
378 module.installer = stub
Dan Albert914449f2016-06-17 16:45:24 -0700379
Colin Cross36242852017-06-23 15:06:31 -0700380 module.AddProperties(&stub.properties, &library.MutatedProperties)
381
382 return module
Dan Albert914449f2016-06-17 16:45:24 -0700383}
384
Patrice Arruda6ea42112019-04-03 08:43:30 -0700385// ndk_library creates a stub library that exposes dummy implementation
386// of functions and variables for use at build time only.
Colin Cross36242852017-06-23 15:06:31 -0700387func ndkLibraryFactory() android.Module {
388 module := newStubLibrary()
389 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibBoth)
dimitry03dc3f62019-05-09 14:07:34 +0200390 module.ModuleBase.EnableNativeBridgeSupportByDefault()
Colin Cross36242852017-06-23 15:06:31 -0700391 return module
Dan Albert914449f2016-06-17 16:45:24 -0700392}