blob: df775de3ea4248144562a2ee9e7324a1160d3802 [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 Albertad665932021-06-07 13:19:49 -070020 "runtime"
Dan Albert914449f2016-06-17 16:45:24 -070021 "strings"
Colin Crosse8a67a72016-08-07 21:17:54 -070022 "sync"
Dan Albert914449f2016-06-17 16:45:24 -070023
24 "github.com/google/blueprint"
Jiyong Parkee9b1172021-04-06 17:40:32 +090025 "github.com/google/blueprint/proptools"
Dan Albert914449f2016-06-17 16:45:24 -070026
27 "android/soong/android"
Spandan Das1278c2c2022-08-19 18:17:28 +000028 "android/soong/bazel"
Jingwen Chen341f7352022-01-11 05:42:49 +000029 "android/soong/cc/config"
Dan Albert914449f2016-06-17 16:45:24 -070030)
31
sophiez58cabb72020-05-29 13:37:12 -070032func init() {
Dan Albert06f58af2020-06-22 15:10:31 -070033 pctx.HostBinToolVariable("ndkStubGenerator", "ndkstubgen")
Matthias Maennichca8ae652023-06-30 21:52:17 +010034 pctx.HostBinToolVariable("stg", "stg")
Matthias Maennich658bb4d2023-06-30 23:28:15 +010035 pctx.HostBinToolVariable("stgdiff", "stgdiff")
sophiez58cabb72020-05-29 13:37:12 -070036}
37
Dan Albert914449f2016-06-17 16:45:24 -070038var (
Colin Cross9d45bb72016-08-29 16:14:13 -070039 genStubSrc = pctx.AndroidStaticRule("genStubSrc",
Dan Albert914449f2016-06-17 16:45:24 -070040 blueprint.RuleParams{
Dan Albert06f58af2020-06-22 15:10:31 -070041 Command: "$ndkStubGenerator --arch $arch --api $apiLevel " +
42 "--api-map $apiMap $flags $in $out",
43 CommandDeps: []string{"$ndkStubGenerator"},
Jiyong Park3fd0baf2018-12-07 16:25:39 +090044 }, "arch", "apiLevel", "apiMap", "flags")
Dan Albert914449f2016-06-17 16:45:24 -070045
Aleksei Vetrov262ed1a2023-08-23 10:06:35 +000046 // $headersList should include paths to public headers. All types
47 // that are defined outside of public headers will be excluded from
48 // ABI monitoring.
49 //
50 // STG tool doesn't access content of files listed in $headersList,
51 // so there is no need to add them to dependencies.
Matthias Maennich55486f82023-07-03 12:25:27 +010052 stg = pctx.AndroidStaticRule("stg",
53 blueprint.RuleParams{
Aleksei Vetrov262ed1a2023-08-23 10:06:35 +000054 Command: "$stg -S :$symbolList --file-filter :$headersList --elf $in -o $out",
Matthias Maennich55486f82023-07-03 12:25:27 +010055 CommandDeps: []string{"$stg"},
Aleksei Vetrov262ed1a2023-08-23 10:06:35 +000056 }, "symbolList", "headersList")
Matthias Maennich55486f82023-07-03 12:25:27 +010057
Matthias Maennich658bb4d2023-06-30 23:28:15 +010058 stgdiff = pctx.AndroidStaticRule("stgdiff",
Dan Albertf1d14c72020-07-30 14:32:55 -070059 blueprint.RuleParams{
60 // Need to create *some* output for ninja. We don't want to use tee
61 // because we don't want to spam the build output with "nothing
62 // changed" messages, so redirect output message to $out, and if
63 // changes were detected print the output and fail.
Dan Albert5b2fd582023-10-05 21:43:41 +000064 Command: "$stgdiff $args --stg $in -o $out || (cat $out && echo 'Run $$ANDROID_BUILD_TOP/development/tools/ndk/update_ndk_abi.sh to update the ABI dumps.' && false)",
Matthias Maennich658bb4d2023-06-30 23:28:15 +010065 CommandDeps: []string{"$stgdiff"},
Dan Albertf1d14c72020-07-30 14:32:55 -070066 }, "args")
67
Dan Albert914449f2016-06-17 16:45:24 -070068 ndkLibrarySuffix = ".ndk"
Colin Cross4d9c2d12016-07-29 12:48:20 -070069
Colin Cross95f1ca02020-10-29 20:47:22 -070070 ndkKnownLibsKey = android.NewOnceKey("ndkKnownLibsKey")
Dan Albertde5aade2020-06-30 12:32:51 -070071 // protects ndkKnownLibs writes during parallel BeginMutator.
72 ndkKnownLibsLock sync.Mutex
Dan Albertf1d14c72020-07-30 14:32:55 -070073
74 stubImplementation = dependencyTag{name: "stubImplementation"}
Dan Albert914449f2016-06-17 16:45:24 -070075)
76
Dan Albert1a246272020-07-06 14:49:35 -070077// The First_version and Unversioned_until properties of this struct should not
78// be used directly, but rather through the ApiLevel returning methods
79// firstVersion() and unversionedUntil().
80
Dan Albert914449f2016-06-17 16:45:24 -070081// Creates a stub shared library based on the provided version file.
82//
Dan Albert914449f2016-06-17 16:45:24 -070083// Example:
84//
Spandan Das73bcafc2022-08-18 23:26:00 +000085// ndk_library {
86//
87// name: "libfoo",
88// symbol_file: "libfoo.map.txt",
89// first_version: "9",
90//
91// }
Dan Albert914449f2016-06-17 16:45:24 -070092type libraryProperties struct {
93 // Relative path to the symbol map.
94 // An example file can be seen here: TODO(danalbert): Make an example.
Inseob Kim5eb7ee92022-04-27 10:30:34 +090095 Symbol_file *string `android:"path"`
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 Albert604086f2021-06-15 13:23:44 -0700106
Spandan Das73bcafc2022-08-18 23:26:00 +0000107 // Headers presented by this library to the Public API Surface
108 Export_header_libs []string
Dan Albert914449f2016-06-17 16:45:24 -0700109}
110
Colin Crossb916a382016-07-29 17:28:03 -0700111type stubDecorator struct {
112 *libraryDecorator
Dan Albert914449f2016-06-17 16:45:24 -0700113
114 properties libraryProperties
Dan Albert2bc91ba2016-07-28 17:40:28 -0700115
sophiez58cabb72020-05-29 13:37:12 -0700116 versionScriptPath android.ModuleGenPath
117 parsedCoverageXmlPath android.ModuleOutPath
118 installPath android.Path
Dan Albertf1d14c72020-07-30 14:32:55 -0700119 abiDumpPath android.OutputPath
120 abiDiffPaths android.Paths
Dan Albert1a246272020-07-06 14:49:35 -0700121
122 apiLevel android.ApiLevel
123 firstVersion android.ApiLevel
124 unversionedUntil android.ApiLevel
Dan Albert914449f2016-06-17 16:45:24 -0700125}
126
Colin Cross0477b422020-10-13 18:43:54 -0700127var _ versionedInterface = (*stubDecorator)(nil)
128
Dan Albert1a246272020-07-06 14:49:35 -0700129func shouldUseVersionScript(ctx BaseModuleContext, stub *stubDecorator) bool {
130 return stub.apiLevel.GreaterThanOrEqualTo(stub.unversionedUntil)
Dan Albert98dbb3b2017-01-03 15:16:29 -0800131}
132
Colin Cross0477b422020-10-13 18:43:54 -0700133func (stub *stubDecorator) implementationModuleName(name string) string {
134 return strings.TrimSuffix(name, ndkLibrarySuffix)
135}
136
Colin Cross3572cf72020-10-01 15:58:11 -0700137func ndkLibraryVersions(ctx android.BaseMutatorContext, from android.ApiLevel) []string {
Dan Albert1a246272020-07-06 14:49:35 -0700138 var versions []android.ApiLevel
139 versionStrs := []string{}
140 for _, version := range ctx.Config().AllSupportedApiLevels() {
141 if version.GreaterThanOrEqualTo(from) {
142 versions = append(versions, version)
143 versionStrs = append(versionStrs, version.String())
144 }
Dan Albert914449f2016-06-17 16:45:24 -0700145 }
Dan Albert0b176c82020-07-23 16:43:25 -0700146 versionStrs = append(versionStrs, android.FutureApiLevel.String())
Dan Albert914449f2016-06-17 16:45:24 -0700147
Colin Cross5ec407b2020-09-30 11:41:33 -0700148 return versionStrs
149}
150
Colin Cross3572cf72020-10-01 15:58:11 -0700151func (this *stubDecorator) stubsVersions(ctx android.BaseMutatorContext) []string {
152 if !ctx.Module().Enabled() {
153 return nil
154 }
Dan Albertf1d14c72020-07-30 14:32:55 -0700155 if ctx.Target().NativeBridge == android.NativeBridgeEnabled {
156 ctx.Module().Disable()
157 return nil
158 }
Colin Cross3572cf72020-10-01 15:58:11 -0700159 firstVersion, err := nativeApiLevelFromUser(ctx,
160 String(this.properties.First_version))
161 if err != nil {
162 ctx.PropertyErrorf("first_version", err.Error())
163 return nil
164 }
165 return ndkLibraryVersions(ctx, firstVersion)
166}
167
Dan Albert1a246272020-07-06 14:49:35 -0700168func (this *stubDecorator) initializeProperties(ctx BaseModuleContext) bool {
Colin Cross5ec407b2020-09-30 11:41:33 -0700169 this.apiLevel = nativeApiLevelOrPanic(ctx, this.stubsVersion())
Dan Albert1a246272020-07-06 14:49:35 -0700170
171 var err error
172 this.firstVersion, err = nativeApiLevelFromUser(ctx,
173 String(this.properties.First_version))
174 if err != nil {
175 ctx.PropertyErrorf("first_version", err.Error())
176 return false
177 }
178
Jiyong Parkee9b1172021-04-06 17:40:32 +0900179 str := proptools.StringDefault(this.properties.Unversioned_until, "minimum")
180 this.unversionedUntil, err = nativeApiLevelFromUser(ctx, str)
Dan Albert1a246272020-07-06 14:49:35 -0700181 if err != nil {
182 ctx.PropertyErrorf("unversioned_until", err.Error())
183 return false
184 }
185
186 return true
187}
188
Colin Cross95f1ca02020-10-29 20:47:22 -0700189func getNDKKnownLibs(config android.Config) *[]string {
190 return config.Once(ndkKnownLibsKey, func() interface{} {
191 return &[]string{}
192 }).(*[]string)
193}
194
Colin Crossb916a382016-07-29 17:28:03 -0700195func (c *stubDecorator) compilerInit(ctx BaseModuleContext) {
Dan Albert7e9d2952016-08-04 13:02:36 -0700196 c.baseCompiler.compilerInit(ctx)
197
Dan Willemsen01a90592017-04-07 15:21:13 -0700198 name := ctx.baseModuleName()
199 if strings.HasSuffix(name, ndkLibrarySuffix) {
200 ctx.PropertyErrorf("name", "Do not append %q manually, just use the base name", ndkLibrarySuffix)
201 }
202
Dan Albertde5aade2020-06-30 12:32:51 -0700203 ndkKnownLibsLock.Lock()
204 defer ndkKnownLibsLock.Unlock()
Colin Cross95f1ca02020-10-29 20:47:22 -0700205 ndkKnownLibs := getNDKKnownLibs(ctx.Config())
206 for _, lib := range *ndkKnownLibs {
Dan Albert7e9d2952016-08-04 13:02:36 -0700207 if lib == name {
208 return
209 }
210 }
Colin Cross95f1ca02020-10-29 20:47:22 -0700211 *ndkKnownLibs = append(*ndkKnownLibs, name)
Dan Albert7e9d2952016-08-04 13:02:36 -0700212}
213
Jingwen Chen341f7352022-01-11 05:42:49 +0000214var stubLibraryCompilerFlags = []string{
215 // We're knowingly doing some otherwise unsightly things with builtin
216 // functions here. We're just generating stub libraries, so ignore it.
217 "-Wno-incompatible-library-redeclaration",
218 "-Wno-incomplete-setjmp-declaration",
219 "-Wno-builtin-requires-header",
220 "-Wno-invalid-noreturn",
221 "-Wall",
222 "-Werror",
223 // These libraries aren't actually used. Don't worry about unwinding
224 // (avoids the need to link an unwinder into a fake library).
225 "-fno-unwind-tables",
226}
227
228func init() {
229 config.ExportStringList("StubLibraryCompilerFlags", stubLibraryCompilerFlags)
230}
231
George Burgess IVf5310e32017-07-19 11:39:53 -0700232func addStubLibraryCompilerFlags(flags Flags) Flags {
Jingwen Chen341f7352022-01-11 05:42:49 +0000233 flags.Global.CFlags = append(flags.Global.CFlags, stubLibraryCompilerFlags...)
Jiyong Park48d75ef2019-11-21 15:11:49 +0900234 // All symbols in the stubs library should be visible.
235 if inList("-fvisibility=hidden", flags.Local.CFlags) {
236 flags.Local.CFlags = append(flags.Local.CFlags, "-fvisibility=default")
237 }
George Burgess IVf5310e32017-07-19 11:39:53 -0700238 return flags
239}
240
Colin Crossf18e1102017-11-16 14:33:08 -0800241func (stub *stubDecorator) compilerFlags(ctx ModuleContext, flags Flags, deps PathDeps) Flags {
242 flags = stub.baseCompiler.compilerFlags(ctx, flags, deps)
George Burgess IVf5310e32017-07-19 11:39:53 -0700243 return addStubLibraryCompilerFlags(flags)
244}
245
Dan Albertf1d14c72020-07-30 14:32:55 -0700246type ndkApiOutputs struct {
247 stubSrc android.ModuleGenPath
248 versionScript android.ModuleGenPath
249 symbolList android.ModuleGenPath
250}
251
252func parseNativeAbiDefinition(ctx ModuleContext, symbolFile string,
253 apiLevel android.ApiLevel, genstubFlags string) ndkApiOutputs {
Dan Albert914449f2016-06-17 16:45:24 -0700254
Dan Willemsenb916b802017-03-19 13:44:32 -0700255 stubSrcPath := android.PathForModuleGen(ctx, "stub.c")
256 versionScriptPath := android.PathForModuleGen(ctx, "stub.map")
257 symbolFilePath := android.PathForModuleSrc(ctx, symbolFile)
Dan Albertf1d14c72020-07-30 14:32:55 -0700258 symbolListPath := android.PathForModuleGen(ctx, "abi_symbol_list.txt")
Dan Albert49927d22017-03-28 15:00:46 -0700259 apiLevelsJson := android.GetApiLevelsJson(ctx)
Colin Crossae887032017-10-23 17:16:14 -0700260 ctx.Build(pctx, android.BuildParams{
Colin Cross67a5c132017-05-09 13:45:28 -0700261 Rule: genStubSrc,
262 Description: "generate stubs " + symbolFilePath.Rel(),
Dan Albertf1d14c72020-07-30 14:32:55 -0700263 Outputs: []android.WritablePath{stubSrcPath, versionScriptPath,
264 symbolListPath},
265 Input: symbolFilePath,
266 Implicits: []android.Path{apiLevelsJson},
Dan Albert914449f2016-06-17 16:45:24 -0700267 Args: map[string]string{
Dan Albertf1d14c72020-07-30 14:32:55 -0700268 "arch": ctx.Arch().ArchType.String(),
269 "apiLevel": apiLevel.String(),
Dan Albert49927d22017-03-28 15:00:46 -0700270 "apiMap": apiLevelsJson.String(),
Jiyong Park3fd0baf2018-12-07 16:25:39 +0900271 "flags": genstubFlags,
Dan Albert914449f2016-06-17 16:45:24 -0700272 },
273 })
274
Dan Albertf1d14c72020-07-30 14:32:55 -0700275 return ndkApiOutputs{
276 stubSrc: stubSrcPath,
277 versionScript: versionScriptPath,
278 symbolList: symbolListPath,
279 }
280}
281
282func compileStubLibrary(ctx ModuleContext, flags Flags, src android.Path) Objects {
Mitch Phillips4e5f9a12022-04-29 13:12:28 -0700283 // libc/libm stubs libraries end up mismatching with clang's internal definition of these
284 // functions (which have noreturn attributes and other things). Because we just want to create a
285 // stub with symbol definitions, and types aren't important in C, ignore the mismatch.
286 flags.Local.ConlyFlags = append(flags.Local.ConlyFlags, "-fno-builtin")
Dan Albertf1d14c72020-07-30 14:32:55 -0700287 return compileObjs(ctx, flagsToBuilderFlags(flags), "",
Chih-Hung Hsieh9db8a0c2022-02-17 12:54:45 -0800288 android.Paths{src}, nil, nil, nil, nil)
Dan Willemsenb916b802017-03-19 13:44:32 -0700289}
290
Dan Albertf1d14c72020-07-30 14:32:55 -0700291func (this *stubDecorator) findImplementationLibrary(ctx ModuleContext) android.Path {
292 dep := ctx.GetDirectDepWithTag(strings.TrimSuffix(ctx.ModuleName(), ndkLibrarySuffix),
293 stubImplementation)
294 if dep == nil {
295 ctx.ModuleErrorf("Could not find implementation for stub")
296 return nil
297 }
298 impl, ok := dep.(*Module)
299 if !ok {
300 ctx.ModuleErrorf("Implementation for stub is not correct module type")
Alan Stokes73d32452022-11-01 14:05:08 +0000301 return nil
Dan Albertf1d14c72020-07-30 14:32:55 -0700302 }
303 output := impl.UnstrippedOutputFile()
304 if output == nil {
305 ctx.ModuleErrorf("implementation module (%s) has no output", impl)
306 return nil
307 }
308
309 return output
310}
311
312func (this *stubDecorator) libraryName(ctx ModuleContext) string {
313 return strings.TrimSuffix(ctx.ModuleName(), ndkLibrarySuffix)
314}
315
316func (this *stubDecorator) findPrebuiltAbiDump(ctx ModuleContext,
317 apiLevel android.ApiLevel) android.OptionalPath {
318
319 subpath := filepath.Join("prebuilts/abi-dumps/ndk", apiLevel.String(),
Matthias Maennichca8ae652023-06-30 21:52:17 +0100320 ctx.Arch().ArchType.String(), this.libraryName(ctx), "abi.stg")
Dan Albertf1d14c72020-07-30 14:32:55 -0700321 return android.ExistentPathForSource(ctx, subpath)
322}
323
324// Feature flag.
Dan Albertf71006a2022-04-14 23:08:51 +0000325func canDumpAbi(config android.Config) bool {
326 if runtime.GOOS == "darwin" {
327 return false
328 }
329 // abidw doesn't currently handle top-byte-ignore correctly. Disable ABI
330 // dumping for those configs while we wait for a fix. We'll still have ABI
331 // checking coverage from non-hwasan builds.
332 // http://b/190554910
333 if android.InList("hwaddress", config.SanitizeDevice()) {
334 return false
335 }
Dan Albert326ab242023-04-20 17:38:29 +0000336 // http://b/156513478
337 // http://b/277624006
338 // This step is expensive. We're not able to do anything with the outputs of
339 // this step yet (canDiffAbi is flagged off because libabigail isn't able to
340 // handle all our libraries), disable it. There's no sense in protecting
341 // against checking in code that breaks abidw since by the time any of this
342 // can be turned on we'll need to migrate to STG anyway.
343 return false
Dan Albertf1d14c72020-07-30 14:32:55 -0700344}
345
346// Feature flag to disable diffing against prebuilts.
Dan Albertad665932021-06-07 13:19:49 -0700347func canDiffAbi() bool {
Dan Albertf1d14c72020-07-30 14:32:55 -0700348 return false
349}
350
Matthias Maennich55486f82023-07-03 12:25:27 +0100351func (this *stubDecorator) dumpAbi(ctx ModuleContext, symbolList android.Path) {
352 implementationLibrary := this.findImplementationLibrary(ctx)
353 this.abiDumpPath = getNdkAbiDumpInstallBase(ctx).Join(ctx,
354 this.apiLevel.String(), ctx.Arch().ArchType.String(),
355 this.libraryName(ctx), "abi.stg")
Aleksei Vetrov262ed1a2023-08-23 10:06:35 +0000356 headersList := getNdkABIHeadersFile(ctx)
Matthias Maennich55486f82023-07-03 12:25:27 +0100357 ctx.Build(pctx, android.BuildParams{
358 Rule: stg,
359 Description: fmt.Sprintf("stg %s", implementationLibrary),
360 Input: implementationLibrary,
Aleksei Vetrov262ed1a2023-08-23 10:06:35 +0000361 Implicits: []android.Path{
362 symbolList,
363 headersList,
364 },
365 Output: this.abiDumpPath,
Matthias Maennich55486f82023-07-03 12:25:27 +0100366 Args: map[string]string{
Aleksei Vetrov262ed1a2023-08-23 10:06:35 +0000367 "symbolList": symbolList.String(),
368 "headersList": headersList.String(),
Matthias Maennich55486f82023-07-03 12:25:27 +0100369 },
370 })
371}
372
Dan Albertf1d14c72020-07-30 14:32:55 -0700373func findNextApiLevel(ctx ModuleContext, apiLevel android.ApiLevel) *android.ApiLevel {
374 apiLevels := append(ctx.Config().AllSupportedApiLevels(),
375 android.FutureApiLevel)
376 for _, api := range apiLevels {
377 if api.GreaterThan(apiLevel) {
378 return &api
379 }
380 }
381 return nil
382}
383
384func (this *stubDecorator) diffAbi(ctx ModuleContext) {
Dan Albertf1d14c72020-07-30 14:32:55 -0700385 // Catch any ABI changes compared to the checked-in definition of this API
386 // level.
Matthias Maennich658bb4d2023-06-30 23:28:15 +0100387 abiDiffPath := android.PathForModuleOut(ctx, "stgdiff.timestamp")
Dan Albertf1d14c72020-07-30 14:32:55 -0700388 prebuiltAbiDump := this.findPrebuiltAbiDump(ctx, this.apiLevel)
Aleksei Vetrov9c8c5ab2023-09-20 13:09:25 +0000389 missingPrebuiltErrorTemplate :=
390 "Did not find prebuilt ABI dump for %q (%q). Generate with " +
391 "//development/tools/ndk/update_ndk_abi.sh."
Dan Albertf7cb5632022-11-29 17:20:16 +0000392 missingPrebuiltError := fmt.Sprintf(
Aleksei Vetrov9c8c5ab2023-09-20 13:09:25 +0000393 missingPrebuiltErrorTemplate, this.libraryName(ctx),
Dan Albertf7cb5632022-11-29 17:20:16 +0000394 prebuiltAbiDump.InvalidReason())
Dan Albertf1d14c72020-07-30 14:32:55 -0700395 if !prebuiltAbiDump.Valid() {
396 ctx.Build(pctx, android.BuildParams{
397 Rule: android.ErrorRule,
398 Output: abiDiffPath,
399 Args: map[string]string{
400 "error": missingPrebuiltError,
401 },
402 })
403 } else {
404 ctx.Build(pctx, android.BuildParams{
Matthias Maennich658bb4d2023-06-30 23:28:15 +0100405 Rule: stgdiff,
406 Description: fmt.Sprintf("Comparing ABI %s %s", prebuiltAbiDump,
Dan Albertf1d14c72020-07-30 14:32:55 -0700407 this.abiDumpPath),
408 Output: abiDiffPath,
409 Inputs: android.Paths{prebuiltAbiDump.Path(), this.abiDumpPath},
Matthias Maennich658bb4d2023-06-30 23:28:15 +0100410 Args: map[string]string{
411 "args": "--format=small",
412 },
Dan Albertf1d14c72020-07-30 14:32:55 -0700413 })
414 }
415 this.abiDiffPaths = append(this.abiDiffPaths, abiDiffPath)
416
417 // Also ensure that the ABI of the next API level (if there is one) matches
418 // this API level. *New* ABI is allowed, but any changes to APIs that exist
419 // in this API level are disallowed.
Aleksei Vetrov8c02bbc2023-09-19 15:35:08 +0000420 if !this.apiLevel.IsCurrent() && prebuiltAbiDump.Valid() {
Dan Albertf1d14c72020-07-30 14:32:55 -0700421 nextApiLevel := findNextApiLevel(ctx, this.apiLevel)
422 if nextApiLevel == nil {
423 panic(fmt.Errorf("could not determine which API level follows "+
424 "non-current API level %s", this.apiLevel))
425 }
426 nextAbiDiffPath := android.PathForModuleOut(ctx,
427 "abidiff_next.timestamp")
428 nextAbiDump := this.findPrebuiltAbiDump(ctx, *nextApiLevel)
Aleksei Vetrov9c8c5ab2023-09-20 13:09:25 +0000429 missingNextPrebuiltError := fmt.Sprintf(
430 missingPrebuiltErrorTemplate, this.libraryName(ctx),
431 nextAbiDump.InvalidReason())
Dan Albertf1d14c72020-07-30 14:32:55 -0700432 if !nextAbiDump.Valid() {
433 ctx.Build(pctx, android.BuildParams{
434 Rule: android.ErrorRule,
435 Output: nextAbiDiffPath,
436 Args: map[string]string{
Aleksei Vetrov9c8c5ab2023-09-20 13:09:25 +0000437 "error": missingNextPrebuiltError,
Dan Albertf1d14c72020-07-30 14:32:55 -0700438 },
439 })
440 } else {
441 ctx.Build(pctx, android.BuildParams{
Matthias Maennich658bb4d2023-06-30 23:28:15 +0100442 Rule: stgdiff,
Aleksei Vetrov8c02bbc2023-09-19 15:35:08 +0000443 Description: fmt.Sprintf(
444 "Comparing ABI to the next API level %s %s",
445 prebuiltAbiDump, nextAbiDump),
Dan Albertf1d14c72020-07-30 14:32:55 -0700446 Output: nextAbiDiffPath,
Aleksei Vetrov8c02bbc2023-09-19 15:35:08 +0000447 Inputs: android.Paths{
448 prebuiltAbiDump.Path(), nextAbiDump.Path()},
Dan Albertf1d14c72020-07-30 14:32:55 -0700449 Args: map[string]string{
Matthias Maennich658bb4d2023-06-30 23:28:15 +0100450 "args": "--format=small --ignore=interface_addition",
Dan Albertf1d14c72020-07-30 14:32:55 -0700451 },
452 })
453 }
454 this.abiDiffPaths = append(this.abiDiffPaths, nextAbiDiffPath)
455 }
456}
457
Dan Willemsenb916b802017-03-19 13:44:32 -0700458func (c *stubDecorator) compile(ctx ModuleContext, flags Flags, deps PathDeps) Objects {
Nan Zhang0007d812017-11-07 10:57:05 -0800459 if !strings.HasSuffix(String(c.properties.Symbol_file), ".map.txt") {
Dan Albert15be0c62017-06-13 15:14:56 -0700460 ctx.PropertyErrorf("symbol_file", "must end with .map.txt")
461 }
462
Colin Cross5ec407b2020-09-30 11:41:33 -0700463 if !c.buildStubs() {
464 // NDK libraries have no implementation variant, nothing to do
465 return Objects{}
466 }
467
Dan Albert1a246272020-07-06 14:49:35 -0700468 if !c.initializeProperties(ctx) {
469 // Emits its own errors, so we don't need to.
470 return Objects{}
471 }
472
sophiez58cabb72020-05-29 13:37:12 -0700473 symbolFile := String(c.properties.Symbol_file)
Dan Albertf1d14c72020-07-30 14:32:55 -0700474 nativeAbiResult := parseNativeAbiDefinition(ctx, symbolFile, c.apiLevel, "")
475 objs := compileStubLibrary(ctx, flags, nativeAbiResult.stubSrc)
476 c.versionScriptPath = nativeAbiResult.versionScript
Dan Albertf71006a2022-04-14 23:08:51 +0000477 if canDumpAbi(ctx.Config()) {
Matthias Maenniche914f2d2023-07-03 12:29:27 +0100478 c.dumpAbi(ctx, nativeAbiResult.symbolList)
Dan Albertad665932021-06-07 13:19:49 -0700479 if canDiffAbi() {
Dan Albertf1d14c72020-07-30 14:32:55 -0700480 c.diffAbi(ctx)
481 }
482 }
Dan Albert1a246272020-07-06 14:49:35 -0700483 if c.apiLevel.IsCurrent() && ctx.PrimaryArch() {
sophiez4c4f8032021-08-16 22:54:00 -0700484 c.parsedCoverageXmlPath = parseSymbolFileForAPICoverage(ctx, symbolFile)
sophiez58cabb72020-05-29 13:37:12 -0700485 }
Dan Willemsenb916b802017-03-19 13:44:32 -0700486 return objs
Dan Albert914449f2016-06-17 16:45:24 -0700487}
488
Spandan Das73bcafc2022-08-18 23:26:00 +0000489// Add a dependency on the header modules of this ndk_library
Colin Cross37047f12016-12-13 17:06:13 -0800490func (linker *stubDecorator) linkerDeps(ctx DepsContext, deps Deps) Deps {
Spandan Das73bcafc2022-08-18 23:26:00 +0000491 return Deps{
492 HeaderLibs: linker.properties.Export_header_libs,
493 }
Dan Albert914449f2016-06-17 16:45:24 -0700494}
495
Dan Willemsen01a90592017-04-07 15:21:13 -0700496func (linker *stubDecorator) Name(name string) string {
497 return name + ndkLibrarySuffix
498}
499
Colin Crossb916a382016-07-29 17:28:03 -0700500func (stub *stubDecorator) linkerFlags(ctx ModuleContext, flags Flags) Flags {
Dan Willemsen01a90592017-04-07 15:21:13 -0700501 stub.libraryDecorator.libName = ctx.baseModuleName()
Colin Crossb916a382016-07-29 17:28:03 -0700502 return stub.libraryDecorator.linkerFlags(ctx, flags)
Dan Albert914449f2016-06-17 16:45:24 -0700503}
504
Colin Crossb916a382016-07-29 17:28:03 -0700505func (stub *stubDecorator) link(ctx ModuleContext, flags Flags, deps PathDeps,
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700506 objs Objects) android.Path {
Dan Albert2bc91ba2016-07-28 17:40:28 -0700507
Colin Cross5ec407b2020-09-30 11:41:33 -0700508 if !stub.buildStubs() {
509 // NDK libraries have no implementation variant, nothing to do
510 return nil
511 }
512
Dan Albert1a246272020-07-06 14:49:35 -0700513 if shouldUseVersionScript(ctx, stub) {
Dan Albert98dbb3b2017-01-03 15:16:29 -0800514 linkerScriptFlag := "-Wl,--version-script," + stub.versionScriptPath.String()
Colin Cross4af21ed2019-11-04 09:37:55 -0800515 flags.Local.LdFlags = append(flags.Local.LdFlags, linkerScriptFlag)
Dan Willemsen939408a2019-06-10 18:02:25 -0700516 flags.LdFlagsDeps = append(flags.LdFlagsDeps, stub.versionScriptPath)
Dan Albert98dbb3b2017-01-03 15:16:29 -0800517 }
518
Colin Cross5ec407b2020-09-30 11:41:33 -0700519 stub.libraryDecorator.skipAPIDefine = true
Dan Willemsen5cb580f2016-09-26 17:33:01 -0700520 return stub.libraryDecorator.link(ctx, flags, deps, objs)
Dan Albert2bc91ba2016-07-28 17:40:28 -0700521}
522
Pirama Arumuga Nainar65c95ff2019-03-25 10:21:31 -0700523func (stub *stubDecorator) nativeCoverage() bool {
524 return false
525}
526
Dan Albert4048bb02023-04-03 20:19:07 +0000527// Returns the install path for unversioned NDK libraries (currently only static
528// libraries).
529func getUnversionedLibraryInstallPath(ctx ModuleContext) android.InstallPath {
530 return getNdkSysrootBase(ctx).Join(ctx, "usr/lib", config.NDKTriple(ctx.toolchain()))
531}
Rebecca Chyung961cf1c2023-04-03 05:17:17 +0000532
Dan Albert4048bb02023-04-03 20:19:07 +0000533// Returns the install path for versioned NDK libraries. These are most often
534// stubs, but the same paths are used for CRT objects.
535func getVersionedLibraryInstallPath(ctx ModuleContext, apiLevel android.ApiLevel) android.InstallPath {
536 return getUnversionedLibraryInstallPath(ctx).Join(ctx, apiLevel.String())
537}
538
539func (stub *stubDecorator) install(ctx ModuleContext, path android.Path) {
540 installDir := getVersionedLibraryInstallPath(ctx, stub.apiLevel)
Colin Cross0875c522017-11-28 17:34:01 -0800541 stub.installPath = ctx.InstallFile(installDir, path.Base(), path)
Dan Albert914449f2016-06-17 16:45:24 -0700542}
543
Colin Cross36242852017-06-23 15:06:31 -0700544func newStubLibrary() *Module {
Colin Crossab3b7322016-12-09 14:46:15 -0800545 module, library := NewLibrary(android.DeviceSupported)
546 library.BuildOnlyShared()
Dan Albert914449f2016-06-17 16:45:24 -0700547 module.stl = nil
Colin Crossb916a382016-07-29 17:28:03 -0700548 module.sanitize = nil
ThiƩbaud Weksteend4587452020-08-19 14:53:01 +0200549 library.disableStripping()
Dan Albert914449f2016-06-17 16:45:24 -0700550
Colin Crossb916a382016-07-29 17:28:03 -0700551 stub := &stubDecorator{
552 libraryDecorator: library,
553 }
554 module.compiler = stub
555 module.linker = stub
556 module.installer = stub
Colin Cross31076b32020-10-23 17:22:06 -0700557 module.library = stub
Dan Albert914449f2016-06-17 16:45:24 -0700558
Colin Crossc511bc52020-04-07 16:50:32 +0000559 module.Properties.AlwaysSdk = true
560 module.Properties.Sdk_version = StringPtr("current")
561
Colin Cross36242852017-06-23 15:06:31 -0700562 module.AddProperties(&stub.properties, &library.MutatedProperties)
563
564 return module
Dan Albert914449f2016-06-17 16:45:24 -0700565}
566
Dan Albertf740ed02020-07-24 14:19:06 -0700567// ndk_library creates a library that exposes a stub implementation of functions
568// and variables for use at build time only.
Jooyung Hanb90e4912019-12-09 18:21:48 +0900569func NdkLibraryFactory() android.Module {
Colin Cross36242852017-06-23 15:06:31 -0700570 module := newStubLibrary()
571 android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibBoth)
Spandan Das1278c2c2022-08-19 18:17:28 +0000572 android.InitBazelModule(module)
Colin Cross36242852017-06-23 15:06:31 -0700573 return module
Dan Albert914449f2016-06-17 16:45:24 -0700574}
Spandan Das1278c2c2022-08-19 18:17:28 +0000575
576type bazelCcApiContributionAttributes struct {
577 Api bazel.LabelAttribute
578 Api_surfaces bazel.StringListAttribute
579 Hdrs bazel.LabelListAttribute
580 Library_name string
581}
582
Chris Parsons637458d2023-09-19 20:09:00 +0000583func ndkLibraryBp2build(ctx android.Bp2buildMutatorContext, c *Module) {
Spandan Das63acae92023-09-14 22:34:34 +0000584 ndk, _ := c.linker.(*stubDecorator)
585 props := bazel.BazelTargetModuleProperties{
586 Rule_class: "cc_stub_suite",
587 Bzl_load_location: "//build/bazel/rules/cc:cc_stub_library.bzl",
588 }
589 sourceLibraryName := strings.TrimSuffix(c.Name(), ".ndk")
590 fromApiLevel, err := android.ApiLevelFromUser(ctx, proptools.String(ndk.properties.First_version))
591 if err != nil {
592 ctx.PropertyErrorf("first_version", "error converting first_version %v", proptools.String(ndk.properties.First_version))
593 }
594 symbolFileLabel := android.BazelLabelForModuleSrcSingle(ctx, proptools.String(ndk.properties.Symbol_file))
595 attrs := &bazelCcStubSuiteAttributes{
596 // TODO - b/300504837 Add ndk headers
Spandan Das17a27f02023-10-06 21:35:21 +0000597 Symbol_file: proptools.StringPtr(symbolFileLabel.Label),
598 Soname: proptools.StringPtr(sourceLibraryName + ".so"),
599 Api_surface: proptools.StringPtr(android.PublicApi.String()),
600 Included_in_ndk: proptools.BoolPtr(true),
Spandan Das63acae92023-09-14 22:34:34 +0000601 }
602 if sourceLibrary, exists := ctx.ModuleFromName(sourceLibraryName); exists {
603 // the source library might not exist in minimal/unbuildable branches like kernel-build-tools.
604 // check for its existence
605 attrs.Source_library_label = proptools.StringPtr(c.GetBazelLabel(ctx, sourceLibrary))
606 }
607 if ctx.Config().RawPlatformSdkVersion() != nil {
608 // This is a hack to populate `versions` only on branches that set a platform_sdk_version
609 // This prevents errors on branches such as kernel-build-tools
610 // This hack is acceptable since we are not required to support NDK Bazel builds on those branches
611 attrs.Versions = bazel.MakeStringListAttribute(ndkLibraryVersions(ctx, fromApiLevel))
612 }
613
614 ctx.CreateBazelTargetModule(
615 props,
616 android.CommonAttributes{Name: c.Name() + "_stub_libs"},
617 attrs,
618 )
619}