blob: f3d2ba16975ef53ce96d6afe4a3ca5a1d7e68b2b [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"
Dan Albertf1d14c72020-07-30 14:32:55 -070019 "path/filepath"
Dan Albert914449f2016-06-17 16:45:24 -070020 "strings"
Colin Crosse8a67a72016-08-07 21:17:54 -070021 "sync"
Dan Albert914449f2016-06-17 16:45:24 -070022
23 "github.com/google/blueprint"
Jiyong Parkee9b1172021-04-06 17:40:32 +090024 "github.com/google/blueprint/proptools"
Dan Albert914449f2016-06-17 16:45:24 -070025
26 "android/soong/android"
27)
28
sophiez58cabb72020-05-29 13:37:12 -070029func init() {
Dan Albert06f58af2020-06-22 15:10:31 -070030 pctx.HostBinToolVariable("ndkStubGenerator", "ndkstubgen")
sophiez58cabb72020-05-29 13:37:12 -070031 pctx.HostBinToolVariable("ndk_api_coverage_parser", "ndk_api_coverage_parser")
Dan Albertf1d14c72020-07-30 14:32:55 -070032 pctx.HostBinToolVariable("abidiff", "abidiff")
33 pctx.HostBinToolVariable("abitidy", "abitidy")
34 pctx.HostBinToolVariable("abidw", "abidw")
sophiez58cabb72020-05-29 13:37:12 -070035}
36
Dan Albert914449f2016-06-17 16:45:24 -070037var (
Colin Cross9d45bb72016-08-29 16:14:13 -070038 genStubSrc = pctx.AndroidStaticRule("genStubSrc",
Dan Albert914449f2016-06-17 16:45:24 -070039 blueprint.RuleParams{
Dan Albert06f58af2020-06-22 15:10:31 -070040 Command: "$ndkStubGenerator --arch $arch --api $apiLevel " +
41 "--api-map $apiMap $flags $in $out",
42 CommandDeps: []string{"$ndkStubGenerator"},
Jiyong Park3fd0baf2018-12-07 16:25:39 +090043 }, "arch", "apiLevel", "apiMap", "flags")
Dan Albert914449f2016-06-17 16:45:24 -070044
sophiez58cabb72020-05-29 13:37:12 -070045 parseNdkApiRule = pctx.AndroidStaticRule("parseNdkApiRule",
46 blueprint.RuleParams{
47 Command: "$ndk_api_coverage_parser $in $out --api-map $apiMap",
48 CommandDeps: []string{"$ndk_api_coverage_parser"},
49 }, "apiMap")
50
Dan Albertf1d14c72020-07-30 14:32:55 -070051 abidw = pctx.AndroidStaticRule("abidw",
52 blueprint.RuleParams{
53 Command: "$abidw --type-id-style hash --no-corpus-path " +
54 "--no-show-locs --no-comp-dir-path -w $symbolList $in | " +
55 "$abitidy --all -o $out",
56 CommandDeps: []string{"$abitidy", "$abidw"},
57 }, "symbolList")
58
59 abidiff = pctx.AndroidStaticRule("abidiff",
60 blueprint.RuleParams{
61 // Need to create *some* output for ninja. We don't want to use tee
62 // because we don't want to spam the build output with "nothing
63 // changed" messages, so redirect output message to $out, and if
64 // changes were detected print the output and fail.
65 Command: "$abidiff $args $in > $out || (cat $out && false)",
66 CommandDeps: []string{"$abidiff"},
67 }, "args")
68
Dan Albert914449f2016-06-17 16:45:24 -070069 ndkLibrarySuffix = ".ndk"
Colin Cross4d9c2d12016-07-29 12:48:20 -070070
Colin Cross95f1ca02020-10-29 20:47:22 -070071 ndkKnownLibsKey = android.NewOnceKey("ndkKnownLibsKey")
Dan Albertde5aade2020-06-30 12:32:51 -070072 // protects ndkKnownLibs writes during parallel BeginMutator.
73 ndkKnownLibsLock sync.Mutex
Dan Albertf1d14c72020-07-30 14:32:55 -070074
75 stubImplementation = dependencyTag{name: "stubImplementation"}
Dan Albert914449f2016-06-17 16:45:24 -070076)
77
Dan Albert1a246272020-07-06 14:49:35 -070078// The First_version and Unversioned_until properties of this struct should not
79// be used directly, but rather through the ApiLevel returning methods
80// firstVersion() and unversionedUntil().
81
Dan Albert914449f2016-06-17 16:45:24 -070082// Creates a stub shared library based on the provided version file.
83//
Dan Albert914449f2016-06-17 16:45:24 -070084// Example:
85//
86// ndk_library {
Dan Willemsen01a90592017-04-07 15:21:13 -070087// name: "libfoo",
Dan Albert914449f2016-06-17 16:45:24 -070088// symbol_file: "libfoo.map.txt",
89// first_version: "9",
90// }
91//
92type libraryProperties struct {
93 // Relative path to the symbol map.
94 // An example file can be seen here: TODO(danalbert): Make an example.
Nan Zhang0007d812017-11-07 10:57:05 -080095 Symbol_file *string
Dan Albert914449f2016-06-17 16:45:24 -070096
97 // The first API level a library was available. A library will be generated
98 // for every API level beginning with this one.
Nan Zhang0007d812017-11-07 10:57:05 -080099 First_version *string
Dan Albert914449f2016-06-17 16:45:24 -0700100
Dan Albert98dbb3b2017-01-03 15:16:29 -0800101 // The first API level that library should have the version script applied.
102 // This defaults to the value of first_version, and should almost never be
103 // used. This is only needed to work around platform bugs like
104 // https://github.com/android-ndk/ndk/issues/265.
Nan Zhang0007d812017-11-07 10:57:05 -0800105 Unversioned_until *string
Dan Albert914449f2016-06-17 16:45:24 -0700106}
107
Colin Crossb916a382016-07-29 17:28:03 -0700108type stubDecorator struct {
109 *libraryDecorator
Dan Albert914449f2016-06-17 16:45:24 -0700110
111 properties libraryProperties
Dan Albert2bc91ba2016-07-28 17:40:28 -0700112
sophiez58cabb72020-05-29 13:37:12 -0700113 versionScriptPath android.ModuleGenPath
114 parsedCoverageXmlPath android.ModuleOutPath
115 installPath android.Path
Dan Albertf1d14c72020-07-30 14:32:55 -0700116 abiDumpPath android.OutputPath
117 abiDiffPaths android.Paths
Dan Albert1a246272020-07-06 14:49:35 -0700118
119 apiLevel android.ApiLevel
120 firstVersion android.ApiLevel
121 unversionedUntil android.ApiLevel
Dan Albert914449f2016-06-17 16:45:24 -0700122}
123
Colin Cross0477b422020-10-13 18:43:54 -0700124var _ versionedInterface = (*stubDecorator)(nil)
125
Dan Albert1a246272020-07-06 14:49:35 -0700126func shouldUseVersionScript(ctx BaseModuleContext, stub *stubDecorator) bool {
127 return stub.apiLevel.GreaterThanOrEqualTo(stub.unversionedUntil)
Dan Albert98dbb3b2017-01-03 15:16:29 -0800128}
129
Colin Cross0477b422020-10-13 18:43:54 -0700130func (stub *stubDecorator) implementationModuleName(name string) string {
131 return strings.TrimSuffix(name, ndkLibrarySuffix)
132}
133
Colin Cross3572cf72020-10-01 15:58:11 -0700134func ndkLibraryVersions(ctx android.BaseMutatorContext, from android.ApiLevel) []string {
Dan Albert1a246272020-07-06 14:49:35 -0700135 var versions []android.ApiLevel
136 versionStrs := []string{}
137 for _, version := range ctx.Config().AllSupportedApiLevels() {
138 if version.GreaterThanOrEqualTo(from) {
139 versions = append(versions, version)
140 versionStrs = append(versionStrs, version.String())
141 }
Dan Albert914449f2016-06-17 16:45:24 -0700142 }
Dan Albert0b176c82020-07-23 16:43:25 -0700143 versionStrs = append(versionStrs, android.FutureApiLevel.String())
Dan Albert914449f2016-06-17 16:45:24 -0700144
Colin Cross5ec407b2020-09-30 11:41:33 -0700145 return versionStrs
146}
147
Colin Cross3572cf72020-10-01 15:58:11 -0700148func (this *stubDecorator) stubsVersions(ctx android.BaseMutatorContext) []string {
149 if !ctx.Module().Enabled() {
150 return nil
151 }
Dan Albertf1d14c72020-07-30 14:32:55 -0700152 if ctx.Os() != android.Android {
153 // These modules are always android.DeviceEnabled only, but
154 // those include Fuchsia devices, which we don't support.
155 ctx.Module().Disable()
156 return nil
157 }
158 if ctx.Target().NativeBridge == android.NativeBridgeEnabled {
159 ctx.Module().Disable()
160 return nil
161 }
Colin Cross3572cf72020-10-01 15:58:11 -0700162 firstVersion, err := nativeApiLevelFromUser(ctx,
163 String(this.properties.First_version))
164 if err != nil {
165 ctx.PropertyErrorf("first_version", err.Error())
166 return nil
167 }
168 return ndkLibraryVersions(ctx, firstVersion)
169}
170
Dan Albert1a246272020-07-06 14:49:35 -0700171func (this *stubDecorator) initializeProperties(ctx BaseModuleContext) bool {
Colin Cross5ec407b2020-09-30 11:41:33 -0700172 this.apiLevel = nativeApiLevelOrPanic(ctx, this.stubsVersion())
Dan Albert1a246272020-07-06 14:49:35 -0700173
174 var err error
175 this.firstVersion, err = nativeApiLevelFromUser(ctx,
176 String(this.properties.First_version))
177 if err != nil {
178 ctx.PropertyErrorf("first_version", err.Error())
179 return false
180 }
181
Jiyong Parkee9b1172021-04-06 17:40:32 +0900182 str := proptools.StringDefault(this.properties.Unversioned_until, "minimum")
183 this.unversionedUntil, err = nativeApiLevelFromUser(ctx, str)
Dan Albert1a246272020-07-06 14:49:35 -0700184 if err != nil {
185 ctx.PropertyErrorf("unversioned_until", err.Error())
186 return false
187 }
188
189 return true
190}
191
Colin Cross95f1ca02020-10-29 20:47:22 -0700192func getNDKKnownLibs(config android.Config) *[]string {
193 return config.Once(ndkKnownLibsKey, func() interface{} {
194 return &[]string{}
195 }).(*[]string)
196}
197
Colin Crossb916a382016-07-29 17:28:03 -0700198func (c *stubDecorator) compilerInit(ctx BaseModuleContext) {
Dan Albert7e9d2952016-08-04 13:02:36 -0700199 c.baseCompiler.compilerInit(ctx)
200
Dan Willemsen01a90592017-04-07 15:21:13 -0700201 name := ctx.baseModuleName()
202 if strings.HasSuffix(name, ndkLibrarySuffix) {
203 ctx.PropertyErrorf("name", "Do not append %q manually, just use the base name", ndkLibrarySuffix)
204 }
205
Dan Albertde5aade2020-06-30 12:32:51 -0700206 ndkKnownLibsLock.Lock()
207 defer ndkKnownLibsLock.Unlock()
Colin Cross95f1ca02020-10-29 20:47:22 -0700208 ndkKnownLibs := getNDKKnownLibs(ctx.Config())
209 for _, lib := range *ndkKnownLibs {
Dan Albert7e9d2952016-08-04 13:02:36 -0700210 if lib == name {
211 return
212 }
213 }
Colin Cross95f1ca02020-10-29 20:47:22 -0700214 *ndkKnownLibs = append(*ndkKnownLibs, name)
Dan Albert7e9d2952016-08-04 13:02:36 -0700215}
216
George Burgess IVf5310e32017-07-19 11:39:53 -0700217func addStubLibraryCompilerFlags(flags Flags) Flags {
Colin Cross4af21ed2019-11-04 09:37:55 -0800218 flags.Global.CFlags = append(flags.Global.CFlags,
George Burgess IVf5310e32017-07-19 11:39:53 -0700219 // We're knowingly doing some otherwise unsightly things with builtin
220 // functions here. We're just generating stub libraries, so ignore it.
221 "-Wno-incompatible-library-redeclaration",
Nick Desaulnierseb207442019-12-12 10:15:42 -0800222 "-Wno-incomplete-setjmp-declaration",
George Burgess IVf5310e32017-07-19 11:39:53 -0700223 "-Wno-builtin-requires-header",
224 "-Wno-invalid-noreturn",
Chih-Hung Hsieh64a38dc2017-11-14 14:09:14 -0800225 "-Wall",
226 "-Werror",
George Burgess IVf5310e32017-07-19 11:39:53 -0700227 // These libraries aren't actually used. Don't worry about unwinding
228 // (avoids the need to link an unwinder into a fake library).
229 "-fno-unwind-tables",
230 )
Jiyong Park48d75ef2019-11-21 15:11:49 +0900231 // All symbols in the stubs library should be visible.
232 if inList("-fvisibility=hidden", flags.Local.CFlags) {
233 flags.Local.CFlags = append(flags.Local.CFlags, "-fvisibility=default")
234 }
George Burgess IVf5310e32017-07-19 11:39:53 -0700235 return flags
236}
237
Colin Crossf18e1102017-11-16 14:33:08 -0800238func (stub *stubDecorator) compilerFlags(ctx ModuleContext, flags Flags, deps PathDeps) Flags {
239 flags = stub.baseCompiler.compilerFlags(ctx, flags, deps)
George Burgess IVf5310e32017-07-19 11:39:53 -0700240 return addStubLibraryCompilerFlags(flags)
241}
242
Dan Albertf1d14c72020-07-30 14:32:55 -0700243type ndkApiOutputs struct {
244 stubSrc android.ModuleGenPath
245 versionScript android.ModuleGenPath
246 symbolList android.ModuleGenPath
247}
248
249func parseNativeAbiDefinition(ctx ModuleContext, symbolFile string,
250 apiLevel android.ApiLevel, genstubFlags string) ndkApiOutputs {
Dan Albert914449f2016-06-17 16:45:24 -0700251
Dan Willemsenb916b802017-03-19 13:44:32 -0700252 stubSrcPath := android.PathForModuleGen(ctx, "stub.c")
253 versionScriptPath := android.PathForModuleGen(ctx, "stub.map")
254 symbolFilePath := android.PathForModuleSrc(ctx, symbolFile)
Dan Albertf1d14c72020-07-30 14:32:55 -0700255 symbolListPath := android.PathForModuleGen(ctx, "abi_symbol_list.txt")
Dan Albert49927d22017-03-28 15:00:46 -0700256 apiLevelsJson := android.GetApiLevelsJson(ctx)
Colin Crossae887032017-10-23 17:16:14 -0700257 ctx.Build(pctx, android.BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -0700258 Rule: genStubSrc,
259 Description: "generate stubs " + symbolFilePath.Rel(),
Dan Albertf1d14c72020-07-30 14:32:55 -0700260 Outputs: []android.WritablePath{stubSrcPath, versionScriptPath,
261 symbolListPath},
262 Input: symbolFilePath,
263 Implicits: []android.Path{apiLevelsJson},
Dan Albert914449f2016-06-17 16:45:24 -0700264 Args: map[string]string{
Dan Albertf1d14c72020-07-30 14:32:55 -0700265 "arch": ctx.Arch().ArchType.String(),
266 "apiLevel": apiLevel.String(),
Dan Albert49927d22017-03-28 15:00:46 -0700267 "apiMap": apiLevelsJson.String(),
Jiyong Park3fd0baf2018-12-07 16:25:39 +0900268 "flags": genstubFlags,
Dan Albert914449f2016-06-17 16:45:24 -0700269 },
270 })
271
Dan Albertf1d14c72020-07-30 14:32:55 -0700272 return ndkApiOutputs{
273 stubSrc: stubSrcPath,
274 versionScript: versionScriptPath,
275 symbolList: symbolListPath,
276 }
277}
278
279func compileStubLibrary(ctx ModuleContext, flags Flags, src android.Path) Objects {
280 return compileObjs(ctx, flagsToBuilderFlags(flags), "",
281 android.Paths{src}, nil, nil)
Dan Willemsenb916b802017-03-19 13:44:32 -0700282}
283
sophiez58cabb72020-05-29 13:37:12 -0700284func parseSymbolFileForCoverage(ctx ModuleContext, symbolFile string) android.ModuleOutPath {
285 apiLevelsJson := android.GetApiLevelsJson(ctx)
286 symbolFilePath := android.PathForModuleSrc(ctx, symbolFile)
287 outputFileName := strings.Split(symbolFilePath.Base(), ".")[0]
288 parsedApiCoveragePath := android.PathForModuleOut(ctx, outputFileName+".xml")
289 ctx.Build(pctx, android.BuildParams{
290 Rule: parseNdkApiRule,
291 Description: "parse ndk api symbol file for api coverage: " + symbolFilePath.Rel(),
292 Outputs: []android.WritablePath{parsedApiCoveragePath},
293 Input: symbolFilePath,
sophiez148b3172020-06-11 17:27:56 -0700294 Implicits: []android.Path{apiLevelsJson},
sophiez58cabb72020-05-29 13:37:12 -0700295 Args: map[string]string{
296 "apiMap": apiLevelsJson.String(),
297 },
298 })
299 return parsedApiCoveragePath
300}
301
Dan Albertf1d14c72020-07-30 14:32:55 -0700302func (this *stubDecorator) findImplementationLibrary(ctx ModuleContext) android.Path {
303 dep := ctx.GetDirectDepWithTag(strings.TrimSuffix(ctx.ModuleName(), ndkLibrarySuffix),
304 stubImplementation)
305 if dep == nil {
306 ctx.ModuleErrorf("Could not find implementation for stub")
307 return nil
308 }
309 impl, ok := dep.(*Module)
310 if !ok {
311 ctx.ModuleErrorf("Implementation for stub is not correct module type")
312 }
313 output := impl.UnstrippedOutputFile()
314 if output == nil {
315 ctx.ModuleErrorf("implementation module (%s) has no output", impl)
316 return nil
317 }
318
319 return output
320}
321
322func (this *stubDecorator) libraryName(ctx ModuleContext) string {
323 return strings.TrimSuffix(ctx.ModuleName(), ndkLibrarySuffix)
324}
325
326func (this *stubDecorator) findPrebuiltAbiDump(ctx ModuleContext,
327 apiLevel android.ApiLevel) android.OptionalPath {
328
329 subpath := filepath.Join("prebuilts/abi-dumps/ndk", apiLevel.String(),
330 ctx.Arch().ArchType.String(), this.libraryName(ctx), "abi.xml")
331 return android.ExistentPathForSource(ctx, subpath)
332}
333
334// Feature flag.
335func canDumpAbi(module android.Module) bool {
336 return true
337}
338
339// Feature flag to disable diffing against prebuilts.
340func canDiffAbi(module android.Module) bool {
341 return false
342}
343
344func (this *stubDecorator) dumpAbi(ctx ModuleContext, symbolList android.Path) {
345 implementationLibrary := this.findImplementationLibrary(ctx)
346 this.abiDumpPath = getNdkAbiDumpInstallBase(ctx).Join(ctx,
347 this.apiLevel.String(), ctx.Arch().ArchType.String(),
348 this.libraryName(ctx), "abi.xml")
349 ctx.Build(pctx, android.BuildParams{
350 Rule: abidw,
351 Description: fmt.Sprintf("abidw %s", implementationLibrary),
352 Output: this.abiDumpPath,
353 Input: implementationLibrary,
354 Implicit: symbolList,
355 Args: map[string]string{
356 "symbolList": symbolList.String(),
357 },
358 })
359}
360
361func findNextApiLevel(ctx ModuleContext, apiLevel android.ApiLevel) *android.ApiLevel {
362 apiLevels := append(ctx.Config().AllSupportedApiLevels(),
363 android.FutureApiLevel)
364 for _, api := range apiLevels {
365 if api.GreaterThan(apiLevel) {
366 return &api
367 }
368 }
369 return nil
370}
371
372func (this *stubDecorator) diffAbi(ctx ModuleContext) {
373 missingPrebuiltError := fmt.Sprintf(
374 "Did not find prebuilt ABI dump for %q. Generate with "+
375 "//development/tools/ndk/update_ndk_abi.sh.", this.libraryName(ctx))
376
377 // Catch any ABI changes compared to the checked-in definition of this API
378 // level.
379 abiDiffPath := android.PathForModuleOut(ctx, "abidiff.timestamp")
380 prebuiltAbiDump := this.findPrebuiltAbiDump(ctx, this.apiLevel)
381 if !prebuiltAbiDump.Valid() {
382 ctx.Build(pctx, android.BuildParams{
383 Rule: android.ErrorRule,
384 Output: abiDiffPath,
385 Args: map[string]string{
386 "error": missingPrebuiltError,
387 },
388 })
389 } else {
390 ctx.Build(pctx, android.BuildParams{
391 Rule: abidiff,
392 Description: fmt.Sprintf("abidiff %s %s", prebuiltAbiDump,
393 this.abiDumpPath),
394 Output: abiDiffPath,
395 Inputs: android.Paths{prebuiltAbiDump.Path(), this.abiDumpPath},
396 })
397 }
398 this.abiDiffPaths = append(this.abiDiffPaths, abiDiffPath)
399
400 // Also ensure that the ABI of the next API level (if there is one) matches
401 // this API level. *New* ABI is allowed, but any changes to APIs that exist
402 // in this API level are disallowed.
403 if !this.apiLevel.IsCurrent() {
404 nextApiLevel := findNextApiLevel(ctx, this.apiLevel)
405 if nextApiLevel == nil {
406 panic(fmt.Errorf("could not determine which API level follows "+
407 "non-current API level %s", this.apiLevel))
408 }
409 nextAbiDiffPath := android.PathForModuleOut(ctx,
410 "abidiff_next.timestamp")
411 nextAbiDump := this.findPrebuiltAbiDump(ctx, *nextApiLevel)
412 if !nextAbiDump.Valid() {
413 ctx.Build(pctx, android.BuildParams{
414 Rule: android.ErrorRule,
415 Output: nextAbiDiffPath,
416 Args: map[string]string{
417 "error": missingPrebuiltError,
418 },
419 })
420 } else {
421 ctx.Build(pctx, android.BuildParams{
422 Rule: abidiff,
423 Description: fmt.Sprintf("abidiff %s %s", this.abiDumpPath,
424 nextAbiDump),
425 Output: nextAbiDiffPath,
426 Inputs: android.Paths{this.abiDumpPath, nextAbiDump.Path()},
427 Args: map[string]string{
428 "args": "--no-added-syms",
429 },
430 })
431 }
432 this.abiDiffPaths = append(this.abiDiffPaths, nextAbiDiffPath)
433 }
434}
435
Dan Willemsenb916b802017-03-19 13:44:32 -0700436func (c *stubDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Nan Zhang0007d812017-11-07 10:57:05 -0800437 if !strings.HasSuffix(String(c.properties.Symbol_file), ".map.txt") {
Dan Albert15be0c62017-06-13 15:14:56 -0700438 ctx.PropertyErrorf("symbol_file", "must end with .map.txt")
439 }
440
Colin Cross5ec407b2020-09-30 11:41:33 -0700441 if !c.buildStubs() {
442 // NDK libraries have no implementation variant, nothing to do
443 return Objects{}
444 }
445
Dan Albert1a246272020-07-06 14:49:35 -0700446 if !c.initializeProperties(ctx) {
447 // Emits its own errors, so we don't need to.
448 return Objects{}
449 }
450
sophiez58cabb72020-05-29 13:37:12 -0700451 symbolFile := String(c.properties.Symbol_file)
Dan Albertf1d14c72020-07-30 14:32:55 -0700452 nativeAbiResult := parseNativeAbiDefinition(ctx, symbolFile, c.apiLevel, "")
453 objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
454 c.versionScriptPath = nativeAbiResult.versionScript
455 if canDumpAbi(ctx.Module()) {
456 c.dumpAbi(ctx, nativeAbiResult.symbolList)
457 if canDiffAbi(ctx.Module()) {
458 c.diffAbi(ctx)
459 }
460 }
Dan Albert1a246272020-07-06 14:49:35 -0700461 if c.apiLevel.IsCurrent() && ctx.PrimaryArch() {
sophiez58cabb72020-05-29 13:37:12 -0700462 c.parsedCoverageXmlPath = parseSymbolFileForCoverage(ctx, symbolFile)
463 }
Dan Willemsenb916b802017-03-19 13:44:32 -0700464 return objs
Dan Albert914449f2016-06-17 16:45:24 -0700465}
466
Colin Cross37047f12016-12-13 17:06:13 -0800467func (linker *stubDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Dan Albert914449f2016-06-17 16:45:24 -0700468 return Deps{}
469}
470
Dan Willemsen01a90592017-04-07 15:21:13 -0700471func (linker *stubDecorator) Name(name string) string {
472 return name + ndkLibrarySuffix
473}
474
Colin Crossb916a382016-07-29 17:28:03 -0700475func (stub *stubDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Dan Willemsen01a90592017-04-07 15:21:13 -0700476 stub.libraryDecorator.libName = ctx.baseModuleName()
Colin Crossb916a382016-07-29 17:28:03 -0700477 return stub.libraryDecorator.linkerFlags(ctx, flags)
Dan Albert914449f2016-06-17 16:45:24 -0700478}
479
Colin Crossb916a382016-07-29 17:28:03 -0700480func (stub *stubDecorator) link(ctx ModuleContext, flags Flags, deps PathDeps,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700481 objs Objects) android.Path {
Dan Albert2bc91ba2016-07-28 17:40:28 -0700482
Colin Cross5ec407b2020-09-30 11:41:33 -0700483 if !stub.buildStubs() {
484 // NDK libraries have no implementation variant, nothing to do
485 return nil
486 }
487
Dan Albert1a246272020-07-06 14:49:35 -0700488 if shouldUseVersionScript(ctx, stub) {
Dan Albert98dbb3b2017-01-03 15:16:29 -0800489 linkerScriptFlag := "-Wl,--version-script," + stub.versionScriptPath.String()
Colin Cross4af21ed2019-11-04 09:37:55 -0800490 flags.Local.LdFlags = append(flags.Local.LdFlags, linkerScriptFlag)
Dan Willemsen939408a2019-06-10 18:02:25 -0700491 flags.LdFlagsDeps = append(flags.LdFlagsDeps, stub.versionScriptPath)
Dan Albert98dbb3b2017-01-03 15:16:29 -0800492 }
493
Colin Cross5ec407b2020-09-30 11:41:33 -0700494 stub.libraryDecorator.skipAPIDefine = true
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700495 return stub.libraryDecorator.link(ctx, flags, deps, objs)
Dan Albert2bc91ba2016-07-28 17:40:28 -0700496}
497
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -0700498func (stub *stubDecorator) nativeCoverage() bool {
499 return false
500}
501
Colin Crossb916a382016-07-29 17:28:03 -0700502func (stub *stubDecorator) install(ctx ModuleContext, path android.Path) {
Dan Albert914449f2016-06-17 16:45:24 -0700503 arch := ctx.Target().Arch.ArchType.Name
Dan Albert914449f2016-06-17 16:45:24 -0700504 // arm64 isn't actually a multilib toolchain, so unlike the other LP64
505 // architectures it's just installed to lib.
506 libDir := "lib"
507 if ctx.toolchain().Is64Bit() && arch != "arm64" {
508 libDir = "lib64"
509 }
510
511 installDir := getNdkInstallBase(ctx).Join(ctx, fmt.Sprintf(
Dan Albert1a246272020-07-06 14:49:35 -0700512 "platforms/android-%s/arch-%s/usr/%s", stub.apiLevel, arch, libDir))
Colin Cross0875c522017-11-28 17:34:01 -0800513 stub.installPath = ctx.InstallFile(installDir, path.Base(), path)
Dan Albert914449f2016-06-17 16:45:24 -0700514}
515
Colin Cross36242852017-06-23 15:06:31 -0700516func newStubLibrary() *Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800517 module, library := NewLibrary(android.DeviceSupported)
518 library.BuildOnlyShared()
Dan Albert914449f2016-06-17 16:45:24 -0700519 module.stl = nil
Colin Crossb916a382016-07-29 17:28:03 -0700520 module.sanitize = nil
ThiƩbaud Weksteend4587452020-08-19 14:53:01 +0200521 library.disableStripping()
Dan Albert914449f2016-06-17 16:45:24 -0700522
Colin Crossb916a382016-07-29 17:28:03 -0700523 stub := &stubDecorator{
524 libraryDecorator: library,
525 }
526 module.compiler = stub
527 module.linker = stub
528 module.installer = stub
Colin Cross31076b32020-10-23 17:22:06 -0700529 module.library = stub
Dan Albert914449f2016-06-17 16:45:24 -0700530
Colin Crossc511bc52020-04-07 16:50:32 +0000531 module.Properties.AlwaysSdk = true
532 module.Properties.Sdk_version = StringPtr("current")
533
Colin Cross36242852017-06-23 15:06:31 -0700534 module.AddProperties(&stub.properties, &library.MutatedProperties)
535
536 return module
Dan Albert914449f2016-06-17 16:45:24 -0700537}
538
Dan Albertf740ed02020-07-24 14:19:06 -0700539// ndk_library creates a library that exposes a stub implementation of functions
540// and variables for use at build time only.
Jooyung Hanb90e4912019-12-09 18:21:48 +0900541func NdkLibraryFactory() android.Module {
Colin Cross36242852017-06-23 15:06:31 -0700542 module := newStubLibrary()
543 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibBoth)
544 return module
Dan Albert914449f2016-06-17 16:45:24 -0700545}