blob: 066fb988e7c207843b3c03c82f34d65c7b3488db [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 " +
34 "$apiMap $vndk $in $out",
Dan Albert914449f2016-06-17 16:45:24 -070035 CommandDeps: []string{"$toolPath"},
Dan Albert49927d22017-03-28 15:00:46 -070036 }, "arch", "apiLevel", "apiMap", "vndk")
Dan Albert914449f2016-06-17 16:45:24 -070037
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//
Dan Albert914449f2016-06-17 16:45:24 -070068// Example:
69//
70// ndk_library {
Dan Willemsen01a90592017-04-07 15:21:13 -070071// name: "libfoo",
Dan Albert914449f2016-06-17 16:45:24 -070072// symbol_file: "libfoo.map.txt",
73// first_version: "9",
74// }
75//
76type libraryProperties struct {
77 // Relative path to the symbol map.
78 // An example file can be seen here: TODO(danalbert): Make an example.
Nan Zhang0007d812017-11-07 10:57:05 -080079 Symbol_file *string
Dan Albert914449f2016-06-17 16:45:24 -070080
81 // The first API level a library was available. A library will be generated
82 // for every API level beginning with this one.
Nan Zhang0007d812017-11-07 10:57:05 -080083 First_version *string
Dan Albert914449f2016-06-17 16:45:24 -070084
Dan Albert98dbb3b2017-01-03 15:16:29 -080085 // The first API level that library should have the version script applied.
86 // This defaults to the value of first_version, and should almost never be
87 // used. This is only needed to work around platform bugs like
88 // https://github.com/android-ndk/ndk/issues/265.
Nan Zhang0007d812017-11-07 10:57:05 -080089 Unversioned_until *string
Dan Albert98dbb3b2017-01-03 15:16:29 -080090
Dan Albert914449f2016-06-17 16:45:24 -070091 // Private property for use by the mutator that splits per-API level.
Dan Albertfd86e9e2016-11-08 13:35:12 -080092 ApiLevel string `blueprint:"mutated"`
Dan Albert914449f2016-06-17 16:45:24 -070093}
94
Colin Crossb916a382016-07-29 17:28:03 -070095type stubDecorator struct {
96 *libraryDecorator
Dan Albert914449f2016-06-17 16:45:24 -070097
98 properties libraryProperties
Dan Albert2bc91ba2016-07-28 17:40:28 -070099
Colin Crossb916a382016-07-29 17:28:03 -0700100 versionScriptPath android.ModuleGenPath
101 installPath string
Dan Albert914449f2016-06-17 16:45:24 -0700102}
103
104// OMG GO
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700105func intMax(a int, b int) int {
106 if a > b {
Dan Albert914449f2016-06-17 16:45:24 -0700107 return a
108 } else {
109 return b
110 }
111}
112
Dan Albertf5415d72017-08-17 16:19:59 -0700113func normalizeNdkApiLevel(ctx android.BaseContext, apiLevel string,
114 arch android.Arch) (string, error) {
115
Dan Albert90f7a4d2016-11-08 14:34:24 -0800116 if apiLevel == "current" {
117 return apiLevel, nil
118 }
119
Dan Albertf5415d72017-08-17 16:19:59 -0700120 minVersion := ctx.AConfig().MinSupportedSdkVersion()
Colin Crossce87b802017-04-13 13:00:26 -0700121 firstArchVersions := map[android.ArchType]int{
Dan Albertf5415d72017-08-17 16:19:59 -0700122 android.Arm: minVersion,
Colin Crossce87b802017-04-13 13:00:26 -0700123 android.Arm64: 21,
Dan Albertf5415d72017-08-17 16:19:59 -0700124 android.Mips: minVersion,
Colin Crossce87b802017-04-13 13:00:26 -0700125 android.Mips64: 21,
Dan Albertf5415d72017-08-17 16:19:59 -0700126 android.X86: minVersion,
Colin Crossce87b802017-04-13 13:00:26 -0700127 android.X86_64: 21,
Dan Albert914449f2016-06-17 16:45:24 -0700128 }
129
Colin Crossce87b802017-04-13 13:00:26 -0700130 firstArchVersion, ok := firstArchVersions[arch.ArchType]
Dan Albert2e5d7d42017-03-29 18:22:39 -0700131 if !ok {
Colin Crossce87b802017-04-13 13:00:26 -0700132 panic(fmt.Errorf("Arch %q not found in firstArchVersions", arch.ArchType))
Dan Albert2e5d7d42017-03-29 18:22:39 -0700133 }
134
135 if apiLevel == "minimum" {
136 return strconv.Itoa(firstArchVersion), nil
137 }
138
Dan Albert914449f2016-06-17 16:45:24 -0700139 // If the NDK drops support for a platform version, we don't want to have to
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700140 // fix up every module that was using it as its SDK version. Clip to the
Dan Albert914449f2016-06-17 16:45:24 -0700141 // supported version here instead.
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700142 version, err := strconv.Atoi(apiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700143 if err != nil {
Dan Albert90f7a4d2016-11-08 14:34:24 -0800144 return "", fmt.Errorf("API level must be an integer (is %q)", apiLevel)
Dan Albert914449f2016-06-17 16:45:24 -0700145 }
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700146 version = intMax(version, minVersion)
147
Dan Albert90f7a4d2016-11-08 14:34:24 -0800148 return strconv.Itoa(intMax(version, firstArchVersion)), nil
149}
150
151func getFirstGeneratedVersion(firstSupportedVersion string, platformVersion int) (int, error) {
152 if firstSupportedVersion == "current" {
153 return platformVersion + 1, nil
154 }
155
156 return strconv.Atoi(firstSupportedVersion)
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700157}
158
Dan Albert98dbb3b2017-01-03 15:16:29 -0800159func shouldUseVersionScript(stub *stubDecorator) (bool, error) {
160 // unversioned_until is normally empty, in which case we should use the version script.
Nan Zhang0007d812017-11-07 10:57:05 -0800161 if String(stub.properties.Unversioned_until) == "" {
Dan Albert98dbb3b2017-01-03 15:16:29 -0800162 return true, nil
163 }
164
Nan Zhang0007d812017-11-07 10:57:05 -0800165 if String(stub.properties.Unversioned_until) == "current" {
Dan Albert022e7a32017-01-05 15:49:09 -0800166 if stub.properties.ApiLevel == "current" {
167 return true, nil
168 } else {
169 return false, nil
170 }
171 }
172
Dan Albert98dbb3b2017-01-03 15:16:29 -0800173 if stub.properties.ApiLevel == "current" {
174 return true, nil
175 }
176
Nan Zhang0007d812017-11-07 10:57:05 -0800177 unversionedUntil, err := strconv.Atoi(String(stub.properties.Unversioned_until))
Dan Albert98dbb3b2017-01-03 15:16:29 -0800178 if err != nil {
179 return true, err
180 }
181
182 version, err := strconv.Atoi(stub.properties.ApiLevel)
183 if err != nil {
184 return true, err
185 }
186
187 return version >= unversionedUntil, nil
188}
189
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700190func generateStubApiVariants(mctx android.BottomUpMutatorContext, c *stubDecorator) {
Dan Albert90f7a4d2016-11-08 14:34:24 -0800191 platformVersion := mctx.AConfig().PlatformSdkVersionInt()
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700192
Nan Zhang0007d812017-11-07 10:57:05 -0800193 firstSupportedVersion, err := normalizeNdkApiLevel(mctx, String(c.properties.First_version),
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700194 mctx.Arch())
195 if err != nil {
196 mctx.PropertyErrorf("first_version", err.Error())
Dan Albert914449f2016-06-17 16:45:24 -0700197 }
Dan Albert7fa7b2e2016-08-05 16:37:52 -0700198
Dan Albert90f7a4d2016-11-08 14:34:24 -0800199 firstGenVersion, err := getFirstGeneratedVersion(firstSupportedVersion, platformVersion)
200 if err != nil {
201 // In theory this is impossible because we've already run this through
202 // normalizeNdkApiLevel above.
203 mctx.PropertyErrorf("first_version", err.Error())
204 }
205
Dan Albertfd86e9e2016-11-08 13:35:12 -0800206 var versionStrs []string
Dan Albert90f7a4d2016-11-08 14:34:24 -0800207 for version := firstGenVersion; version <= platformVersion; version++ {
Dan Albertfd86e9e2016-11-08 13:35:12 -0800208 versionStrs = append(versionStrs, strconv.Itoa(version))
Dan Albert914449f2016-06-17 16:45:24 -0700209 }
Dan Albert31384de2017-07-28 12:39:46 -0700210 versionStrs = append(versionStrs, mctx.AConfig().PlatformVersionActiveCodenames()...)
Dan Albertfd86e9e2016-11-08 13:35:12 -0800211 versionStrs = append(versionStrs, "current")
Dan Albert914449f2016-06-17 16:45:24 -0700212
213 modules := mctx.CreateVariations(versionStrs...)
214 for i, module := range modules {
Dan Albertfd86e9e2016-11-08 13:35:12 -0800215 module.(*Module).compiler.(*stubDecorator).properties.ApiLevel = versionStrs[i]
Dan Albert914449f2016-06-17 16:45:24 -0700216 }
217}
218
219func ndkApiMutator(mctx android.BottomUpMutatorContext) {
220 if m, ok := mctx.Module().(*Module); ok {
Colin Crossd4025822017-04-13 12:53:07 -0700221 if m.Enabled() {
222 if compiler, ok := m.compiler.(*stubDecorator); ok {
223 generateStubApiVariants(mctx, compiler)
224 }
Dan Albert914449f2016-06-17 16:45:24 -0700225 }
226 }
227}
228
Colin Crossb916a382016-07-29 17:28:03 -0700229func (c *stubDecorator) compilerInit(ctx BaseModuleContext) {
Dan Albert7e9d2952016-08-04 13:02:36 -0700230 c.baseCompiler.compilerInit(ctx)
231
Dan Willemsen01a90592017-04-07 15:21:13 -0700232 name := ctx.baseModuleName()
233 if strings.HasSuffix(name, ndkLibrarySuffix) {
234 ctx.PropertyErrorf("name", "Do not append %q manually, just use the base name", ndkLibrarySuffix)
235 }
236
Colin Crosse8a67a72016-08-07 21:17:54 -0700237 ndkMigratedLibsLock.Lock()
238 defer ndkMigratedLibsLock.Unlock()
Dan Albert7e9d2952016-08-04 13:02:36 -0700239 for _, lib := range ndkMigratedLibs {
240 if lib == name {
241 return
242 }
243 }
244 ndkMigratedLibs = append(ndkMigratedLibs, name)
245}
246
George Burgess IVf5310e32017-07-19 11:39:53 -0700247func addStubLibraryCompilerFlags(flags Flags) Flags {
248 flags.CFlags = append(flags.CFlags,
249 // We're knowingly doing some otherwise unsightly things with builtin
250 // functions here. We're just generating stub libraries, so ignore it.
251 "-Wno-incompatible-library-redeclaration",
252 "-Wno-builtin-requires-header",
253 "-Wno-invalid-noreturn",
254 // These libraries aren't actually used. Don't worry about unwinding
255 // (avoids the need to link an unwinder into a fake library).
256 "-fno-unwind-tables",
257 )
258 return flags
259}
260
261func (stub *stubDecorator) compilerFlags(ctx ModuleContext, flags Flags) Flags {
262 flags = stub.baseCompiler.compilerFlags(ctx, flags)
263 return addStubLibraryCompilerFlags(flags)
264}
265
Dan Willemsenb916b802017-03-19 13:44:32 -0700266func compileStubLibrary(ctx ModuleContext, flags Flags, symbolFile, apiLevel, vndk string) (Objects, android.ModuleGenPath) {
Dan Albert914449f2016-06-17 16:45:24 -0700267 arch := ctx.Arch().ArchType.String()
268
Dan Willemsenb916b802017-03-19 13:44:32 -0700269 stubSrcPath := android.PathForModuleGen(ctx, "stub.c")
270 versionScriptPath := android.PathForModuleGen(ctx, "stub.map")
271 symbolFilePath := android.PathForModuleSrc(ctx, symbolFile)
Dan Albert49927d22017-03-28 15:00:46 -0700272 apiLevelsJson := android.GetApiLevelsJson(ctx)
Colin Crossae887032017-10-23 17:16:14 -0700273 ctx.Build(pctx, android.BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -0700274 Rule: genStubSrc,
275 Description: "generate stubs " + symbolFilePath.Rel(),
276 Outputs: []android.WritablePath{stubSrcPath, versionScriptPath},
277 Input: symbolFilePath,
278 Implicits: []android.Path{apiLevelsJson},
Dan Albert914449f2016-06-17 16:45:24 -0700279 Args: map[string]string{
280 "arch": arch,
Dan Willemsenb916b802017-03-19 13:44:32 -0700281 "apiLevel": apiLevel,
Dan Albert49927d22017-03-28 15:00:46 -0700282 "apiMap": apiLevelsJson.String(),
Dan Willemsenb916b802017-03-19 13:44:32 -0700283 "vndk": vndk,
Dan Albert914449f2016-06-17 16:45:24 -0700284 },
285 })
286
Dan Albert914449f2016-06-17 16:45:24 -0700287 subdir := ""
Colin Cross2f336352016-10-26 10:03:47 -0700288 srcs := []android.Path{stubSrcPath}
Dan Willemsenb916b802017-03-19 13:44:32 -0700289 return compileObjs(ctx, flagsToBuilderFlags(flags), subdir, srcs, nil), versionScriptPath
290}
291
292func (c *stubDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Nan Zhang0007d812017-11-07 10:57:05 -0800293 if !strings.HasSuffix(String(c.properties.Symbol_file), ".map.txt") {
Dan Albert15be0c62017-06-13 15:14:56 -0700294 ctx.PropertyErrorf("symbol_file", "must end with .map.txt")
295 }
296
Nan Zhang0007d812017-11-07 10:57:05 -0800297 objs, versionScript := compileStubLibrary(ctx, flags, String(c.properties.Symbol_file),
298 c.properties.ApiLevel, "")
Dan Willemsenb916b802017-03-19 13:44:32 -0700299 c.versionScriptPath = versionScript
300 return objs
Dan Albert914449f2016-06-17 16:45:24 -0700301}
302
Colin Cross37047f12016-12-13 17:06:13 -0800303func (linker *stubDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Dan Albert914449f2016-06-17 16:45:24 -0700304 return Deps{}
305}
306
Dan Willemsen01a90592017-04-07 15:21:13 -0700307func (linker *stubDecorator) Name(name string) string {
308 return name + ndkLibrarySuffix
309}
310
Colin Crossb916a382016-07-29 17:28:03 -0700311func (stub *stubDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Dan Willemsen01a90592017-04-07 15:21:13 -0700312 stub.libraryDecorator.libName = ctx.baseModuleName()
Colin Crossb916a382016-07-29 17:28:03 -0700313 return stub.libraryDecorator.linkerFlags(ctx, flags)
Dan Albert914449f2016-06-17 16:45:24 -0700314}
315
Colin Crossb916a382016-07-29 17:28:03 -0700316func (stub *stubDecorator) link(ctx ModuleContext, flags Flags, deps PathDeps,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700317 objs Objects) android.Path {
Dan Albert2bc91ba2016-07-28 17:40:28 -0700318
Dan Albert98dbb3b2017-01-03 15:16:29 -0800319 useVersionScript, err := shouldUseVersionScript(stub)
320 if err != nil {
321 ctx.ModuleErrorf(err.Error())
322 }
323
324 if useVersionScript {
325 linkerScriptFlag := "-Wl,--version-script," + stub.versionScriptPath.String()
326 flags.LdFlags = append(flags.LdFlags, linkerScriptFlag)
327 }
328
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700329 return stub.libraryDecorator.link(ctx, flags, deps, objs)
Dan Albert2bc91ba2016-07-28 17:40:28 -0700330}
331
Colin Crossb916a382016-07-29 17:28:03 -0700332func (stub *stubDecorator) install(ctx ModuleContext, path android.Path) {
Dan Albert914449f2016-06-17 16:45:24 -0700333 arch := ctx.Target().Arch.ArchType.Name
Colin Crossb916a382016-07-29 17:28:03 -0700334 apiLevel := stub.properties.ApiLevel
Dan Albert914449f2016-06-17 16:45:24 -0700335
336 // arm64 isn't actually a multilib toolchain, so unlike the other LP64
337 // architectures it's just installed to lib.
338 libDir := "lib"
339 if ctx.toolchain().Is64Bit() && arch != "arm64" {
340 libDir = "lib64"
341 }
342
343 installDir := getNdkInstallBase(ctx).Join(ctx, fmt.Sprintf(
Dan Albertfd86e9e2016-11-08 13:35:12 -0800344 "platforms/android-%s/arch-%s/usr/%s", apiLevel, arch, libDir))
Colin Cross5c517922017-08-31 12:29:17 -0700345 stub.installPath = ctx.InstallFile(installDir, path.Base(), path).String()
Dan Albert914449f2016-06-17 16:45:24 -0700346}
347
Colin Cross36242852017-06-23 15:06:31 -0700348func newStubLibrary() *Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800349 module, library := NewLibrary(android.DeviceSupported)
350 library.BuildOnlyShared()
Dan Albert914449f2016-06-17 16:45:24 -0700351 module.stl = nil
Colin Crossb916a382016-07-29 17:28:03 -0700352 module.sanitize = nil
Nan Zhang0007d812017-11-07 10:57:05 -0800353 library.StripProperties.Strip.None = BoolPtr(true)
Dan Albert914449f2016-06-17 16:45:24 -0700354
Colin Crossb916a382016-07-29 17:28:03 -0700355 stub := &stubDecorator{
356 libraryDecorator: library,
357 }
358 module.compiler = stub
359 module.linker = stub
360 module.installer = stub
Dan Albert914449f2016-06-17 16:45:24 -0700361
Colin Cross36242852017-06-23 15:06:31 -0700362 module.AddProperties(&stub.properties, &library.MutatedProperties)
363
364 return module
Dan Albert914449f2016-06-17 16:45:24 -0700365}
366
Colin Cross36242852017-06-23 15:06:31 -0700367func ndkLibraryFactory() android.Module {
368 module := newStubLibrary()
369 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibBoth)
370 return module
Dan Albert914449f2016-06-17 16:45:24 -0700371}